/* $Id: VBoxSharedClipboardSvc.cpp 81843 2019-11-14 16:30:44Z vboxsync $ */ /** @file * Shared Clipboard Service - Host service entry points. */ /* * 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. */ /** @page pg_hostclip The Shared Clipboard Host Service * * The shared clipboard host service is the host half of the clibpoard proxying * between the host and the guest. The guest parts live in VBoxClient, VBoxTray * and VBoxService depending on the OS, with code shared between host and guest * under src/VBox/GuestHost/SharedClipboard/. * * The service is split into a platform-independent core and platform-specific * backends. The service defines two communication protocols - one to * communicate with the clipboard service running on the guest, and one to * communicate with the backend. These will be described in a very skeletal * fashion here. * * r=bird: The "two communication protocols" does not seems to be factual, there * is only one protocol, the first one mentioned. It cannot be backend * specific, because the guest/host protocol is platform and backend agnostic in * nature. You may call it versions, but I take a great dislike to "protocol * versions" here, as you've just extended the existing protocol with a feature * that allows to transfer files and directories too. See @bugref{9437#c39}. * * * @section sec_hostclip_guest_proto The guest communication protocol * * The guest clipboard service communicates with the host service over HGCM * (the host is a HGCM service). HGCM is connection based, so the guest side * has to connect before anything else can be done. (Windows hosts currently * only support one simultaneous connection.) Once it has connected, it can * send messages to the host services, some of which will receive immediate * replies from the host, others which will block till a reply becomes * available. The latter is because HGCM does't allow the host to initiate * communication, it must be guest triggered. The HGCM service is single * threaded, so it doesn't matter if the guest tries to send lots of requests in * parallel, the service will process them one at the time. * * There are currently four messages defined. The first is * VBOX_SHCL_GUEST_FN_MSG_GET / VBOX_SHCL_GUEST_FN_GET_HOST_MSG_OLD, which waits * for a message from the host. If a host message is sent while the guest is * not waiting, it will be queued until the guest requests it. The host code * only supports a single simultaneous GET call from one client guest. * * The second guest message is VBOX_SHCL_GUEST_FN_FORMATS_REPORT, which tells * the host that the guest has new clipboard data available. The third is * VBOX_SHCL_GUEST_FN_DATA_READ, which asks the host to send its clipboard data * and waits until it arrives. The host supports at most one simultaneous * VBOX_SHCL_GUEST_FN_DATA_READ call from a guest - if a second call is made * before the first has returned, the first will be aborted. * * The last guest message is VBOX_SHCL_GUEST_FN_DATA_WRITE, which is used to * send the contents of the guest clipboard to the host. This call should be * used after the host has requested data from the guest. * * * @section sec_hostclip_backend_proto The communication protocol with the * platform-specific backend * * The initial protocol implementation (called protocol v0) was very simple, * and could only handle simple data (like copied text and so on). It also * was limited to two (2) fixed parameters at all times. * * Since VBox 6.1 a newer protocol (v1) has been established to also support * file transfers. This protocol uses a (per-client) message queue instead * (see VBOX_SHCL_GUEST_FN_GET_HOST_MSG_OLD vs. VBOX_SHCL_GUEST_FN_GET_HOST_MSG). * * To distinguish the old (legacy) or new(er) protocol, the VBOX_SHCL_GUEST_FN_CONNECT * message has been introduced. If an older guest does not send this message, * an appropriate translation will be done to serve older Guest Additions (< 6.1). * * The protocol also support out-of-order messages by using so-called "context IDs", * which are generated by the host. A context ID consists of a so-called "source event ID" * and a so-called "event ID". Each HGCM client has an own, random, source event ID and * generates non-deterministic event IDs so that the guest side does not known what * comes next; the guest side has to reply with the same conext ID which was sent by * the host request. * * Also see the protocol changelog at VBoxShClSvc.h. * * * @section sec_uri_intro Transferring files * * Since VBox x.x.x transferring files via Shared Clipboard is supported. * See the VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS define for supported / enabled * platforms. This is called "Shared Clipboard transfers". * * Copying files / directories from guest A to guest B requires the host * service to act as a proxy and cache, as we don't allow direct VM-to-VM * communication. Copying from / to the host also is taken into account. * * At the moment a transfer is a all-or-nothing operation, e.g. it either * completes or fails completely. There might be callbacks in the future * to e.g. skip failing entries. * * Known limitations: * * - Support for VRDE (VRDP) is not implemented yet (see #9498). * - Unicode support on Windows hosts / guests is not enabled (yet). * - Symbolic links / Windows junctions are not allowed. * - Windows alternate data streams (ADS) are not allowed. * - No support for ACLs yet. * - No (maybe never) support for NT4. * * @section sec_transfers_areas Clipboard areas. * * For larger / longer transfers there might be file data * temporarily cached on the host, which has not been transferred to the * destination yet. Such a cache is called a "Shared Clipboard Area", which * in turn is identified by a unique ID across all VMs running on the same * host. To control the access (and needed cleanup) of such clipboard areas, * VBoxSVC (Main) is used for this task. A Shared Clipboard client can register, * unregister, attach to and detach from a clipboard area. If all references * to a clipboard area are released, a clipboard area gets detroyed automatically * by VBoxSVC. * * By default a clipboard area lives in the user's temporary directory in the * sub folder "VirtualBox Shared Clipboards/clipboard-". VBoxSVC does not * do any file locking in a clipboard area, but keeps the clipboard areas's * directory open to prevent deletion by third party processes. * * @todo We might use some VFS / container (IPRT?) for this instead of the * host's file system directly? * bird> Yes, but may take some work as we don't have the pick and choose * kind of VFS containers implemented yet. * * @section sec_transfer_structure Transfer handling structure * * All structures / classes are designed for running on both, on the guest * (via VBoxTray / VBoxClient) or on the host (host service) to avoid code * duplication where applicable. * * Per HGCM client there is a so-called "transfer context", which in turn can * have one or mulitple so-called "Shared Clipboard transfer" objects. At the * moment we only support on concurrent Shared Clipboard transfer per transfer * context. It's being used for reading from a source or writing to destination, * depening on its direction. An Shared Clipboard transfer can have optional * callbacks which might be needed by various implementations. Also, transfers * optionally can run in an asynchronous thread to prevent blocking the UI while * running. * * A Shared Clipboard transfer can maintain its own clipboard area; for the host * service such a clipboard area is coupled to a clipboard area registered or * attached with VBoxSVC. This is needed because multiple transfers from * multiple VMs (n:n) can rely on the same clipboard area, so there needs a * master keeping tracking of a clipboard area. To minimize IPC traffic only the * minimum de/attaching is done at the moment. A clipboard area gets cleaned up * (i.e. physically deleted) if no references are held to it anymore, or if * VBoxSVC goes down. * * @section sec_transfer_providers Transfer providers * * For certain implementations (for example on Windows guests / hosts, using * IDataObject and IStream objects) a more flexible approach reqarding reading / * writing is needed. For this so-called transfer providers abstract the way of how * data is being read / written in the current context (host / guest), while * the rest of the code stays the same. * * @section sec_transfer_protocol Transfer protocol * * The host service issues commands which the guest has to respond with an own * message to. The protocol itself is designed so that it has primitives to list * directories and open/close/read/write file system objects. * * Note that this is different from the DnD approach, as Shared Clipboard transfers * need to be deeper integrated within the host / guest OS (i.e. for progress UI), * and this might require non-monolithic / random access APIs to achieve. * * As there can be multiple file system objects (fs objects) selected for transfer, * a transfer can be queried for its root entries, which then contains the top-level * elements. Based on these elements, (a) (recursive) listing(s) can be performed * to (partially) walk down into directories and query fs object information. The * provider provides appropriate interface for this, even if not all implementations * might need this mechanism. * * An Shared Clipboard transfer has three stages: * - 1. Announcement: An Shared Clipboard transfer-compatible format (currently only one format available) * has been announced, the destination side creates a transfer object, which then, * depending on the actual implementation, can be used to tell the OS that * there is transfer (file) data available. * At this point this just acts as a (kind-of) promise to the OS that we * can provide (file) data at some later point in time. * * - 2. Initialization: As soon as the OS requests the (file) data, mostly triggered * by the user starting a paste operation (CTRL + V), the transfer get initialized * on the destination side, which in turn lets the source know that a transfer * is going to happen. * * - 3. Transfer: At this stage the actual transfer from source to the destination takes * place. How the actual transfer is structurized (e.g. which files / directories * are transferred in which order) depends on the destination implementation. This * is necessary in order to fulfill requirements on the destination side with * regards to ETA calculation or other dependencies. * Both sides can abort or cancel the transfer at any time. */ /********************************************************************************************************************************* * Header Files * *********************************************************************************************************************************/ #define LOG_GROUP LOG_GROUP_SHARED_CLIPBOARD #include #include #include #include #include #include #include #include #include #include #include #include #include #include "VBoxSharedClipboardSvc-internal.h" #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS # include "VBoxSharedClipboardSvc-transfers.h" #endif using namespace HGCM; /********************************************************************************************************************************* * Global Variables * *********************************************************************************************************************************/ PVBOXHGCMSVCHELPERS g_pHelpers; static RTCRITSECT g_CritSect; /** Global Shared Clipboard mode. */ static uint32_t g_uMode = VBOX_SHCL_MODE_OFF; #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS /** Global Shared Clipboard (file) transfer mode. */ uint32_t g_fTransferMode = VBOX_SHCL_TRANSFER_MODE_DISABLED; #endif /** Is the clipboard running in headless mode? */ static bool g_fHeadless = false; /** Holds the service extension state. */ SHCLEXTSTATE g_ExtState = { 0 }; /** Global map of all connected clients. */ ClipboardClientMap g_mapClients; /** Global list of all clients which are queued up (deferred return) and ready * to process new commands. The key is the (unique) client ID. */ ClipboardClientQueue g_listClientsDeferred; /** Host feature mask for VBOX_SHCL_GUEST_FN_REPORT_FEATURES/VBOX_SHCL_GUEST_FN_QUERY_FEATURES. */ static uint64_t const g_fHostFeatures0 = VBOX_SHCL_HF_NONE; /** * Returns the current Shared Clipboard service mode. * * @returns Current Shared Clipboard service mode. */ uint32_t ShClSvcGetMode(void) { return g_uMode; } /** * Getter for headless setting. Also needed by testcase. * * @returns Whether service currently running in headless mode or not. */ bool ShClSvcGetHeadless(void) { return g_fHeadless; } static int shClSvcModeSet(uint32_t uMode) { int rc = VERR_NOT_SUPPORTED; switch (uMode) { case VBOX_SHCL_MODE_OFF: RT_FALL_THROUGH(); case VBOX_SHCL_MODE_HOST_TO_GUEST: RT_FALL_THROUGH(); case VBOX_SHCL_MODE_GUEST_TO_HOST: RT_FALL_THROUGH(); case VBOX_SHCL_MODE_BIDIRECTIONAL: { g_uMode = uMode; rc = VINF_SUCCESS; break; } default: { g_uMode = VBOX_SHCL_MODE_OFF; break; } } LogFlowFuncLeaveRC(rc); return rc; } bool ShClSvcLock(void) { return RT_SUCCESS(RTCritSectEnter(&g_CritSect)); } void ShClSvcUnlock(void) { int rc2 = RTCritSectLeave(&g_CritSect); AssertRC(rc2); } /** * Resets a client's state message queue. * * @param pClient Pointer to the client data structure to reset message queue for. */ void shClSvcMsgQueueReset(PSHCLCLIENT pClient) { LogFlowFuncEnter(); while (!pClient->queueMsg.isEmpty()) { RTMemFree(pClient->queueMsg.last()); pClient->queueMsg.removeLast(); } } /** * Allocates a new clipboard message. * * @returns Allocated clipboard message, or NULL on failure. * @param uMsg Message type of message to allocate. * @param cParms Number of HGCM parameters to allocate. */ PSHCLCLIENTMSG shClSvcMsgAlloc(uint32_t uMsg, uint32_t cParms) { PSHCLCLIENTMSG pMsg = (PSHCLCLIENTMSG)RTMemAlloc(sizeof(SHCLCLIENTMSG)); if (pMsg) { pMsg->paParms = (PVBOXHGCMSVCPARM)RTMemAllocZ(sizeof(VBOXHGCMSVCPARM) * cParms); if (pMsg->paParms) { pMsg->cParms = cParms; pMsg->uMsg = uMsg; return pMsg; } } RTMemFree(pMsg); return NULL; } /** * Frees a formerly allocated clipboard message. * * @param pMsg Clipboard message to free. * The pointer will be invalid after calling this function. */ void shClSvcMsgFree(PSHCLCLIENTMSG pMsg) { if (!pMsg) return; if (pMsg->paParms) RTMemFree(pMsg->paParms); RTMemFree(pMsg); pMsg = NULL; } /** * Sets the VBOX_SHCL_GUEST_FN_MSG_PEEK_WAIT and VBOX_SHCL_GUEST_FN_MSG_PEEK_NOWAIT * return parameters. * * @param pMsg Message to set return parameters to. * @param paDstParms The peek parameter vector. * @param cDstParms The number of peek parameters (at least two). * @remarks ASSUMES the parameters has been cleared by clientMsgPeek. */ void shClSvcMsgSetPeekReturn(PSHCLCLIENTMSG pMsg, PVBOXHGCMSVCPARM paDstParms, uint32_t cDstParms) { Assert(cDstParms >= 2); if (paDstParms[0].type == VBOX_HGCM_SVC_PARM_32BIT) paDstParms[0].u.uint32 = pMsg->uMsg; else paDstParms[0].u.uint64 = pMsg->uMsg; paDstParms[1].u.uint32 = pMsg->cParms; uint32_t i = RT_MIN(cDstParms, pMsg->cParms + 2); while (i-- > 2) switch (pMsg->paParms[i - 2].type) { case VBOX_HGCM_SVC_PARM_32BIT: paDstParms[i].u.uint32 = ~(uint32_t)sizeof(uint32_t); break; case VBOX_HGCM_SVC_PARM_64BIT: paDstParms[i].u.uint32 = ~(uint32_t)sizeof(uint64_t); break; case VBOX_HGCM_SVC_PARM_PTR: paDstParms[i].u.uint32 = pMsg->paParms[i - 2].u.pointer.size; break; } } /** * Sets the VBOX_SHCL_GUEST_FN_GET_HOST_MSG_OLD return parameters. * * This function does the necessary translation between the legacy protocol (<= VBox 6.0) and the new protocols (>= VBox 6.1), * as messages are always stored as >= v1 messages in the message queue. * * @returns VBox status code. * @param pMsg Message to set return parameters to. * @param paDstParms The peek parameter vector. * @param cDstParms The number of peek parameters (at least two). * @param pfRemove Returns whether the message can be removed from the queue or not. * This is needed for certain messages which need to stay around for more than one (guest) call. */ int shClSvcMsgSetGetHostMsgOldReturn(PSHCLCLIENTMSG pMsg, PVBOXHGCMSVCPARM paDstParms, uint32_t cDstParms, bool *pfRemove) { AssertPtrReturn(pMsg, VERR_INVALID_POINTER); AssertPtrReturn(paDstParms, VERR_INVALID_POINTER); AssertReturn (cDstParms >= 2, VERR_INVALID_PARAMETER); AssertPtrReturn(pfRemove, VERR_INVALID_POINTER); int rc = VINF_SUCCESS; bool fRemove = true; LogFlowFunc(("uMsg=%RU32 (%s), cParms=%RU32\n", pMsg->uMsg, ShClHostMsgToStr(pMsg->uMsg), pMsg->cParms)); switch (pMsg->uMsg) { case VBOX_SHCL_HOST_MSG_QUIT: { HGCMSvcSetU32(&paDstParms[0], VBOX_SHCL_HOST_MSG_QUIT); HGCMSvcSetU32(&paDstParms[1], 0 /* Not used */); break; } case VBOX_SHCL_HOST_MSG_READ_DATA: { HGCMSvcSetU32(&paDstParms[0], VBOX_SHCL_HOST_MSG_READ_DATA); AssertBreakStmt(pMsg->cParms >= 2, rc = VERR_INVALID_PARAMETER); /* Paranoia. */ uint32_t uSrcFmt = VBOX_SHCL_FMT_NONE; uint32_t uDstFmt = VBOX_SHCL_FMT_NONE; rc = HGCMSvcGetU32(&pMsg->paParms[2] /* uFormat */, &uSrcFmt); if (RT_SUCCESS(rc)) { if (uSrcFmt & VBOX_SHCL_FMT_UNICODETEXT) uDstFmt = VBOX_SHCL_FMT_UNICODETEXT; else if (uSrcFmt & VBOX_SHCL_FMT_BITMAP) uDstFmt = VBOX_SHCL_FMT_BITMAP; else if (uSrcFmt & VBOX_SHCL_FMT_HTML) uDstFmt = VBOX_SHCL_FMT_HTML; else AssertStmt(uSrcFmt == VBOX_SHCL_FMT_NONE, uSrcFmt = VBOX_SHCL_FMT_NONE); LogFlowFunc(("uSrcFmt=0x%x, uDstFmt=0x%x\n", uSrcFmt, uDstFmt)); /* Remove format we're going to return from the queued message. */ uSrcFmt &= ~uDstFmt; HGCMSvcSetU32(&pMsg->paParms[2], uSrcFmt); /* Only report back one format at a time. */ HGCMSvcSetU32(&paDstParms[1], uDstFmt); /* If no more formats are left, the message can be removed finally. */ fRemove = uSrcFmt == VBOX_SHCL_FMT_NONE; LogFlowFunc(("uSrcFmt=0x%x, fRemove=%RTbool\n", uSrcFmt, fRemove)); } break; } case VBOX_SHCL_HOST_MSG_FORMATS_REPORT: { HGCMSvcSetU32(&paDstParms[0], VBOX_SHCL_HOST_MSG_FORMATS_REPORT); AssertBreakStmt(pMsg->cParms >= 2, rc = VERR_INVALID_PARAMETER); /* Paranoia. */ uint32_t uFmts; rc = HGCMSvcGetU32(&pMsg->paParms[1] /* uFormats */, &uFmts); if (RT_SUCCESS(rc)) HGCMSvcSetU32(&paDstParms[1], uFmts); break; } default: AssertFailed(); /* Not supported by legacy protocol. */ rc = VERR_NOT_SUPPORTED; break; } *pfRemove = fRemove; LogFlowFuncLeaveRC(rc); return rc; } /** * Adds a new message to a client'S message queue. * * @returns IPRT status code. * @param pClient Pointer to the client data structure to add new message to. * @param pMsg Pointer to message to add. The queue then owns the pointer. * @param fAppend Whether to append or prepend the message to the queue. */ int shClSvcMsgAdd(PSHCLCLIENT pClient, PSHCLCLIENTMSG pMsg, bool fAppend) { AssertPtrReturn(pMsg, VERR_INVALID_POINTER); LogFlowFunc(("uMsg=%RU32 (%s), cParms=%RU32, fAppend=%RTbool\n", pMsg->uMsg, ShClHostMsgToStr(pMsg->uMsg), pMsg->cParms, fAppend)); if (fAppend) pClient->queueMsg.append(pMsg); else pClient->queueMsg.prepend(pMsg); /** @todo Catch / handle OOM? */ return VINF_SUCCESS; } /** * Initializes a Shared Clipboard client. * * @param pClient Client to initialize. * @param uClientID HGCM client ID to assign client to. */ int shClSvcClientInit(PSHCLCLIENT pClient, uint32_t uClientID) { AssertPtrReturn(pClient, VERR_INVALID_POINTER); /* Assign the client ID. */ pClient->State.uClientID = uClientID; LogFlowFunc(("[Client %RU32]\n", pClient->State.uClientID)); int rc = RTCritSectInit(&pClient->CritSect); if (RT_SUCCESS(rc)) { /* Create the client's own event source. */ rc = ShClEventSourceCreate(&pClient->Events, 0 /* ID, ignored */); if (RT_SUCCESS(rc)) { LogFlowFunc(("[Client %RU32] Using event source %RU32\n", uClientID, pClient->Events.uID)); /* Reset the client state. */ shclSvcClientStateReset(&pClient->State); /* (Re-)initialize the client state. */ rc = shClSvcClientStateInit(&pClient->State, uClientID); #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS if (RT_SUCCESS(rc)) rc = ShClTransferCtxInit(&pClient->TransferCtx); #endif } } LogFlowFuncLeaveRC(rc); return rc; } /** * Destroys a Shared Clipboard client. * * @param pClient Client to destroy. */ void shClSvcClientDestroy(PSHCLCLIENT pClient) { AssertPtrReturnVoid(pClient); LogFlowFunc(("[Client %RU32]\n", pClient->State.uClientID)); /* Make sure to send a quit message to the guest so that it can terminate gracefully. */ if (pClient->Pending.uType) { if (pClient->Pending.cParms >= 2) { HGCMSvcSetU32(&pClient->Pending.paParms[0], VBOX_SHCL_HOST_MSG_QUIT); HGCMSvcSetU32(&pClient->Pending.paParms[1], 0); } g_pHelpers->pfnCallComplete(pClient->Pending.hHandle, VINF_SUCCESS); pClient->Pending.uType = 0; } ShClEventSourceDestroy(&pClient->Events); shClSvcClientStateDestroy(&pClient->State); int rc2 = RTCritSectDelete(&pClient->CritSect); AssertRC(rc2); ClipboardClientMap::iterator itClient = g_mapClients.find(pClient->State.uClientID); if (itClient != g_mapClients.end()) { g_mapClients.erase(itClient); } else AssertFailed(); LogFlowFuncLeave(); } /** * Resets a Shared Clipboard client. * * @param pClient Client to reset. */ void shClSvcClientReset(PSHCLCLIENT pClient) { if (!pClient) return; LogFlowFunc(("[Client %RU32]\n", pClient->State.uClientID)); /* Reset message queue. */ shClSvcMsgQueueReset(pClient); /* Reset event source. */ ShClEventSourceReset(&pClient->Events); /* Reset pending state. */ RT_ZERO(pClient->Pending); #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS shClSvcClientTransfersReset(pClient); #endif shclSvcClientStateReset(&pClient->State); } /** * Implements VBOX_SHCL_GUEST_FN_REPORT_FEATURES. * * @returns VBox status code. * @retval VINF_HGCM_ASYNC_EXECUTE on success (we complete the message here). * @retval VERR_ACCESS_DENIED if not master * @retval VERR_INVALID_PARAMETER if bit 63 in the 2nd parameter isn't set. * @retval VERR_WRONG_PARAMETER_COUNT * * @param pClient The client state. * @param hCall The client's call handle. * @param cParms Number of parameters. * @param paParms Array of parameters. */ int shClSvcClientReportFeatures(PSHCLCLIENT pClient, VBOXHGCMCALLHANDLE hCall, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) { /* * Validate the request. */ ASSERT_GUEST_RETURN(cParms == 2, VERR_WRONG_PARAMETER_COUNT); ASSERT_GUEST_RETURN(paParms[0].type == VBOX_HGCM_SVC_PARM_64BIT, VERR_WRONG_PARAMETER_TYPE); uint64_t const fFeatures0 = paParms[0].u.uint64; ASSERT_GUEST_RETURN(paParms[1].type == VBOX_HGCM_SVC_PARM_64BIT, VERR_WRONG_PARAMETER_TYPE); uint64_t const fFeatures1 = paParms[1].u.uint64; ASSERT_GUEST_RETURN(fFeatures1 & VBOX_SHCL_GF_1_MUST_BE_ONE, VERR_INVALID_PARAMETER); /* * Do the work. */ paParms[0].u.uint64 = g_fHostFeatures0; paParms[1].u.uint64 = 0; int rc = g_pHelpers->pfnCallComplete(hCall, VINF_SUCCESS); if (RT_SUCCESS(rc)) { pClient->State.fGuestFeatures0 = fFeatures0; pClient->State.fGuestFeatures1 = fFeatures1; Log(("[Client %RU32] features: %#RX64 %#RX64\n", pClient->State.uClientID, fFeatures0, fFeatures1)); /* * Forward the info to Main. */ /** @todo Not needed yet. */ } else LogFunc(("pfnCallComplete -> %Rrc\n", rc)); return VINF_HGCM_ASYNC_EXECUTE; } /** * Implements VBOX_SHCL_GUEST_FN_QUERY_FEATURES. * * @returns VBox status code. * @retval VINF_HGCM_ASYNC_EXECUTE on success (we complete the message here). * @retval VERR_WRONG_PARAMETER_COUNT * * @param hCall The client's call handle. * @param cParms Number of parameters. * @param paParms Array of parameters. */ int shClSvcClientQueryFeatures(VBOXHGCMCALLHANDLE hCall, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) { /* * Validate the request. */ ASSERT_GUEST_RETURN(cParms == 2, VERR_WRONG_PARAMETER_COUNT); ASSERT_GUEST_RETURN(paParms[0].type == VBOX_HGCM_SVC_PARM_64BIT, VERR_WRONG_PARAMETER_TYPE); ASSERT_GUEST_RETURN(paParms[1].type == VBOX_HGCM_SVC_PARM_64BIT, VERR_WRONG_PARAMETER_TYPE); ASSERT_GUEST(paParms[1].u.uint64 & RT_BIT_64(63)); /* * Do the work. */ paParms[0].u.uint64 = g_fHostFeatures0; paParms[1].u.uint64 = 0; int rc = g_pHelpers->pfnCallComplete(hCall, VINF_SUCCESS); if (RT_FAILURE(rc)) LogFunc(("pfnCallComplete -> %Rrc\n", rc)); return VINF_HGCM_ASYNC_EXECUTE; } /** * Implements VBOX_SHCL_GUEST_FN_MSG_PEEK_WAIT and VBOX_SHCL_GUEST_FN_MSG_PEEK_NOWAIT. * * @returns VBox status code. * @retval VINF_SUCCESS if a message was pending and is being returned. * @retval VERR_TRY_AGAIN if no message pending and not blocking. * @retval VERR_RESOURCE_BUSY if another read already made a waiting call. * @retval VINF_HGCM_ASYNC_EXECUTE if message wait is pending. * * @param pClient The client state. * @param hCall The client's call handle. * @param cParms Number of parameters. * @param paParms Array of parameters. * @param fWait Set if we should wait for a message, clear if to return * immediately. */ int shClSvcMsgPeek(PSHCLCLIENT pClient, VBOXHGCMCALLHANDLE hCall, uint32_t cParms, VBOXHGCMSVCPARM paParms[], bool fWait) { /* * Validate the request. */ ASSERT_GUEST_MSG_RETURN(cParms >= 2, ("cParms=%u!\n", cParms), VERR_WRONG_PARAMETER_COUNT); uint64_t idRestoreCheck = 0; uint32_t i = 0; if (paParms[i].type == VBOX_HGCM_SVC_PARM_64BIT) { idRestoreCheck = paParms[0].u.uint64; paParms[0].u.uint64 = 0; i++; } for (; i < cParms; i++) { ASSERT_GUEST_MSG_RETURN(paParms[i].type == VBOX_HGCM_SVC_PARM_32BIT, ("#%u type=%u\n", i, paParms[i].type), VERR_WRONG_PARAMETER_TYPE); paParms[i].u.uint32 = 0; } /* * Check restore session ID. */ if (idRestoreCheck != 0) { uint64_t idRestore = g_pHelpers->pfnGetVMMDevSessionId(g_pHelpers); if (idRestoreCheck != idRestore) { paParms[0].u.uint64 = idRestore; LogFlowFunc(("[Client %RU32] VBOX_SHCL_GUEST_FN_MSG_PEEK_XXX -> VERR_VM_RESTORED (%#RX64 -> %#RX64)\n", pClient->State.uClientID, idRestoreCheck, idRestore)); return VERR_VM_RESTORED; } Assert(!g_pHelpers->pfnIsCallRestored(hCall)); } /* * Return information about the first message if one is pending in the list. */ if (!pClient->queueMsg.isEmpty()) { PSHCLCLIENTMSG pFirstMsg = pClient->queueMsg.first(); if (pFirstMsg) { shClSvcMsgSetPeekReturn(pFirstMsg, paParms, cParms); LogFlowFunc(("[Client %RU32] VBOX_SHCL_GUEST_FN_MSG_PEEK_XXX -> VINF_SUCCESS (idMsg=%u (%s), cParms=%u)\n", pClient->State.uClientID, pFirstMsg->uMsg, ShClHostMsgToStr(pFirstMsg->uMsg), pFirstMsg->cParms)); return VINF_SUCCESS; } } /* * If we cannot wait, fail the call. */ if (!fWait) { LogFlowFunc(("[Client %RU32] GUEST_MSG_PEEK_NOWAIT -> VERR_TRY_AGAIN\n", pClient->State.uClientID)); return VERR_TRY_AGAIN; } /* * Wait for the host to queue a message for this client. */ ASSERT_GUEST_MSG_RETURN(pClient->Pending.uType == 0, ("Already pending! (idClient=%RU32)\n", pClient->State.uClientID), VERR_RESOURCE_BUSY); pClient->Pending.hHandle = hCall; pClient->Pending.cParms = cParms; pClient->Pending.paParms = paParms; pClient->Pending.uType = VBOX_SHCL_GUEST_FN_MSG_PEEK_WAIT; LogFlowFunc(("[Client %RU32] Is now in pending mode...\n", pClient->State.uClientID)); return VINF_HGCM_ASYNC_EXECUTE; } /** * Implements VBOX_SHCL_GUEST_FN_GET_HOST_MSG_OLD. * * @returns VBox status code. * @retval VINF_SUCCESS if a message was pending and is being returned. * @retval VINF_HGCM_ASYNC_EXECUTE if message wait is pending. * * @param pClient The client state. * @param hCall The client's call handle. * @param cParms Number of parameters. * @param paParms Array of parameters. */ int shClSvcMsgGetOld(PSHCLCLIENT pClient, VBOXHGCMCALLHANDLE hCall, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) { int rc; if (cParms != VBOX_SHCL_CPARMS_GET_HOST_MSG_OLD) { rc = VERR_INVALID_PARAMETER; } else if ( paParms[0].type != VBOX_HGCM_SVC_PARM_32BIT /* msg */ || paParms[1].type != VBOX_HGCM_SVC_PARM_32BIT) /* formats */ { rc = VERR_INVALID_PARAMETER; } else { if (!pClient->queueMsg.isEmpty()) { PSHCLCLIENTMSG pFirstMsg = pClient->queueMsg.first(); AssertPtr(pFirstMsg); LogFlowFunc(("[Client %RU32] uMsg=%RU32 (%s), cParms=%RU32\n", pClient->State.uClientID, pFirstMsg->uMsg, ShClHostMsgToStr(pFirstMsg->uMsg), pFirstMsg->cParms)); bool fRemove; rc = shClSvcMsgSetGetHostMsgOldReturn(pFirstMsg, paParms, cParms, &fRemove); if (RT_SUCCESS(rc)) { AssertPtr(g_pHelpers); rc = g_pHelpers->pfnCallComplete(hCall, rc); if ( rc != VERR_CANCELLED && fRemove) { pClient->queueMsg.removeFirst(); shClSvcMsgFree(pFirstMsg); rc = VINF_HGCM_ASYNC_EXECUTE; /* The caller must not complete it. */ } } } else { ASSERT_GUEST_MSG_RETURN(pClient->Pending.uType == 0, ("Already pending! (idClient=%RU32)\n", pClient->State.uClientID), VERR_RESOURCE_BUSY); pClient->Pending.hHandle = hCall; pClient->Pending.cParms = cParms; pClient->Pending.paParms = paParms; pClient->Pending.uType = VBOX_SHCL_GUEST_FN_GET_HOST_MSG_OLD; rc = VINF_HGCM_ASYNC_EXECUTE; /* The caller must not complete it. */ LogFlowFunc(("[Client %RU32] Is now in pending mode...\n", pClient->State.uClientID)); } } LogFlowFunc(("[Client %RU32] rc=%Rrc\n", pClient->State.uClientID, rc)); return rc; } /** * Implements VBOX_SHCL_GUEST_FN_MSG_GET. * * @returns VBox status code. * @retval VINF_SUCCESS if message retrieved and removed from the pending queue. * @retval VERR_TRY_AGAIN if no message pending. * @retval VERR_BUFFER_OVERFLOW if a parmeter buffer is too small. The buffer * size was updated to reflect the required size, though this isn't yet * forwarded to the guest. (The guest is better of using peek with * parameter count + 2 parameters to get the sizes.) * @retval VERR_MISMATCH if the incoming message ID does not match the pending. * @retval VINF_HGCM_ASYNC_EXECUTE if message was completed already. * * @param pClient The client state. * @param hCall The client's call handle. * @param cParms Number of parameters. * @param paParms Array of parameters. */ int shClSvcMsgGet(PSHCLCLIENT pClient, VBOXHGCMCALLHANDLE hCall, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) { /* * Validate the request. */ uint32_t const idMsgExpected = cParms > 0 && paParms[0].type == VBOX_HGCM_SVC_PARM_32BIT ? paParms[0].u.uint32 : cParms > 0 && paParms[0].type == VBOX_HGCM_SVC_PARM_64BIT ? paParms[0].u.uint64 : UINT32_MAX; /* * Return information about the first message if one is pending in the list. */ if (!pClient->queueMsg.isEmpty()) { PSHCLCLIENTMSG pFirstMsg = pClient->queueMsg.first(); if (pFirstMsg) { LogFlowFunc(("First message is: %RU32 (%s), cParms=%RU32\n", pFirstMsg->uMsg, ShClHostMsgToStr(pFirstMsg->uMsg), pFirstMsg->cParms)); ASSERT_GUEST_MSG_RETURN(pFirstMsg->uMsg == idMsgExpected || idMsgExpected == UINT32_MAX, ("idMsg=%u (%s) cParms=%u, caller expected %u (%s) and %u\n", pFirstMsg->uMsg, ShClHostMsgToStr(pFirstMsg->uMsg), pFirstMsg->cParms, idMsgExpected, ShClHostMsgToStr(idMsgExpected), cParms), VERR_MISMATCH); ASSERT_GUEST_MSG_RETURN(pFirstMsg->cParms == cParms, ("idMsg=%u (%s) cParms=%u, caller expected %u (%s) and %u\n", pFirstMsg->uMsg, ShClHostMsgToStr(pFirstMsg->uMsg), pFirstMsg->cParms, idMsgExpected, ShClHostMsgToStr(idMsgExpected), cParms), VERR_WRONG_PARAMETER_COUNT); /* Check the parameter types. */ for (uint32_t i = 0; i < cParms; i++) ASSERT_GUEST_MSG_RETURN(pFirstMsg->paParms[i].type == paParms[i].type, ("param #%u: type %u, caller expected %u (idMsg=%u %s)\n", i, pFirstMsg->paParms[i].type, paParms[i].type, pFirstMsg->uMsg, ShClHostMsgToStr(pFirstMsg->uMsg)), VERR_WRONG_PARAMETER_TYPE); /* * Copy out the parameters. * * No assertions on buffer overflows, and keep going till the end so we can * communicate all the required buffer sizes. */ int rc = VINF_SUCCESS; for (uint32_t i = 0; i < cParms; i++) switch (pFirstMsg->paParms[i].type) { case VBOX_HGCM_SVC_PARM_32BIT: paParms[i].u.uint32 = pFirstMsg->paParms[i].u.uint32; break; case VBOX_HGCM_SVC_PARM_64BIT: paParms[i].u.uint64 = pFirstMsg->paParms[i].u.uint64; break; case VBOX_HGCM_SVC_PARM_PTR: { uint32_t const cbSrc = pFirstMsg->paParms[i].u.pointer.size; uint32_t const cbDst = paParms[i].u.pointer.size; paParms[i].u.pointer.size = cbSrc; /** @todo Check if this is safe in other layers... * Update: Safe, yes, but VMMDevHGCM doesn't pass it along. */ if (cbSrc <= cbDst) memcpy(paParms[i].u.pointer.addr, pFirstMsg->paParms[i].u.pointer.addr, cbSrc); else { AssertMsgFailed(("#%u: cbSrc=%RU32 is bigger than cbDst=%RU32\n", i, cbSrc, cbDst)); rc = VERR_BUFFER_OVERFLOW; } break; } default: AssertMsgFailed(("#%u: %u\n", i, pFirstMsg->paParms[i].type)); rc = VERR_INTERNAL_ERROR; break; } if (RT_SUCCESS(rc)) { /* * Complete the message and remove the pending message unless the * guest raced us and cancelled this call in the meantime. */ AssertPtr(g_pHelpers); rc = g_pHelpers->pfnCallComplete(hCall, rc); LogFlowFunc(("[Client %RU32] pfnCallComplete -> %Rrc\n", pClient->State.uClientID, rc)); if (rc != VERR_CANCELLED) { pClient->queueMsg.removeFirst(); shClSvcMsgFree(pFirstMsg); } return VINF_HGCM_ASYNC_EXECUTE; /* The caller must not complete it. */ } LogFlowFunc(("[Client %RU32] Returning %Rrc\n", pClient->State.uClientID, rc)); return rc; } } paParms[0].u.uint32 = 0; paParms[1].u.uint32 = 0; LogFlowFunc(("[Client %RU32] -> VERR_TRY_AGAIN\n", pClient->State.uClientID)); return VERR_TRY_AGAIN; } /** * Wakes up a pending client (i.e. waiting for new messages). * * @returns VBox status code. * @retval VINF_NO_CHANGE if the client is not in pending mode. * * @param pClient Client to wake up. */ int shClSvcClientWakeup(PSHCLCLIENT pClient) { int rc = VINF_NO_CHANGE; if (pClient->Pending.uType) { LogFunc(("[Client %RU32] Waking up ...\n", pClient->State.uClientID)); rc = VINF_SUCCESS; if (!pClient->queueMsg.isEmpty()) { PSHCLCLIENTMSG pFirstMsg = pClient->queueMsg.first(); if (pFirstMsg) { LogFunc(("[Client %RU32] Current host message is %RU32 (%s), cParms=%RU32\n", pClient->State.uClientID, pFirstMsg->uMsg, ShClHostMsgToStr(pFirstMsg->uMsg), pFirstMsg->cParms)); bool fDonePending = false; if (pClient->Pending.uType == VBOX_SHCL_GUEST_FN_MSG_PEEK_WAIT) { shClSvcMsgSetPeekReturn(pFirstMsg, pClient->Pending.paParms, pClient->Pending.cParms); fDonePending = true; } else if (pClient->Pending.uType == VBOX_SHCL_GUEST_FN_GET_HOST_MSG_OLD) /* Legacy, Guest Additions < 6.1. */ { bool fRemove; rc = shClSvcMsgSetGetHostMsgOldReturn(pFirstMsg, pClient->Pending.paParms, pClient->Pending.cParms, &fRemove); if (RT_SUCCESS(rc)) { if (fRemove) { /* The old (legacy) protocol gets the message right when returning from peeking, so * remove the actual message from our queue right now. */ pClient->queueMsg.removeFirst(); shClSvcMsgFree(pFirstMsg); } fDonePending = true; } } if (fDonePending) { rc = g_pHelpers->pfnCallComplete(pClient->Pending.hHandle, VINF_SUCCESS); pClient->Pending.hHandle = NULL; pClient->Pending.paParms = NULL; pClient->Pending.cParms = 0; pClient->Pending.uType = 0; } } else AssertFailed(); } else AssertMsgFailed(("Waking up client ID=%RU32 with no host message in queue is a bad idea\n", pClient->State.uClientID)); return rc; } else LogFunc(("[Client %RU32] Not in pending state, skipping wakeup\n", pClient->State.uClientID)); return VINF_NO_CHANGE; } /** * Requests to read clipboard data from the guest. * * @returns VBox status code. * @param pClient Client to request to read data form. * @param pDataReq Data request to send to the guest. * @param puEvent Event ID for waiting for new data. Optional. */ int ShClSvcDataReadRequest(PSHCLCLIENT pClient, PSHCLDATAREQ pDataReq, PSHCLEVENTID puEvent) { AssertPtrReturn(pClient, VERR_INVALID_POINTER); AssertPtrReturn(pDataReq, VERR_INVALID_POINTER); /* puEvent is optional. */ LogFlowFuncEnter(); int rc; PSHCLCLIENTMSG pMsgReadData = shClSvcMsgAlloc(VBOX_SHCL_HOST_MSG_READ_DATA, VBOX_SHCL_CPARMS_READ_DATA_REQ); if (pMsgReadData) { const SHCLEVENTID uEvent = ShClEventIDGenerate(&pClient->Events); LogFlowFunc(("uFmt=0x%x\n", pDataReq->uFmt)); HGCMSvcSetU64(&pMsgReadData->paParms[0], VBOX_SHCL_CONTEXTID_MAKE(pClient->State.uSessionID, pClient->Events.uID, uEvent)); HGCMSvcSetU32(&pMsgReadData->paParms[1], 0 /* fFlags */); HGCMSvcSetU32(&pMsgReadData->paParms[2], pDataReq->uFmt); HGCMSvcSetU32(&pMsgReadData->paParms[3], pClient->State.cbChunkSize); rc = shClSvcMsgAdd(pClient, pMsgReadData, true /* fAppend */); if (RT_SUCCESS(rc)) { rc = ShClEventRegister(&pClient->Events, uEvent); if (RT_SUCCESS(rc)) { rc = shClSvcClientWakeup(pClient); if (RT_SUCCESS(rc)) { if (puEvent) *puEvent = uEvent; } else ShClEventUnregister(&pClient->Events, uEvent); } } } else rc = VERR_NO_MEMORY; LogFlowFuncLeaveRC(rc); return rc; } int ShClSvcDataReadSignal(PSHCLCLIENT pClient, PSHCLCLIENTCMDCTX pCmdCtx, PSHCLDATABLOCK pData) { AssertPtrReturn(pClient, VERR_INVALID_POINTER); AssertPtrReturn(pCmdCtx, VERR_INVALID_POINTER); AssertPtrReturn(pData, VERR_INVALID_POINTER); LogFlowFuncEnter(); SHCLEVENTID uEvent; if (!(pClient->State.fGuestFeatures0 & VBOX_SHCL_GF_0_CONTEXT_ID)) /* Legacy, Guest Additions < 6.1. */ { /* Older Guest Additions (<= VBox 6.0) did not have any context ID handling, so we ASSUME that the last event registered * is the one we want to handle (as this all was a synchronous protocol anyway). */ uEvent = ShClEventGetLast(&pClient->Events); } else uEvent = VBOX_SHCL_CONTEXTID_GET_EVENT(pCmdCtx->uContextID); int rc = VINF_SUCCESS; PSHCLEVENTPAYLOAD pPayload = NULL; if (pData->cbData) rc = ShClPayloadAlloc(uEvent, pData->pvData, pData->cbData, &pPayload); if (RT_SUCCESS(rc)) { rc = ShClEventSignal(&pClient->Events, uEvent, pPayload); if (RT_FAILURE(rc)) ShClPayloadFree(pPayload); } LogFlowFuncLeaveRC(rc); return rc; } int ShClSvcFormatsReport(PSHCLCLIENT pClient, PSHCLFORMATDATA pFormats) { AssertPtrReturn(pClient, VERR_INVALID_POINTER); AssertPtrReturn(pFormats, VERR_INVALID_POINTER); LogFlowFuncEnter(); int rc; PSHCLCLIENTMSG pMsg = shClSvcMsgAlloc(VBOX_SHCL_HOST_MSG_FORMATS_REPORT, 3); if (pMsg) { const SHCLEVENTID uEvent = ShClEventIDGenerate(&pClient->Events); HGCMSvcSetU64(&pMsg->paParms[0], VBOX_SHCL_CONTEXTID_MAKE(pClient->State.uSessionID, pClient->Events.uID, uEvent)); HGCMSvcSetU32(&pMsg->paParms[1], pFormats->Formats); HGCMSvcSetU32(&pMsg->paParms[2], 0 /* fFlags */); rc = shClSvcMsgAdd(pClient, pMsg, true /* fAppend */); if (RT_SUCCESS(rc)) { #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS /* If this is an URI list, create a transfer locally and also tell the guest to create * a transfer on the guest side. */ if (pFormats->Formats & VBOX_SHCL_FMT_URI_LIST) { rc = shClSvcTransferStart(pClient, SHCLTRANSFERDIR_WRITE, SHCLSOURCE_LOCAL, NULL /* pTransfer */); if (RT_FAILURE(rc)) LogRel(("Shared Clipboard: Initializing host write transfer failed with %Rrc\n", rc)); } else { #endif pClient->State.fFlags |= SHCLCLIENTSTATE_FLAGS_READ_ACTIVE; rc = shClSvcClientWakeup(pClient); #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS } #endif } } else rc = VERR_NO_MEMORY; LogFlowFuncLeaveRC(rc); return rc; } int shClSvcGetDataRead(PSHCLCLIENT pClient, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) { LogFlowFuncEnter(); if ( ShClSvcGetMode() != VBOX_SHCL_MODE_HOST_TO_GUEST && ShClSvcGetMode() != VBOX_SHCL_MODE_BIDIRECTIONAL) { return VERR_ACCESS_DENIED; } /* Is the guest supposed to read any clipboard data from the host? */ if (!(pClient->State.fFlags & SHCLCLIENTSTATE_FLAGS_READ_ACTIVE)) return VERR_WRONG_ORDER; int rc; SHCLCLIENTCMDCTX cmdCtx; RT_ZERO(cmdCtx); SHCLDATABLOCK dataBlock; RT_ZERO(dataBlock); if (!(pClient->State.fGuestFeatures0 & VBOX_SHCL_GF_0_CONTEXT_ID)) /* Legacy, Guest Additions < 6.1. */ { if (cParms != 3) { rc = VERR_INVALID_PARAMETER; } else { rc = HGCMSvcGetU32(&paParms[0], &dataBlock.uFormat); if (RT_SUCCESS(rc)) { if (pClient->State.POD.uFormat == VBOX_SHCL_FMT_NONE) pClient->State.POD.uFormat = dataBlock.uFormat; if (dataBlock.uFormat != pClient->State.POD.uFormat) { LogFlowFunc(("Invalid format (pClient->State.POD.uFormat=%RU32 vs dataBlock.uFormat=%RU32)\n", pClient->State.POD.uFormat, dataBlock.uFormat)); rc = VERR_INVALID_PARAMETER; } else { rc = HGCMSvcGetBuf(&paParms[1], &dataBlock.pvData, &dataBlock.cbData); } } } } else { if (cParms < VBOX_SHCL_CPARMS_READ_DATA) { rc = VERR_INVALID_PARAMETER; } else { /** @todo Handle paParms[1] flags. */ rc = HGCMSvcGetU32(&paParms[2], &dataBlock.uFormat); if (RT_SUCCESS(rc)) { uint32_t cbData; rc = HGCMSvcGetU32(&paParms[3], &cbData); if (RT_SUCCESS(rc)) { rc = HGCMSvcGetBuf(&paParms[4], &dataBlock.pvData, &dataBlock.cbData); if (RT_SUCCESS(rc)) { if (cbData != dataBlock.cbData) { LogFlowFunc(("Invalid data (cbData=%RU32 vs dataBlock.cbData=%RU32)\n", cbData, dataBlock.cbData)); rc = VERR_INVALID_PARAMETER; } } } } } } if (RT_SUCCESS(rc)) { uint32_t cbActual = 0; /* If there is a service extension active, try reading data from it first. */ if (g_ExtState.pfnExtension) { SHCLEXTPARMS parms; RT_ZERO(parms); parms.uFormat = pClient->State.POD.uFormat; parms.u.pvData = dataBlock.pvData; parms.cbData = dataBlock.cbData; g_ExtState.fReadingData = true; /* Read clipboard data from the extension. */ rc = g_ExtState.pfnExtension(g_ExtState.pvExtension, VBOX_CLIPBOARD_EXT_FN_DATA_READ, &parms, sizeof(parms)); LogFlowFunc(("g_ExtState.fDelayedAnnouncement=%RTbool, g_ExtState.uDelayedFormats=0x%x\n", g_ExtState.fDelayedAnnouncement, g_ExtState.uDelayedFormats)); /* Did the extension send the clipboard formats yet? * Otherwise, do this now. */ if (g_ExtState.fDelayedAnnouncement) { SHCLFORMATDATA formatData; RT_ZERO(formatData); formatData.Formats = g_ExtState.uDelayedFormats; Assert(formatData.Formats != VBOX_SHCL_FMT_NONE); /* There better is *any* format here now. */ int rc2 = ShClSvcFormatsReport(pClient, &formatData); AssertRC(rc2); g_ExtState.fDelayedAnnouncement = false; g_ExtState.uDelayedFormats = 0; } g_ExtState.fReadingData = false; if (RT_SUCCESS(rc)) { cbActual = parms.cbData; } } /* Note: The host clipboard *always* has precedence over the service extension above, * so data which has been read above might get overridden by the host clipboard eventually. */ if (RT_SUCCESS(rc)) { rc = ShClSvcImplReadData(pClient, &cmdCtx, &dataBlock, &cbActual); if (RT_SUCCESS(rc)) { LogFlowFunc(("cbData=%RU32, cbActual=%RU32\n", dataBlock.cbData, cbActual)); /* Return the actual size required to fullfil the request. */ if (!(pClient->State.fGuestFeatures0 & VBOX_SHCL_GF_0_CONTEXT_ID)) /* Legacy, Guest Additions < 6.1. */ { HGCMSvcSetU32(&paParms[2], cbActual); } else { HGCMSvcSetU32(&paParms[3], cbActual); } /* If the data to return exceeds the buffer the guest supplies, tell it (and let it try again). */ if (cbActual >= dataBlock.cbData) rc = VINF_BUFFER_OVERFLOW; if (rc == VINF_SUCCESS) { /* Only remove "read active" flag after successful read again. */ pClient->State.fFlags &= ~SHCLCLIENTSTATE_FLAGS_READ_ACTIVE; } } } } LogFlowFuncLeaveRC(rc); return rc; } int shClSvcGetDataWrite(PSHCLCLIENT pClient, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) { LogFlowFuncEnter(); if ( ShClSvcGetMode() != VBOX_SHCL_MODE_GUEST_TO_HOST && ShClSvcGetMode() != VBOX_SHCL_MODE_BIDIRECTIONAL) { return VERR_ACCESS_DENIED; } /* Is the guest supposed to write any clipboard data from the host? */ if (!(pClient->State.fFlags & SHCLCLIENTSTATE_FLAGS_WRITE_ACTIVE)) return VERR_WRONG_ORDER; int rc; SHCLDATABLOCK dataBlock; RT_ZERO(dataBlock); SHCLCLIENTCMDCTX cmdCtx; RT_ZERO(cmdCtx); if (!(pClient->State.fGuestFeatures0 & VBOX_SHCL_GF_0_CONTEXT_ID)) /* Legacy, Guest Additions < 6.1. */ { if (cParms != 2) { rc = VERR_INVALID_PARAMETER; } else { rc = HGCMSvcGetU32(&paParms[0], &dataBlock.uFormat); if (RT_SUCCESS(rc)) { if (pClient->State.POD.uFormat == VBOX_SHCL_FMT_NONE) pClient->State.POD.uFormat = dataBlock.uFormat; if ( dataBlock.uFormat == VBOX_SHCL_FMT_NONE || dataBlock.uFormat != pClient->State.POD.uFormat) { LogFunc(("Invalid format (client=%RU32 vs host=%RU32)\n", dataBlock.uFormat, pClient->State.POD.uFormat)); rc = VERR_INVALID_PARAMETER; } else rc = HGCMSvcGetBuf(&paParms[1], &dataBlock.pvData, &dataBlock.cbData); } } } else { if (cParms != VBOX_SHCL_CPARMS_WRITE_DATA) { rc = VERR_INVALID_PARAMETER; } else { rc = HGCMSvcGetU64(&paParms[0], &cmdCtx.uContextID); /** @todo Handle paParms[1] flags. */ if (RT_SUCCESS(rc)) rc = HGCMSvcGetU32(&paParms[2], &dataBlock.uFormat); if (RT_SUCCESS(rc)) rc = HGCMSvcGetBuf(&paParms[4], &dataBlock.pvData, &dataBlock.cbData); } } if (RT_SUCCESS(rc)) { if (g_ExtState.pfnExtension) { SHCLEXTPARMS parms; RT_ZERO(parms); parms.uFormat = dataBlock.uFormat; parms.u.pvData = dataBlock.pvData; parms.cbData = dataBlock.cbData; g_ExtState.pfnExtension(g_ExtState.pvExtension, VBOX_CLIPBOARD_EXT_FN_DATA_WRITE, &parms, sizeof(parms)); } rc = ShClSvcImplWriteData(pClient, &cmdCtx, &dataBlock); if (RT_SUCCESS(rc)) { /* Remove "write active" flag after successful read again. */ pClient->State.fFlags &= ~SHCLCLIENTSTATE_FLAGS_WRITE_ACTIVE; } } LogFlowFuncLeaveRC(rc); return rc; } /** * Gets an error from HGCM service parameters. * * @returns VBox status code. * @param cParms Number of HGCM parameters supplied in \a paParms. * @param paParms Array of HGCM parameters. * @param pRc Where to store the received error code. */ static int shClSvcGetError(uint32_t cParms, VBOXHGCMSVCPARM paParms[], int *pRc) { AssertPtrReturn(paParms, VERR_INVALID_PARAMETER); AssertPtrReturn(pRc, VERR_INVALID_PARAMETER); int rc; if (cParms == VBOX_SHCL_CPARMS_ERROR) { rc = HGCMSvcGetU32(&paParms[1], (uint32_t *)pRc); /** @todo int vs. uint32_t !!! */ } else rc = VERR_INVALID_PARAMETER; LogFlowFuncLeaveRC(rc); return rc; } int shClSvcSetSource(PSHCLCLIENT pClient, SHCLSOURCE enmSource) { if (!pClient) /* If no client connected (anymore), bail out. */ return VINF_SUCCESS; int rc = VINF_SUCCESS; if (ShClSvcLock()) { pClient->State.enmSource = enmSource; LogFlowFunc(("Source of client %RU32 is now %RU32\n", pClient->State.uClientID, pClient->State.enmSource)); ShClSvcUnlock(); } LogFlowFuncLeaveRC(rc); return rc; } static int svcInit(void) { int rc = RTCritSectInit(&g_CritSect); if (RT_SUCCESS(rc)) { shClSvcModeSet(VBOX_SHCL_MODE_OFF); rc = ShClSvcImplInit(); /* Clean up on failure, because 'svnUnload' will not be called * if the 'svcInit' returns an error. */ if (RT_FAILURE(rc)) { RTCritSectDelete(&g_CritSect); } } return rc; } static DECLCALLBACK(int) svcUnload(void *) { LogFlowFuncEnter(); ShClSvcImplDestroy(); RTCritSectDelete(&g_CritSect); return VINF_SUCCESS; } static DECLCALLBACK(int) svcDisconnect(void *, uint32_t u32ClientID, void *pvClient) { RT_NOREF(u32ClientID); LogFunc(("u32ClientID=%RU32\n", u32ClientID)); PSHCLCLIENT pClient = (PSHCLCLIENT)pvClient; AssertPtr(pClient); #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS shClSvcClientTransfersReset(pClient); #endif ShClSvcImplDisconnect(pClient); shClSvcClientDestroy(pClient); return VINF_SUCCESS; } static DECLCALLBACK(int) svcConnect(void *, uint32_t u32ClientID, void *pvClient, uint32_t fRequestor, bool fRestoring) { RT_NOREF(fRequestor, fRestoring); PSHCLCLIENT pClient = (PSHCLCLIENT)pvClient; AssertPtr(pvClient); int rc = shClSvcClientInit(pClient, u32ClientID); if (RT_SUCCESS(rc)) { rc = ShClSvcImplConnect(pClient, ShClSvcGetHeadless()); if (RT_SUCCESS(rc)) { /* Sync the host clipboard content with the client. */ rc = ShClSvcImplSync(pClient); if (rc == VINF_NO_CHANGE) { /* * The sync could return VINF_NO_CHANGE if nothing has changed on the host, but older * Guest Additions rely on the fact that only VINF_SUCCESS indicates a successful connect * to the host service (instead of using RT_SUCCESS()). * * So implicitly set VINF_SUCCESS here to not break older Guest Additions. */ rc = VINF_SUCCESS; } if (RT_SUCCESS(rc)) { /* Assign weak pointer to client map .*/ g_mapClients[u32ClientID] = pClient; /** @todo Handle OOM / collisions? */ /* For now we ASSUME that the first client ever connected is in charge for * communicating withe the service extension. * ** @todo This needs to be fixed ASAP w/o breaking older guest / host combos. */ if (g_ExtState.uClientID == 0) g_ExtState.uClientID = u32ClientID; } } } LogFlowFuncLeaveRC(rc); return rc; } static DECLCALLBACK(void) svcCall(void *, VBOXHGCMCALLHANDLE callHandle, uint32_t u32ClientID, void *pvClient, uint32_t u32Function, uint32_t cParms, VBOXHGCMSVCPARM paParms[], uint64_t tsArrival) { RT_NOREF(u32ClientID, pvClient, tsArrival); int rc = VINF_SUCCESS; PSHCLCLIENT pClient = (PSHCLCLIENT)pvClient; AssertPtr(pClient); LogFunc(("u32ClientID=%RU32, fn=%RU32 (%s), cParms=%RU32, paParms=%p\n", u32ClientID, u32Function, ShClGuestMsgToStr(u32Function), cParms, paParms)); #ifdef DEBUG uint32_t i; for (i = 0; i < cParms; i++) { /** @todo parameters other than 32 bit */ LogFunc((" paParms[%d]: type %RU32 - value %RU32\n", i, paParms[i].type, paParms[i].u.uint32)); } #endif LogFunc(("Client state: fFlags=0x%x, fGuestFeatures0=0x%x, fGuestFeatures1=0x%x\n", pClient->State.fFlags, pClient->State.fGuestFeatures0, pClient->State.fGuestFeatures1)); switch (u32Function) { case VBOX_SHCL_GUEST_FN_GET_HOST_MSG_OLD: { rc = shClSvcMsgGetOld(pClient, callHandle, cParms, paParms); break; } case VBOX_SHCL_GUEST_FN_CONNECT: { if (cParms != VBOX_SHCL_CPARMS_CONNECT) { rc = VERR_INVALID_PARAMETER; } else if ( paParms[0].type != VBOX_HGCM_SVC_PARM_32BIT /* cbChunkSize */ || paParms[1].type != VBOX_HGCM_SVC_PARM_32BIT /* enmCompression */ || paParms[2].type != VBOX_HGCM_SVC_PARM_32BIT) /* enmChecksumType */ { rc = VERR_INVALID_PARAMETER; } else if (ShClSvcGetMode() == VBOX_SHCL_MODE_OFF) { rc = VERR_ACCESS_DENIED; } else { /* Report back supported chunk size to the guest. */ HGCMSvcSetU32(&paParms[0], _64K); /* Chunk size */ /** @todo Make chunk size dynamic. */ rc = VINF_SUCCESS; } break; } case VBOX_SHCL_GUEST_FN_REPORT_FEATURES: { rc = shClSvcClientReportFeatures(pClient, callHandle, cParms, paParms); break; } case VBOX_SHCL_GUEST_FN_QUERY_FEATURES: { rc = shClSvcClientQueryFeatures(callHandle, cParms, paParms); break; } case VBOX_SHCL_GUEST_FN_MSG_PEEK_NOWAIT: { rc = shClSvcMsgPeek(pClient, callHandle, cParms, paParms, false /*fWait*/); break; } case VBOX_SHCL_GUEST_FN_MSG_PEEK_WAIT: { rc = shClSvcMsgPeek(pClient, callHandle, cParms, paParms, true /*fWait*/); break; } case VBOX_SHCL_GUEST_FN_MSG_GET: { rc = shClSvcMsgGet(pClient, callHandle, cParms, paParms); break; } case VBOX_SHCL_GUEST_FN_FORMATS_REPORT: { uint32_t uFormats = 0; if (!(pClient->State.fGuestFeatures0 & VBOX_SHCL_GF_0_CONTEXT_ID)) /* Legacy, Guest Additions < 6.1. */ { if (cParms != 1) { rc = VERR_INVALID_PARAMETER; } else if (paParms[0].type != VBOX_HGCM_SVC_PARM_32BIT) /* uFormats */ { rc = VERR_INVALID_PARAMETER; } else { rc = HGCMSvcGetU32(&paParms[0], &uFormats); } } else { if (cParms != 3) { rc = VERR_INVALID_PARAMETER; } else if ( paParms[0].type != VBOX_HGCM_SVC_PARM_64BIT /* uContextID */ || paParms[1].type != VBOX_HGCM_SVC_PARM_32BIT /* uFormats */ || paParms[2].type != VBOX_HGCM_SVC_PARM_32BIT) /* fFlags */ { rc = VERR_INVALID_PARAMETER; } else { rc = HGCMSvcGetU32(&paParms[1], &uFormats); /** @todo Handle rest. */ } } if (RT_SUCCESS(rc)) { if ( ShClSvcGetMode() != VBOX_SHCL_MODE_GUEST_TO_HOST && ShClSvcGetMode() != VBOX_SHCL_MODE_BIDIRECTIONAL) { rc = VERR_ACCESS_DENIED; } else if (uFormats != VBOX_SHCL_FMT_NONE) /* Only announce formats if we actually *have* formats to announce! */ { rc = shClSvcSetSource(pClient, SHCLSOURCE_REMOTE); if (RT_SUCCESS(rc)) { if (g_ExtState.pfnExtension) { SHCLEXTPARMS parms; RT_ZERO(parms); parms.uFormat = uFormats; g_ExtState.pfnExtension(g_ExtState.pvExtension, VBOX_CLIPBOARD_EXT_FN_FORMAT_ANNOUNCE, &parms, sizeof(parms)); } SHCLCLIENTCMDCTX cmdCtx; RT_ZERO(cmdCtx); SHCLFORMATDATA formatData; RT_ZERO(formatData); formatData.Formats = uFormats; Assert(formatData.Formats != VBOX_SHCL_FMT_NONE); /* Sanity. */ rc = ShClSvcImplFormatAnnounce(pClient, &cmdCtx, &formatData); if (RT_SUCCESS(rc)) { pClient->State.fFlags |= SHCLCLIENTSTATE_FLAGS_WRITE_ACTIVE; } } } } break; } case VBOX_SHCL_GUEST_FN_DATA_READ: { rc = shClSvcGetDataRead(pClient, cParms, paParms); break; } case VBOX_SHCL_GUEST_FN_DATA_WRITE: { rc = shClSvcGetDataWrite(pClient, cParms, paParms); break; } case VBOX_SHCL_GUEST_FN_CANCEL: { LogRel2(("Shared Clipboard: Operation canceled by guest side\n")); /* Reset client state and start over. */ shclSvcClientStateReset(&pClient->State); #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS shClSvcClientTransfersReset(pClient); #endif /** @todo Do we need to do anything else here? */ break; } case VBOX_SHCL_GUEST_FN_ERROR: { int rcGuest; rc = shClSvcGetError(cParms,paParms, &rcGuest); if (RT_SUCCESS(rc)) { LogRel(("Shared Clipboard: Error from guest side: %Rrc\n", rcGuest)); /* Reset client state and start over. */ shclSvcClientStateReset(&pClient->State); #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS shClSvcClientTransfersReset(pClient); #endif } break; } default: { #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS if (g_fTransferMode & VBOX_SHCL_TRANSFER_MODE_ENABLED) { rc = shClSvcTransferHandler(pClient, callHandle, u32Function, cParms, paParms, tsArrival); } else { LogRel2(("Shared Clipboard: File transfers are disabled for this VM\n")); rc = VERR_ACCESS_DENIED; } #else rc = VERR_NOT_SUPPORTED; #endif break; } } LogFlowFunc(("[Client %RU32] rc=%Rrc\n", pClient->State.uClientID, rc)); if (rc != VINF_HGCM_ASYNC_EXECUTE) g_pHelpers->pfnCallComplete(callHandle, rc); } /** * Initializes a Shared Clipboard service's client state. * * @returns VBox status code. * @param pClientState Client state to initialize. * @param uClientID Client ID (HGCM) to use for this client state. */ int shClSvcClientStateInit(PSHCLCLIENTSTATE pClientState, uint32_t uClientID) { LogFlowFuncEnter(); shclSvcClientStateReset(pClientState); /* Register the client. */ pClientState->uClientID = uClientID; return VINF_SUCCESS; } /** * Destroys a Shared Clipboard service's client state. * * @returns VBox status code. * @param pClientState Client state to destroy. */ int shClSvcClientStateDestroy(PSHCLCLIENTSTATE pClientState) { RT_NOREF(pClientState); LogFlowFuncEnter(); return VINF_SUCCESS; } /** * Resets a Shared Clipboard service's client state. * * @param pClientState Client state to reset. */ void shclSvcClientStateReset(PSHCLCLIENTSTATE pClientState) { LogFlowFuncEnter(); pClientState->fGuestFeatures0 = VBOX_SHCL_GF_NONE; pClientState->fGuestFeatures1 = VBOX_SHCL_GF_NONE; pClientState->cbChunkSize = _64K; /** Make this configurable. */ pClientState->enmSource = SHCLSOURCE_INVALID; pClientState->fFlags = SHCLCLIENTSTATE_FLAGS_NONE; pClientState->POD.enmDir = SHCLTRANSFERDIR_UNKNOWN; pClientState->POD.uFormat = VBOX_SHCL_FMT_NONE; pClientState->POD.cbToReadWriteTotal = 0; pClientState->POD.cbReadWritten = 0; #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS pClientState->Transfers.enmTransferDir = SHCLTRANSFERDIR_UNKNOWN; #endif } /* * We differentiate between a function handler for the guest and one for the host. */ static DECLCALLBACK(int) svcHostCall(void *, uint32_t u32Function, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) { int rc = VINF_SUCCESS; LogFlowFunc(("u32Function=%RU32 (%s), cParms=%RU32, paParms=%p\n", u32Function, ShClHostFunctionToStr(u32Function), cParms, paParms)); switch (u32Function) { case VBOX_SHCL_HOST_FN_SET_MODE: { if (cParms != 1) { rc = VERR_INVALID_PARAMETER; } else { uint32_t u32Mode = VBOX_SHCL_MODE_OFF; rc = HGCMSvcGetU32(&paParms[0], &u32Mode); if (RT_SUCCESS(rc)) rc = shClSvcModeSet(u32Mode); } break; } #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS case VBOX_SHCL_HOST_FN_SET_TRANSFER_MODE: { if (cParms != 1) { rc = VERR_INVALID_PARAMETER; } else { uint32_t fTransferMode; rc = HGCMSvcGetU32(&paParms[0], &fTransferMode); if (RT_SUCCESS(rc)) rc = shClSvcTransferModeSet(fTransferMode); } break; } #endif case VBOX_SHCL_HOST_FN_SET_HEADLESS: { if (cParms != 1) { rc = VERR_INVALID_PARAMETER; } else { uint32_t uHeadless; rc = HGCMSvcGetU32(&paParms[0], &uHeadless); if (RT_SUCCESS(rc)) { g_fHeadless = RT_BOOL(uHeadless); LogRel(("Shared Clipboard: Service running in %s mode\n", g_fHeadless ? "headless" : "normal")); } } break; } default: { #ifdef VBOX_WITH_SHARED_CLIPBOARD_TRANSFERS rc = shClSvcTransferHostHandler(u32Function, cParms, paParms); #else rc = VERR_NOT_IMPLEMENTED; #endif break; } } LogFlowFuncLeaveRC(rc); return rc; } #ifndef UNIT_TEST /** * SSM descriptor table for the SHCLCLIENTSTATE structure. */ static SSMFIELD const s_aShClSSMClientState[] = { /** Note: Saving the session ID not necessary, as they're not persistent across state save/restore. */ SSMFIELD_ENTRY(SHCLCLIENTSTATE, fGuestFeatures0), SSMFIELD_ENTRY(SHCLCLIENTSTATE, fGuestFeatures1), SSMFIELD_ENTRY(SHCLCLIENTSTATE, cbChunkSize), SSMFIELD_ENTRY(SHCLCLIENTSTATE, enmSource), SSMFIELD_ENTRY(SHCLCLIENTSTATE, fFlags), SSMFIELD_ENTRY_TERM() }; /** * SSM descriptor table for the SHCLCLIENTPODSTATE structure. */ static SSMFIELD const s_aShClSSMClientPODState[] = { SSMFIELD_ENTRY(SHCLCLIENTPODSTATE, enmDir), SSMFIELD_ENTRY(SHCLCLIENTPODSTATE, uFormat), SSMFIELD_ENTRY(SHCLCLIENTPODSTATE, cbToReadWriteTotal), SSMFIELD_ENTRY(SHCLCLIENTPODSTATE, cbReadWritten), SSMFIELD_ENTRY(SHCLCLIENTPODSTATE, tsLastReadWrittenMs), SSMFIELD_ENTRY_TERM() }; /** * SSM descriptor table for the SHCLCLIENTURISTATE structure. */ static SSMFIELD const s_aShClSSMClientTransferState[] = { SSMFIELD_ENTRY(SHCLCLIENTTRANSFERSTATE, enmTransferDir), SSMFIELD_ENTRY_TERM() }; /** * SSM descriptor table for the header of the SHCLCLIENTMSG structure. * The actual message parameters will be serialized separately. */ static SSMFIELD const s_aShClSSMClientMsgHdr[] = { SSMFIELD_ENTRY(SHCLCLIENTMSG, uMsg), SSMFIELD_ENTRY(SHCLCLIENTMSG, cParms), SSMFIELD_ENTRY_TERM() }; /** * SSM descriptor table for the VBOXSHCLMSGCTX structure. */ static SSMFIELD const s_aShClSSMClientMsgCtx[] = { SSMFIELD_ENTRY(SHCLMSGCTX, uContextID), SSMFIELD_ENTRY_TERM() }; #endif /* !UNIT_TEST */ static DECLCALLBACK(int) svcSaveState(void *, uint32_t u32ClientID, void *pvClient, PSSMHANDLE pSSM) { LogFlowFuncEnter(); #ifndef UNIT_TEST /* * When the state will be restored, pending requests will be reissued * by VMMDev. The service therefore must save state as if there were no * pending request. * Pending requests, if any, will be completed in svcDisconnect. */ RT_NOREF(u32ClientID); LogFunc(("u32ClientID=%RU32\n", u32ClientID)); PSHCLCLIENT pClient = (PSHCLCLIENT)pvClient; AssertPtr(pClient); /* Write Shared Clipboard saved state version. */ SSMR3PutU32(pSSM, VBOX_SHCL_SSM_VER_LATEST); int rc = SSMR3PutStructEx(pSSM, &pClient->State, sizeof(pClient->State), 0 /*fFlags*/, &s_aShClSSMClientState[0], NULL); AssertRCReturn(rc, rc); rc = SSMR3PutStructEx(pSSM, &pClient->State.POD, sizeof(pClient->State.POD), 0 /*fFlags*/, &s_aShClSSMClientPODState[0], NULL); AssertRCReturn(rc, rc); rc = SSMR3PutStructEx(pSSM, &pClient->State.Transfers, sizeof(pClient->State.Transfers), 0 /*fFlags*/, &s_aShClSSMClientTransferState[0], NULL); AssertRCReturn(rc, rc); /* Serialize the client's internal message queue. */ rc = SSMR3PutU64(pSSM, (uint64_t)pClient->queueMsg.size()); AssertRCReturn(rc, rc); for (size_t i = 0; i < pClient->queueMsg.size(); i++) { PSHCLCLIENTMSG pMsg = pClient->queueMsg.at(i); AssertPtr(pMsg); rc = SSMR3PutStructEx(pSSM, pMsg, sizeof(SHCLCLIENTMSG), 0 /*fFlags*/, &s_aShClSSMClientMsgHdr[0], NULL); AssertRCReturn(rc, rc); rc = SSMR3PutStructEx(pSSM, &pMsg->Ctx, sizeof(SHCLMSGCTX), 0 /*fFlags*/, &s_aShClSSMClientMsgCtx[0], NULL); AssertRCReturn(rc, rc); for (uint32_t p = 0; p < pMsg->cParms; p++) { rc = HGCMSvcSSMR3Put(&pMsg->paParms[p], pSSM); AssertRCReturn(rc, rc); } } #else /* UNIT_TEST */ RT_NOREF3(u32ClientID, pvClient, pSSM); #endif /* UNIT_TEST */ return VINF_SUCCESS; } #ifndef UNIT_TEST static int svcLoadStateV0(uint32_t u32ClientID, void *pvClient, PSSMHANDLE pSSM, uint32_t uVersion) { RT_NOREF(u32ClientID, pvClient, pSSM, uVersion); uint32_t uMarker; int rc = SSMR3GetU32(pSSM, &uMarker); /* Begin marker. */ AssertRC(rc); Assert(uMarker == UINT32_C(0x19200102) /* SSMR3STRUCT_BEGIN */); rc = SSMR3Skip(pSSM, sizeof(uint32_t)); /* Client ID */ AssertRCReturn(rc, rc); bool fValue; rc = SSMR3GetBool(pSSM, &fValue); /* fHostMsgQuit */ AssertRCReturn(rc, rc); rc = SSMR3GetBool(pSSM, &fValue); /* fHostMsgReadData */ AssertRCReturn(rc, rc); rc = SSMR3GetBool(pSSM, &fValue); /* fHostMsgFormats */ AssertRCReturn(rc, rc); uint32_t fFormats; rc = SSMR3GetU32(pSSM, &fFormats); /* u32RequestedFormat */ AssertRCReturn(rc, rc); rc = SSMR3GetU32(pSSM, &uMarker); /* End marker. */ AssertRCReturn(rc, rc); Assert(uMarker == UINT32_C(0x19920406) /* SSMR3STRUCT_END */); return VINF_SUCCESS; } #endif /* UNIT_TEST */ static DECLCALLBACK(int) svcLoadState(void *, uint32_t u32ClientID, void *pvClient, PSSMHANDLE pSSM, uint32_t uVersion) { LogFlowFuncEnter(); #ifndef UNIT_TEST RT_NOREF(u32ClientID, uVersion); PSHCLCLIENT pClient = (PSHCLCLIENT)pvClient; AssertPtr(pClient); /* Restore the client data. */ uint32_t lenOrVer; int rc = SSMR3GetU32(pSSM, &lenOrVer); AssertRCReturn(rc, rc); LogFunc(("u32ClientID=%RU32, lenOrVer=%#RX64\n", u32ClientID, lenOrVer)); if (lenOrVer == VBOX_SHCL_SSM_VER_0) { return svcLoadStateV0(u32ClientID, pvClient, pSSM, uVersion); } else if (lenOrVer >= VBOX_SHCL_SSM_VER_1) { if (lenOrVer >= VBOX_SHCL_SSM_VER_2) { rc = SSMR3GetStructEx(pSSM, &pClient->State, sizeof(pClient->State), 0 /* fFlags */, &s_aShClSSMClientState[0], NULL); AssertRCReturn(rc, rc); rc = SSMR3GetStructEx(pSSM, &pClient->State.POD, sizeof(pClient->State.POD), 0 /* fFlags */, &s_aShClSSMClientPODState[0], NULL); AssertRCReturn(rc, rc); } else /** @todo Remove this block after 6.1 RC; don't annoy team members with broken saved states. */ { rc = SSMR3Skip(pSSM, sizeof(uint32_t)); /* Begin marker */ AssertRC(rc); rc = SSMR3GetU64(pSSM, &pClient->State.fGuestFeatures0); AssertRC(rc); rc = SSMR3GetU64(pSSM, &pClient->State.fGuestFeatures1); AssertRC(rc); rc = SSMR3GetU32(pSSM, &pClient->State.cbChunkSize); AssertRC(rc); rc = SSMR3GetU32(pSSM, (uint32_t *)&pClient->State.enmSource); AssertRC(rc); rc = SSMR3Skip(pSSM, sizeof(uint32_t)); /* End marker */ AssertRC(rc); } rc = SSMR3GetStructEx(pSSM, &pClient->State.Transfers, sizeof(pClient->State.Transfers), 0 /* fFlags */, &s_aShClSSMClientTransferState[0], NULL); AssertRCReturn(rc, rc); /* Load the client's internal message queue. */ uint64_t cMsg; rc = SSMR3GetU64(pSSM, &cMsg); AssertRCReturn(rc, rc); for (uint64_t i = 0; i < cMsg; i++) { PSHCLCLIENTMSG pMsg = (PSHCLCLIENTMSG)RTMemAlloc(sizeof(SHCLCLIENTMSG)); AssertPtrReturn(pMsg, VERR_NO_MEMORY); rc = SSMR3GetStructEx(pSSM, pMsg, sizeof(SHCLCLIENTMSG), 0 /*fFlags*/, &s_aShClSSMClientMsgHdr[0], NULL); AssertRCReturn(rc, rc); rc = SSMR3GetStructEx(pSSM, &pMsg->Ctx, sizeof(SHCLMSGCTX), 0 /*fFlags*/, &s_aShClSSMClientMsgCtx[0], NULL); AssertRCReturn(rc, rc); pMsg->paParms = (PVBOXHGCMSVCPARM)RTMemAllocZ(sizeof(VBOXHGCMSVCPARM) * pMsg->cParms); AssertPtrReturn(pMsg->paParms, VERR_NO_MEMORY); for (uint32_t p = 0; p < pMsg->cParms; p++) { rc = HGCMSvcSSMR3Get(&pMsg->paParms[p], pSSM); AssertRCReturn(rc, rc); } rc = shClSvcMsgAdd(pClient, pMsg, true /* fAppend */); AssertRCReturn(rc, rc); } } else { LogRel(("Shared Clipboard: Unknown saved state version (%#x)\n", lenOrVer)); return VERR_SSM_DATA_UNIT_FORMAT_CHANGED; } /* Actual host data are to be reported to guest (SYNC). */ ShClSvcImplSync(pClient); #else /* UNIT_TEST */ RT_NOREF(u32ClientID, pvClient, pSSM, uVersion); #endif /* UNIT_TEST */ return VINF_SUCCESS; } static DECLCALLBACK(int) extCallback(uint32_t u32Function, uint32_t u32Format, void *pvData, uint32_t cbData) { RT_NOREF(pvData, cbData); LogFlowFunc(("u32Function=%RU32\n", u32Function)); int rc = VINF_SUCCESS; /* Figure out if the client in charge for the service extension still is connected. */ ClipboardClientMap::const_iterator itClient = g_mapClients.find(g_ExtState.uClientID); if (itClient != g_mapClients.end()) { PSHCLCLIENT pClient = itClient->second; AssertPtr(pClient); switch (u32Function) { /* The service extension announces formats to the guest. */ case VBOX_CLIPBOARD_EXT_FN_FORMAT_ANNOUNCE: { LogFlowFunc(("VBOX_CLIPBOARD_EXT_FN_FORMAT_ANNOUNCE: g_ExtState.fReadingData=%RTbool\n", g_ExtState.fReadingData)); if (g_ExtState.fReadingData) { g_ExtState.fDelayedAnnouncement = true; g_ExtState.uDelayedFormats = u32Format; } else if (u32Format != VBOX_SHCL_FMT_NONE) { SHCLFORMATDATA formatData; RT_ZERO(formatData); formatData.Formats = u32Format; rc = ShClSvcFormatsReport(pClient, &formatData); } break; } /* The service extension wants read data from the guest. */ case VBOX_CLIPBOARD_EXT_FN_DATA_READ: { SHCLDATAREQ dataReq; RT_ZERO(dataReq); dataReq.uFmt = u32Format; dataReq.cbSize = _64K; /** @todo Make this more dynamic. */ rc = ShClSvcDataReadRequest(pClient, &dataReq, NULL /* puEvent */); break; } default: /* Just skip other messages. */ break; } } else rc = VERR_NOT_FOUND; LogFlowFuncLeaveRC(rc); return rc; } static DECLCALLBACK(int) svcRegisterExtension(void *, PFNHGCMSVCEXT pfnExtension, void *pvExtension) { LogFlowFunc(("pfnExtension=%p\n", pfnExtension)); SHCLEXTPARMS parms; RT_ZERO(parms); if (pfnExtension) { /* Install extension. */ g_ExtState.pfnExtension = pfnExtension; g_ExtState.pvExtension = pvExtension; parms.u.pfnCallback = extCallback; g_ExtState.pfnExtension(g_ExtState.pvExtension, VBOX_CLIPBOARD_EXT_FN_SET_CALLBACK, &parms, sizeof(parms)); } else { if (g_ExtState.pfnExtension) g_ExtState.pfnExtension(g_ExtState.pvExtension, VBOX_CLIPBOARD_EXT_FN_SET_CALLBACK, &parms, sizeof(parms)); /* Uninstall extension. */ g_ExtState.pfnExtension = NULL; g_ExtState.pvExtension = NULL; } return VINF_SUCCESS; } extern "C" DECLCALLBACK(DECLEXPORT(int)) VBoxHGCMSvcLoad(VBOXHGCMSVCFNTABLE *pTable) { int rc = VINF_SUCCESS; LogFlowFunc(("pTable=%p\n", pTable)); if (!VALID_PTR(pTable)) { rc = VERR_INVALID_PARAMETER; } else { LogFunc(("pTable->cbSize = %d, ptable->u32Version = 0x%08X\n", pTable->cbSize, pTable->u32Version)); if ( pTable->cbSize != sizeof (VBOXHGCMSVCFNTABLE) || pTable->u32Version != VBOX_HGCM_SVC_VERSION) { rc = VERR_VERSION_MISMATCH; } else { g_pHelpers = pTable->pHelpers; pTable->cbClient = sizeof(SHCLCLIENT); pTable->pfnUnload = svcUnload; pTable->pfnConnect = svcConnect; pTable->pfnDisconnect = svcDisconnect; pTable->pfnCall = svcCall; pTable->pfnHostCall = svcHostCall; pTable->pfnSaveState = svcSaveState; pTable->pfnLoadState = svcLoadState; pTable->pfnRegisterExtension = svcRegisterExtension; pTable->pfnNotify = NULL; pTable->pvService = NULL; /* Service specific initialization. */ rc = svcInit(); } } LogFlowFunc(("Returning %Rrc\n", rc)); return rc; }