VirtualBox

source: vbox/trunk/src/VBox/HostServices/GuestControl/service.cpp@ 61522

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

GuestControl/service.cpp: Follow-up for r106475 (bugref:7179): Explicitly handle std::bad_alloc, destroy service object if constructor has thrown, touched copyright.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 57.8 KB
Line 
1/* $Id: service.cpp 60391 2016-04-08 09:29:58Z vboxsync $ */
2/** @file
3 * Guest Control Service: Controlling the guest.
4 */
5
6/*
7 * Copyright (C) 2011-2016 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18/** @page pg_svc_guest_control Guest Control HGCM Service
19 *
20 * This service acts as a proxy for handling and buffering host command requests
21 * and clients on the guest. It tries to be as transparent as possible to let
22 * the guest (client) and host side do their protocol handling as desired.
23 *
24 * The following terms are used:
25 * - Host: A host process (e.g. VBoxManage or another tool utilizing the Main API)
26 * which wants to control something on the guest.
27 * - Client: A client (e.g. VBoxService) running inside the guest OS waiting for
28 * new host commands to perform. There can be multiple clients connected
29 * to this service. A client is represented by its unique HGCM client ID.
30 * - Context ID: An (almost) unique ID automatically generated on the host (Main API)
31 * to not only distinguish clients but individual requests. Because
32 * the host does not know anything about connected clients it needs
33 * an indicator which it can refer to later. This context ID gets
34 * internally bound by the service to a client which actually processes
35 * the command in order to have a relationship between client<->context ID(s).
36 *
37 * The host can trigger commands which get buffered by the service (with full HGCM
38 * parameter info). As soon as a client connects (or is ready to do some new work)
39 * it gets a buffered host command to process it. This command then will be immediately
40 * removed from the command list. If there are ready clients but no new commands to be
41 * processed, these clients will be set into a deferred state (that is being blocked
42 * to return until a new command is available).
43 *
44 * If a client needs to inform the host that something happened, it can send a
45 * message to a low level HGCM callback registered in Main. This callback contains
46 * the actual data as well as the context ID to let the host do the next necessary
47 * steps for this context. This context ID makes it possible to wait for an event
48 * inside the host's Main API function (like starting a process on the guest and
49 * wait for getting its PID returned by the client) as well as cancelling blocking
50 * host calls in order the client terminated/crashed (HGCM detects disconnected
51 * clients and reports it to this service's callback).
52 *
53 * Starting at VBox 4.2 the context ID itself consists of a session ID, an object
54 * ID (for example a process or file ID) and a count. This is necessary to not break
55 * compatibility between older hosts and to manage guest session on the host.
56 */
57
58
59/*********************************************************************************************************************************
60* Header Files *
61*********************************************************************************************************************************/
62#ifdef LOG_GROUP
63 #undef LOG_GROUP
64#endif
65#define LOG_GROUP LOG_GROUP_GUEST_CONTROL
66#include <VBox/HostServices/GuestControlSvc.h>
67
68#include <VBox/log.h>
69#include <iprt/assert.h>
70#include <iprt/cpp/autores.h>
71#include <iprt/cpp/utils.h>
72#include <iprt/err.h>
73#include <iprt/mem.h>
74#include <iprt/list.h>
75#include <iprt/req.h>
76#include <iprt/string.h>
77#include <iprt/thread.h>
78#include <iprt/time.h>
79
80#include <map>
81#include <memory> /* for auto_ptr */
82#include <string>
83#include <list>
84
85#include "gctrl.h"
86
87namespace guestControl {
88
89/** Flag for indicating that the client only is interested in
90 * messages of specific context IDs. */
91#define CLIENTSTATE_FLAG_CONTEXTFILTER RT_BIT(0)
92
93/**
94 * Structure for maintaining a pending (that is, a deferred and not yet completed)
95 * client command.
96 */
97typedef struct ClientConnection
98{
99 /** The call handle */
100 VBOXHGCMCALLHANDLE mHandle;
101 /** Number of parameters */
102 uint32_t mNumParms;
103 /** The call parameters */
104 VBOXHGCMSVCPARM *mParms;
105 /** The standard constructor. */
106 ClientConnection(void)
107 : mHandle(0), mNumParms(0), mParms(NULL) {}
108} ClientConnection;
109
110/**
111 * Structure for holding a buffered host command which has
112 * not been processed yet.
113 */
114typedef struct HostCommand
115{
116 RTLISTNODE Node;
117
118 uint32_t AddRef(void)
119 {
120#ifdef DEBUG_andy
121 LogFlowFunc(("Adding reference pHostCmd=%p, CID=%RU32, new refCount=%RU32\n",
122 this, mContextID, mRefCount + 1));
123#endif
124 return ++mRefCount;
125 }
126
127 uint32_t Release(void)
128 {
129#ifdef DEBUG_andy
130 LogFlowFunc(("Releasing reference pHostCmd=%p, CID=%RU32, new refCount=%RU32\n",
131 this, mContextID, mRefCount - 1));
132#endif
133 /* Release reference for current command. */
134 Assert(mRefCount);
135 if (--mRefCount == 0)
136 Free();
137
138 return mRefCount;
139 }
140
141 /**
142 * Allocates the command with an HGCM request. Needs to be free'd using Free().
143 *
144 * @return IPRT status code.
145 * @param uMsg Message type.
146 * @param cParms Number of parameters of HGCM request.
147 * @param paParms Array of parameters of HGCM request.
148 */
149 int Allocate(uint32_t uMsg, uint32_t cParms, VBOXHGCMSVCPARM paParms[])
150 {
151 LogFlowFunc(("Allocating pHostCmd=%p, uMsg=%RU32, cParms=%RU32, paParms=%p\n",
152 this, uMsg, cParms, paParms));
153
154 if (!cParms) /* At least one parameter (context ID) must be present. */
155 return VERR_INVALID_PARAMETER;
156
157 AssertPtrReturn(paParms, VERR_INVALID_POINTER);
158
159 /* Paranoia. */
160 if (cParms > 256)
161 cParms = 256;
162
163 int rc = VINF_SUCCESS;
164
165 /*
166 * Don't verify anything here (yet), because this function only buffers
167 * the HGCM data into an internal structure and reaches it back to the guest (client)
168 * in an unmodified state.
169 */
170 mMsgType = uMsg;
171 mParmCount = cParms;
172 if (mParmCount)
173 {
174 mpParms = (VBOXHGCMSVCPARM*)RTMemAllocZ(sizeof(VBOXHGCMSVCPARM) * mParmCount);
175 if (NULL == mpParms)
176 rc = VERR_NO_MEMORY;
177 }
178
179 if (RT_SUCCESS(rc))
180 {
181 for (uint32_t i = 0; i < mParmCount; i++)
182 {
183 mpParms[i].type = paParms[i].type;
184 switch (paParms[i].type)
185 {
186 case VBOX_HGCM_SVC_PARM_32BIT:
187 mpParms[i].u.uint32 = paParms[i].u.uint32;
188 break;
189
190 case VBOX_HGCM_SVC_PARM_64BIT:
191 mpParms[i].u.uint64 = paParms[i].u.uint64;
192 break;
193
194 case VBOX_HGCM_SVC_PARM_PTR:
195 mpParms[i].u.pointer.size = paParms[i].u.pointer.size;
196 if (mpParms[i].u.pointer.size > 0)
197 {
198 mpParms[i].u.pointer.addr = RTMemAlloc(mpParms[i].u.pointer.size);
199 if (NULL == mpParms[i].u.pointer.addr)
200 {
201 rc = VERR_NO_MEMORY;
202 break;
203 }
204 else
205 memcpy(mpParms[i].u.pointer.addr,
206 paParms[i].u.pointer.addr,
207 mpParms[i].u.pointer.size);
208 }
209 else
210 {
211 /* Size is 0 -- make sure we don't have any pointer. */
212 mpParms[i].u.pointer.addr = NULL;
213 }
214 break;
215
216 default:
217 break;
218 }
219 if (RT_FAILURE(rc))
220 break;
221 }
222 }
223
224 if (RT_SUCCESS(rc))
225 {
226 /*
227 * Assume that the context ID *always* is the first parameter,
228 * assign the context ID to the command.
229 */
230 rc = mpParms[0].getUInt32(&mContextID);
231
232 /* Set timestamp so that clients can distinguish between already
233 * processed commands and new ones. */
234 mTimestamp = RTTimeNanoTS();
235 }
236
237 LogFlowFunc(("Returned with rc=%Rrc\n", rc));
238 return rc;
239 }
240
241 /**
242 * Frees the buffered HGCM request.
243 *
244 * @return IPRT status code.
245 */
246 void Free(void)
247 {
248 AssertMsg(mRefCount == 0, ("pHostCmd=%p, CID=%RU32 still being used by a client (%RU32 refs), cannot free yet\n",
249 this, mContextID, mRefCount));
250
251 LogFlowFunc(("Freeing host command pHostCmd=%p, CID=%RU32, mMsgType=%RU32, mParmCount=%RU32, mpParms=%p\n",
252 this, mContextID, mMsgType, mParmCount, mpParms));
253
254 for (uint32_t i = 0; i < mParmCount; i++)
255 {
256 switch (mpParms[i].type)
257 {
258 case VBOX_HGCM_SVC_PARM_PTR:
259 if (mpParms[i].u.pointer.size > 0)
260 RTMemFree(mpParms[i].u.pointer.addr);
261 break;
262
263 default:
264 break;
265 }
266 }
267
268 if (mpParms)
269 {
270 RTMemFree(mpParms);
271 mpParms = NULL;
272 }
273
274 mParmCount = 0;
275
276 /* Removes the command from its list */
277 RTListNodeRemove(&Node);
278 }
279
280 /**
281 * Copies data from the buffered HGCM request to the current HGCM request.
282 *
283 * @return IPRT status code.
284 * @param paDstParms Array of parameters of HGCM request to fill the data into.
285 * @param cPDstarms Number of parameters the HGCM request can handle.
286 * @param pSrcBuf Parameter buffer to assign.
287 */
288 int CopyTo(VBOXHGCMSVCPARM paDstParms[], uint32_t cDstParms) const
289 {
290 LogFlowFunc(("pHostCmd=%p, mMsgType=%RU32, mParmCount=%RU32, mContextID=%RU32 (Session %RU32)\n",
291 this, mMsgType, mParmCount, mContextID, VBOX_GUESTCTRL_CONTEXTID_GET_SESSION(mContextID)));
292
293 int rc = VINF_SUCCESS;
294 if (cDstParms != mParmCount)
295 {
296 LogFlowFunc(("Parameter count does not match (got %RU32, expected %RU32)\n",
297 cDstParms, mParmCount));
298 rc = VERR_INVALID_PARAMETER;
299 }
300
301 if (RT_SUCCESS(rc))
302 {
303 for (uint32_t i = 0; i < mParmCount; i++)
304 {
305 if (paDstParms[i].type != mpParms[i].type)
306 {
307 LogFlowFunc(("Parameter %RU32 type mismatch (got %RU32, expected %RU32)\n",
308 i, paDstParms[i].type, mpParms[i].type));
309 rc = VERR_INVALID_PARAMETER;
310 }
311 else
312 {
313 switch (mpParms[i].type)
314 {
315 case VBOX_HGCM_SVC_PARM_32BIT:
316#ifdef DEBUG_andy
317 LogFlowFunc(("\tmpParms[%RU32] = %RU32 (uint32_t)\n",
318 i, mpParms[i].u.uint32));
319#endif
320 paDstParms[i].u.uint32 = mpParms[i].u.uint32;
321 break;
322
323 case VBOX_HGCM_SVC_PARM_64BIT:
324#ifdef DEBUG_andy
325 LogFlowFunc(("\tmpParms[%RU32] = %RU64 (uint64_t)\n",
326 i, mpParms[i].u.uint64));
327#endif
328 paDstParms[i].u.uint64 = mpParms[i].u.uint64;
329 break;
330
331 case VBOX_HGCM_SVC_PARM_PTR:
332 {
333#ifdef DEBUG_andy
334 LogFlowFunc(("\tmpParms[%RU32] = %p (ptr), size = %RU32\n",
335 i, mpParms[i].u.pointer.addr, mpParms[i].u.pointer.size));
336#endif
337 if (!mpParms[i].u.pointer.size)
338 continue; /* Only copy buffer if there actually is something to copy. */
339
340 if (!paDstParms[i].u.pointer.addr)
341 rc = VERR_INVALID_PARAMETER;
342
343 if ( RT_SUCCESS(rc)
344 && paDstParms[i].u.pointer.size < mpParms[i].u.pointer.size)
345 rc = VERR_BUFFER_OVERFLOW;
346
347 if (RT_SUCCESS(rc))
348 {
349 memcpy(paDstParms[i].u.pointer.addr,
350 mpParms[i].u.pointer.addr,
351 mpParms[i].u.pointer.size);
352 }
353
354 break;
355 }
356
357 default:
358 LogFlowFunc(("Parameter %RU32 of type %RU32 is not supported yet\n",
359 i, mpParms[i].type));
360 rc = VERR_NOT_SUPPORTED;
361 break;
362 }
363 }
364
365 if (RT_FAILURE(rc))
366 {
367 LogFlowFunc(("Parameter %RU32 invalid (%Rrc), refusing\n",
368 i, rc));
369 break;
370 }
371 }
372 }
373
374 LogFlowFunc(("Returned with rc=%Rrc\n", rc));
375 return rc;
376 }
377
378 int Assign(const ClientConnection *pConnection)
379 {
380 AssertPtrReturn(pConnection, VERR_INVALID_POINTER);
381
382 int rc;
383
384 LogFlowFunc(("pHostCmd=%p, mMsgType=%RU32, mParmCount=%RU32, mpParms=%p\n",
385 this, mMsgType, mParmCount, mpParms));
386
387 /* Does the current host command need more parameter space which
388 * the client does not provide yet? */
389 if (mParmCount > pConnection->mNumParms)
390 {
391 LogFlowFunc(("pHostCmd=%p requires %RU32 parms, only got %RU32 from client\n",
392 this, mParmCount, pConnection->mNumParms));
393
394 /*
395 * So this call apparently failed because the guest wanted to peek
396 * how much parameters it has to supply in order to successfully retrieve
397 * this command. Let's tell him so!
398 */
399 rc = VERR_TOO_MUCH_DATA;
400 }
401 else
402 {
403 rc = CopyTo(pConnection->mParms, pConnection->mNumParms);
404
405 /*
406 * Has there been enough parameter space but the wrong parameter types
407 * were submitted -- maybe the client was just asking for the next upcoming
408 * host message?
409 *
410 * Note: To keep this compatible to older clients we return VERR_TOO_MUCH_DATA
411 * in every case.
412 */
413 if (RT_FAILURE(rc))
414 rc = VERR_TOO_MUCH_DATA;
415 }
416
417 return rc;
418 }
419
420 int Peek(const ClientConnection *pConnection)
421 {
422 AssertPtrReturn(pConnection, VERR_INVALID_POINTER);
423
424 LogFlowFunc(("pHostCmd=%p, mMsgType=%RU32, mParmCount=%RU32, mpParms=%p\n",
425 this, mMsgType, mParmCount, mpParms));
426
427 if (pConnection->mNumParms >= 2)
428 {
429 pConnection->mParms[0].setUInt32(mMsgType); /* Message ID */
430 pConnection->mParms[1].setUInt32(mParmCount); /* Required parameters for message */
431 }
432 else
433 LogFlowFunc(("Warning: Client has not (yet) submitted enough parameters (%RU32, must be at least 2) to at least peak for the next message\n",
434 pConnection->mNumParms));
435
436 /*
437 * Always return VERR_TOO_MUCH_DATA data here to
438 * keep it compatible with older clients and to
439 * have correct accounting (mHostRc + mHostCmdTries).
440 */
441 return VERR_TOO_MUCH_DATA;
442 }
443
444 /** Reference count for keeping track how many connected
445 * clients still need to process this command until it can
446 * be removed. */
447 uint32_t mRefCount;
448 /** The context ID this command belongs to. Will be extracted
449 * *always* from HGCM parameter [0]. */
450 uint32_t mContextID;
451 /** Dynamic structure for holding the HGCM parms */
452 uint32_t mMsgType;
453 /** Number of HGCM parameters. */
454 uint32_t mParmCount;
455 /** Array of HGCM parameters. */
456 PVBOXHGCMSVCPARM mpParms;
457 /** Incoming timestamp (us). */
458 uint64_t mTimestamp;
459} HostCommand;
460typedef std::list< HostCommand *> HostCmdList;
461typedef std::list< HostCommand *>::iterator HostCmdListIter;
462typedef std::list< HostCommand *>::const_iterator HostCmdListIterConst;
463
464/**
465 * Per-client structure used for book keeping/state tracking a
466 * certain host command.
467 */
468typedef struct ClientContext
469{
470 /* Pointer to list node of this command. */
471 HostCommand *mpHostCmd;
472 /** The standard constructor. */
473 ClientContext(void) : mpHostCmd(NULL) {}
474 /** Internal constrcutor. */
475 ClientContext(HostCommand *pHostCmd) : mpHostCmd(pHostCmd) {}
476} ClientContext;
477typedef std::map< uint32_t, ClientContext > ClientContextMap;
478typedef std::map< uint32_t, ClientContext >::iterator ClientContextMapIter;
479typedef std::map< uint32_t, ClientContext >::const_iterator ClientContextMapIterConst;
480
481/**
482 * Structure for holding a connected guest client
483 * state.
484 */
485typedef struct ClientState
486{
487 ClientState(void)
488 : mSvcHelpers(NULL),
489 mID(0),
490 mFlags(0),
491 mFilterMask(0), mFilterValue(0),
492 mHostCmdRc(VINF_SUCCESS), mHostCmdTries(0),
493 mHostCmdTS(0),
494 mIsPending(false),
495 mPeekCount(0) { }
496
497 ClientState(PVBOXHGCMSVCHELPERS pSvcHelpers, uint32_t uClientID)
498 : mSvcHelpers(pSvcHelpers),
499 mID(uClientID),
500 mFlags(0),
501 mFilterMask(0), mFilterValue(0),
502 mHostCmdRc(VINF_SUCCESS), mHostCmdTries(0),
503 mHostCmdTS(0),
504 mIsPending(false),
505 mPeekCount(0){ }
506
507 void DequeueAll(void)
508 {
509 HostCmdListIter curItem = mHostCmdList.begin();
510 while (curItem != mHostCmdList.end())
511 curItem = Dequeue(curItem);
512 }
513
514 void DequeueCurrent(void)
515 {
516 HostCmdListIter curCmd = mHostCmdList.begin();
517 if (curCmd != mHostCmdList.end())
518 Dequeue(curCmd);
519 }
520
521 HostCmdListIter Dequeue(HostCmdListIter &curItem)
522 {
523 HostCommand *pHostCmd = (*curItem);
524 AssertPtr(pHostCmd);
525
526 if (pHostCmd->Release() == 0)
527 {
528 LogFlowFunc(("[Client %RU32] Destroying pHostCmd=%p\n",
529 mID, (*curItem)));
530
531 delete pHostCmd;
532 pHostCmd = NULL;
533 }
534
535 HostCmdListIter nextItem = mHostCmdList.erase(curItem);
536
537 /* Reset everything else. */
538 mHostCmdRc = VINF_SUCCESS;
539 mHostCmdTries = 0;
540 mPeekCount = 0;
541
542 return nextItem;
543 }
544
545 int EnqueueCommand(HostCommand *pHostCmd)
546 {
547 AssertPtrReturn(pHostCmd, VERR_INVALID_POINTER);
548
549 int rc = VINF_SUCCESS;
550
551 try
552 {
553 mHostCmdList.push_back(pHostCmd);
554 pHostCmd->AddRef();
555 }
556 catch (std::bad_alloc)
557 {
558 rc = VERR_NO_MEMORY;
559 }
560
561 return rc;
562 }
563
564 bool WantsHostCommand(const HostCommand *pHostCmd) const
565 {
566 AssertPtrReturn(pHostCmd, false);
567
568#ifdef DEBUG_andy
569 LogFlowFunc(("mHostCmdTS=%RU64, pHostCmdTS=%RU64\n",
570 mHostCmdTS, pHostCmd->mTimestamp));
571#endif
572
573 /* Only process newer commands. */
574 if (pHostCmd->mTimestamp <= mHostCmdTS)
575 return false;
576
577 /*
578 * If a sesseion filter is set, only obey those commands we're interested in
579 * by applying our context ID filter mask and compare the result with the
580 * original context ID.
581 */
582 bool fWant;
583 if (mFlags & CLIENTSTATE_FLAG_CONTEXTFILTER)
584 {
585 fWant = (pHostCmd->mContextID & mFilterMask) == mFilterValue;
586 }
587 else /* Client is interested in all commands. */
588 fWant = true;
589
590 LogFlowFunc(("[Client %RU32] mFlags=0x%x, mContextID=%RU32 (session %RU32), mFilterMask=0x%x, mFilterValue=%RU32, fWant=%RTbool\n",
591 mID, mFlags, pHostCmd->mContextID,
592 VBOX_GUESTCTRL_CONTEXTID_GET_SESSION(pHostCmd->mContextID),
593 mFilterMask, mFilterValue, fWant));
594
595 return fWant;
596 }
597
598 int SetPending(const ClientConnection *pConnection)
599 {
600 AssertPtrReturn(pConnection, VERR_INVALID_POINTER);
601
602 if (mIsPending)
603 {
604 LogFlowFunc(("[Client %RU32] Already is in pending mode\n", mID));
605
606 /*
607 * Signal that we don't and can't return yet.
608 */
609 return VINF_HGCM_ASYNC_EXECUTE;
610 }
611
612 if (mHostCmdList.empty())
613 {
614 AssertMsg(mIsPending == false,
615 ("Client ID=%RU32 already is pending but tried to receive a new host command\n", mID));
616
617 mPendingCon.mHandle = pConnection->mHandle;
618 mPendingCon.mNumParms = pConnection->mNumParms;
619 mPendingCon.mParms = pConnection->mParms;
620
621 mIsPending = true;
622
623 LogFlowFunc(("[Client %RU32] Is now in pending mode\n", mID));
624
625 /*
626 * Signal that we don't and can't return yet.
627 */
628 return VINF_HGCM_ASYNC_EXECUTE;
629 }
630
631 /*
632 * Signal that there already is a connection pending.
633 * Shouldn't happen in daily usage.
634 */
635 AssertMsgFailed(("Client already has a connection pending\n"));
636 return VERR_SIGNAL_PENDING;
637 }
638
639 int Run(const ClientConnection *pConnection,
640 HostCommand *pHostCmd)
641 {
642 AssertPtrReturn(pConnection, VERR_INVALID_POINTER);
643 AssertPtrReturn(pHostCmd, VERR_INVALID_POINTER);
644
645 int rc = VINF_SUCCESS;
646
647 LogFlowFunc(("[Client %RU32] pConnection=%p, mHostCmdRc=%Rrc, mHostCmdTries=%RU32, mPeekCount=%RU32\n",
648 mID, pConnection, mHostCmdRc, mHostCmdTries, mPeekCount));
649
650 mHostCmdRc = SendReply(pConnection, pHostCmd);
651 LogFlowFunc(("[Client %RU32] Processing pHostCmd=%p ended with rc=%Rrc\n",
652 mID, pHostCmd, mHostCmdRc));
653
654 bool fRemove = false;
655 if (RT_FAILURE(mHostCmdRc))
656 {
657 mHostCmdTries++;
658
659 /*
660 * If the client understood the message but supplied too little buffer space
661 * don't send this message again and drop it after 6 unsuccessful attempts.
662 *
663 * Note: Due to legacy reasons this the retry counter has to be even because on
664 * every peek there will be the actual command retrieval from the client side.
665 * To not get the actual command if the client actually only wants to peek for
666 * the next command, there needs to be two rounds per try, e.g. 3 rounds = 6 tries.
667 *
668 ** @todo Fix the mess stated above. GUEST_MSG_WAIT should be become GUEST_MSG_PEEK, *only*
669 * (and every time) returning the next upcoming host command (if any, blocking). Then
670 * it's up to the client what to do next, either peeking again or getting the actual
671 * host command via an own GUEST_ type message.
672 */
673 if (mHostCmdRc == VERR_TOO_MUCH_DATA)
674 {
675 if (mHostCmdTries == 6)
676 fRemove = true;
677 }
678 /* Client did not understand the message or something else weird happened. Try again one
679 * more time and drop it if it didn't get handled then. */
680 else if (mHostCmdTries > 1)
681 fRemove = true;
682 }
683 else
684 fRemove = true; /* Everything went fine, remove it. */
685
686 LogFlowFunc(("[Client %RU32] Tried pHostCmd=%p for %RU32 times, (last result=%Rrc, fRemove=%RTbool)\n",
687 mID, pHostCmd, mHostCmdTries, mHostCmdRc, fRemove));
688
689 if (RT_SUCCESS(rc))
690 rc = mHostCmdRc;
691
692 if (fRemove)
693 {
694 /** @todo Fix this (slow) lookup. Too late today. */
695 HostCmdListIter curItem = mHostCmdList.begin();
696 while (curItem != mHostCmdList.end())
697 {
698 if ((*curItem) == pHostCmd)
699 {
700 Dequeue(curItem);
701 break;
702 }
703
704 ++curItem;
705 }
706 }
707
708 LogFlowFunc(("[Client %RU32] Returned with rc=%Rrc\n", mID, rc));
709 return rc;
710 }
711
712 int RunCurrent(const ClientConnection *pConnection)
713 {
714 AssertPtrReturn(pConnection, VERR_INVALID_POINTER);
715
716 int rc;
717 if (mHostCmdList.empty())
718 {
719 rc = SetPending(pConnection);
720 }
721 else
722 {
723 AssertMsgReturn(!mIsPending,
724 ("Client ID=%RU32 still is in pending mode; can't use another connection\n", mID), VERR_INVALID_PARAMETER);
725
726 HostCmdListIter curCmd = mHostCmdList.begin();
727 Assert(curCmd != mHostCmdList.end());
728 HostCommand *pHostCmd = (*curCmd);
729 AssertPtrReturn(pHostCmd, VERR_INVALID_POINTER);
730
731 rc = Run(pConnection, pHostCmd);
732 }
733
734 return rc;
735 }
736
737 int Wakeup(void)
738 {
739 int rc = VINF_NO_CHANGE;
740
741 if (mIsPending)
742 {
743 LogFlowFunc(("[Client %RU32] Waking up ...\n", mID));
744
745 rc = VINF_SUCCESS;
746
747 HostCmdListIter curCmd = mHostCmdList.begin();
748 if (curCmd != mHostCmdList.end())
749 {
750 HostCommand *pHostCmd = (*curCmd);
751 AssertPtrReturn(pHostCmd, VERR_INVALID_POINTER);
752
753 LogFlowFunc(("[Client %RU32] Current host command is pHostCmd=%p, CID=%RU32, cmdType=%RU32, cmdParms=%RU32, refCount=%RU32\n",
754 mID, pHostCmd, pHostCmd->mContextID, pHostCmd->mMsgType, pHostCmd->mParmCount, pHostCmd->mRefCount));
755
756 rc = Run(&mPendingCon, pHostCmd);
757 }
758 else
759 AssertMsgFailed(("Waking up client ID=%RU32 with no host command in queue is a bad idea\n", mID));
760
761 return rc;
762 }
763
764 return VINF_NO_CHANGE;
765 }
766
767 int CancelWaiting(int rcPending)
768 {
769 LogFlowFunc(("[Client %RU32] Cancelling waiting with %Rrc, isPending=%RTbool, pendingNumParms=%RU32, flags=%x\n",
770 mID, rcPending, mIsPending, mPendingCon.mNumParms, mFlags));
771
772 int rc;
773 if ( mIsPending
774 && mPendingCon.mNumParms >= 2)
775 {
776 mPendingCon.mParms[0].setUInt32(HOST_CANCEL_PENDING_WAITS); /* Message ID. */
777 mPendingCon.mParms[1].setUInt32(0); /* Required parameters for message. */
778
779 AssertPtr(mSvcHelpers);
780 mSvcHelpers->pfnCallComplete(mPendingCon.mHandle, rcPending);
781
782 mIsPending = false;
783
784 rc = VINF_SUCCESS;
785 }
786 else if (mPendingCon.mNumParms < 2)
787 rc = VERR_BUFFER_OVERFLOW;
788 else /** @todo Enqueue command instead of dropping? */
789 rc = VERR_WRONG_ORDER;
790
791 return rc;
792 }
793
794 int SendReply(const ClientConnection *pConnection,
795 HostCommand *pHostCmd)
796 {
797 AssertPtrReturn(pConnection, VERR_INVALID_POINTER);
798 AssertPtrReturn(pHostCmd, VERR_INVALID_POINTER);
799
800 int rc;
801 /* If the client is in pending mode, always send back
802 * the peek result first. */
803 if (mIsPending)
804 {
805 rc = pHostCmd->Peek(pConnection);
806 mPeekCount++;
807 }
808 else
809 {
810 /* If this is the very first peek, make sure to *always* give back the peeking answer
811 * instead of the actual command, even if this command would fit into the current
812 * connection buffer. */
813 if (!mPeekCount)
814 {
815 rc = pHostCmd->Peek(pConnection);
816 mPeekCount++;
817 }
818 else
819 {
820 /* Try assigning the host command to the client and store the
821 * result code for later use. */
822 rc = pHostCmd->Assign(pConnection);
823 if (RT_FAILURE(rc)) /* If something failed, let the client peek (again). */
824 {
825 rc = pHostCmd->Peek(pConnection);
826 mPeekCount++;
827 }
828 else
829 mPeekCount = 0;
830 }
831 }
832
833 /* Reset pending status. */
834 mIsPending = false;
835
836 /* In any case the client did something, so complete
837 * the pending call with the result we just got. */
838 AssertPtr(mSvcHelpers);
839 mSvcHelpers->pfnCallComplete(pConnection->mHandle, rc);
840
841 LogFlowFunc(("[Client %RU32] mPeekCount=%RU32, pConnection=%p, pHostCmd=%p, replyRc=%Rrc\n",
842 mID, mPeekCount, pConnection, pHostCmd, rc));
843 return rc;
844 }
845
846 PVBOXHGCMSVCHELPERS mSvcHelpers;
847 /** The client's ID. */
848 uint32_t mID;
849 /** Client flags. @sa CLIENTSTATE_FLAG_ flags. */
850 uint32_t mFlags;
851 /** The context ID filter mask, if any. */
852 uint32_t mFilterMask;
853 /** The context ID filter value, if any. */
854 uint32_t mFilterValue;
855 /** Host command list to process. */
856 HostCmdList mHostCmdList;
857 /** Last (most recent) rc after handling the
858 * host command. */
859 int mHostCmdRc;
860 /** How many times the host service has tried to deliver this
861 * command to the according client. */
862 uint32_t mHostCmdTries;
863 /** Timestamp (us) of last host command processed. */
864 uint64_t mHostCmdTS;
865 /**
866 * Flag indicating whether the client currently is pending.
867 * This means the client waits for a new host command to reply
868 * and won't return from the waiting call until a new host
869 * command is available.
870 */
871 bool mIsPending;
872 /**
873 * This is necessary for being compatible with older
874 * Guest Additions. In case there are commands which only
875 * have two (2) parameters and therefore would fit into the
876 * GUEST_MSG_WAIT reply immediately, we now can make sure
877 * that the client first gets back the GUEST_MSG_WAIT results
878 * first.
879 */
880 uint32_t mPeekCount;
881 /** The client's pending connection. */
882 ClientConnection mPendingCon;
883} ClientState;
884typedef std::map< uint32_t, ClientState > ClientStateMap;
885typedef std::map< uint32_t, ClientState >::iterator ClientStateMapIter;
886typedef std::map< uint32_t, ClientState >::const_iterator ClientStateMapIterConst;
887
888/**
889 * Class containing the shared information service functionality.
890 */
891class Service : public RTCNonCopyable
892{
893
894private:
895
896 /** Type definition for use in callback functions. */
897 typedef Service SELF;
898 /** HGCM helper functions. */
899 PVBOXHGCMSVCHELPERS mpHelpers;
900 /**
901 * Callback function supplied by the host for notification of updates
902 * to properties.
903 */
904 PFNHGCMSVCEXT mpfnHostCallback;
905 /** User data pointer to be supplied to the host callback function. */
906 void *mpvHostData;
907 /** List containing all buffered host commands. */
908 RTLISTANCHOR mHostCmdList;
909 /** Map containing all connected clients. The primary key contains
910 * the HGCM client ID to identify the client. */
911 ClientStateMap mClientStateMap;
912public:
913 explicit Service(PVBOXHGCMSVCHELPERS pHelpers)
914 : mpHelpers(pHelpers)
915 , mpfnHostCallback(NULL)
916 , mpvHostData(NULL)
917 {
918 RTListInit(&mHostCmdList);
919 }
920
921 /**
922 * @interface_method_impl{VBOXHGCMSVCFNTABLE,pfnUnload}
923 * Simply deletes the service object
924 */
925 static DECLCALLBACK(int) svcUnload(void *pvService)
926 {
927 AssertLogRelReturn(VALID_PTR(pvService), VERR_INVALID_PARAMETER);
928 SELF *pSelf = reinterpret_cast<SELF *>(pvService);
929 int rc = pSelf->uninit();
930 AssertRC(rc);
931 if (RT_SUCCESS(rc))
932 delete pSelf;
933 return rc;
934 }
935
936 /**
937 * @interface_method_impl{VBOXHGCMSVCFNTABLE,pfnConnect}
938 * Stub implementation of pfnConnect and pfnDisconnect.
939 */
940 static DECLCALLBACK(int) svcConnect(void *pvService,
941 uint32_t u32ClientID,
942 void *pvClient)
943 {
944 AssertLogRelReturn(VALID_PTR(pvService), VERR_INVALID_PARAMETER);
945 SELF *pSelf = reinterpret_cast<SELF *>(pvService);
946 AssertPtrReturn(pSelf, VERR_INVALID_POINTER);
947 return pSelf->clientConnect(u32ClientID, pvClient);
948 }
949
950 /**
951 * @interface_method_impl{VBOXHGCMSVCFNTABLE,pfnConnect}
952 * Stub implementation of pfnConnect and pfnDisconnect.
953 */
954 static DECLCALLBACK(int) svcDisconnect(void *pvService,
955 uint32_t u32ClientID,
956 void *pvClient)
957 {
958 AssertLogRelReturn(VALID_PTR(pvService), VERR_INVALID_PARAMETER);
959 SELF *pSelf = reinterpret_cast<SELF *>(pvService);
960 AssertPtrReturn(pSelf, VERR_INVALID_POINTER);
961 return pSelf->clientDisconnect(u32ClientID, pvClient);
962 }
963
964 /**
965 * @interface_method_impl{VBOXHGCMSVCFNTABLE,pfnCall}
966 * Wraps to the call member function
967 */
968 static DECLCALLBACK(void) svcCall(void * pvService,
969 VBOXHGCMCALLHANDLE callHandle,
970 uint32_t u32ClientID,
971 void *pvClient,
972 uint32_t u32Function,
973 uint32_t cParms,
974 VBOXHGCMSVCPARM paParms[])
975 {
976 AssertLogRelReturnVoid(VALID_PTR(pvService));
977 SELF *pSelf = reinterpret_cast<SELF *>(pvService);
978 AssertPtrReturnVoid(pSelf);
979 pSelf->call(callHandle, u32ClientID, pvClient, u32Function, cParms, paParms);
980 }
981
982 /**
983 * @interface_method_impl{VBOXHGCMSVCFNTABLE,pfnHostCall}
984 * Wraps to the hostCall member function
985 */
986 static DECLCALLBACK(int) svcHostCall(void *pvService,
987 uint32_t u32Function,
988 uint32_t cParms,
989 VBOXHGCMSVCPARM paParms[])
990 {
991 AssertLogRelReturn(VALID_PTR(pvService), VERR_INVALID_PARAMETER);
992 SELF *pSelf = reinterpret_cast<SELF *>(pvService);
993 AssertPtrReturn(pSelf, VERR_INVALID_POINTER);
994 return pSelf->hostCall(u32Function, cParms, paParms);
995 }
996
997 /**
998 * @interface_method_impl{VBOXHGCMSVCFNTABLE,pfnRegisterExtension}
999 * Installs a host callback for notifications of property changes.
1000 */
1001 static DECLCALLBACK(int) svcRegisterExtension(void *pvService,
1002 PFNHGCMSVCEXT pfnExtension,
1003 void *pvExtension)
1004 {
1005 AssertLogRelReturn(VALID_PTR(pvService), VERR_INVALID_PARAMETER);
1006 SELF *pSelf = reinterpret_cast<SELF *>(pvService);
1007 AssertPtrReturn(pSelf, VERR_INVALID_POINTER);
1008 pSelf->mpfnHostCallback = pfnExtension;
1009 pSelf->mpvHostData = pvExtension;
1010 return VINF_SUCCESS;
1011 }
1012
1013private:
1014
1015 int prepareExecute(uint32_t cParms, VBOXHGCMSVCPARM paParms[]);
1016 int clientConnect(uint32_t u32ClientID, void *pvClient);
1017 int clientDisconnect(uint32_t u32ClientID, void *pvClient);
1018 int clientGetCommand(uint32_t u32ClientID, VBOXHGCMCALLHANDLE callHandle, uint32_t cParms, VBOXHGCMSVCPARM paParms[]);
1019 int clientSetMsgFilterSet(uint32_t u32ClientID, VBOXHGCMCALLHANDLE callHandle, uint32_t cParms, VBOXHGCMSVCPARM paParms[]);
1020 int clientSetMsgFilterUnset(uint32_t u32ClientID, VBOXHGCMCALLHANDLE callHandle, uint32_t cParms, VBOXHGCMSVCPARM paParms[]);
1021 int clientSkipMsg(uint32_t u32ClientID, VBOXHGCMCALLHANDLE callHandle, uint32_t cParms, VBOXHGCMSVCPARM paParms[]);
1022 int cancelHostCmd(uint32_t u32ContextID);
1023 int cancelPendingWaits(uint32_t u32ClientID, int rcPending);
1024 int hostCallback(uint32_t eFunction, uint32_t cParms, VBOXHGCMSVCPARM paParms[]);
1025 int hostProcessCommand(uint32_t eFunction, uint32_t cParms, VBOXHGCMSVCPARM paParms[]);
1026 void call(VBOXHGCMCALLHANDLE callHandle, uint32_t u32ClientID, void *pvClient, uint32_t eFunction, uint32_t cParms, VBOXHGCMSVCPARM paParms[]);
1027 int hostCall(uint32_t eFunction, uint32_t cParms, VBOXHGCMSVCPARM paParms[]);
1028 int sessionClose(uint32_t u32ClientID, VBOXHGCMCALLHANDLE callHandle, uint32_t cParms, VBOXHGCMSVCPARM paParms[]);
1029 int uninit(void);
1030};
1031
1032/**
1033 * Handles a client which just connected.
1034 *
1035 * @return IPRT status code.
1036 * @param u32ClientID
1037 * @param pvClient
1038 */
1039int Service::clientConnect(uint32_t u32ClientID, void *pvClient)
1040{
1041 LogFlowFunc(("[Client %RU32] Connected\n", u32ClientID));
1042#ifdef VBOX_STRICT
1043 ClientStateMapIterConst it = mClientStateMap.find(u32ClientID);
1044 if (it != mClientStateMap.end())
1045 {
1046 AssertMsgFailed(("Client with ID=%RU32 already connected when it should not\n",
1047 u32ClientID));
1048 return VERR_ALREADY_EXISTS;
1049 }
1050#endif
1051 ClientState clientState(mpHelpers, u32ClientID);
1052 mClientStateMap[u32ClientID] = clientState;
1053 /** @todo Exception handling! */
1054 return VINF_SUCCESS;
1055}
1056
1057/**
1058 * Handles a client which disconnected. This functiond does some
1059 * internal cleanup as well as sends notifications to the host so
1060 * that the host can do the same (if required).
1061 *
1062 * @return IPRT status code.
1063 * @param u32ClientID The client's ID of which disconnected.
1064 * @param pvClient User data, not used at the moment.
1065 */
1066int Service::clientDisconnect(uint32_t u32ClientID, void *pvClient)
1067{
1068 LogFlowFunc(("[Client %RU32] Disconnected (%zu clients total)\n",
1069 u32ClientID, mClientStateMap.size()));
1070
1071 AssertMsg(mClientStateMap.size(),
1072 ("No clients in list anymore when there should (client ID=%RU32)\n", u32ClientID));
1073
1074 int rc = VINF_SUCCESS;
1075
1076 ClientStateMapIter itClientState = mClientStateMap.find(u32ClientID);
1077 AssertMsg(itClientState != mClientStateMap.end(),
1078 ("Client ID=%RU32 not found in client list when it should be there\n", u32ClientID));
1079
1080 if (itClientState != mClientStateMap.end())
1081 {
1082 itClientState->second.DequeueAll();
1083
1084 mClientStateMap.erase(itClientState);
1085 }
1086
1087 bool fAllClientsDisconnected = mClientStateMap.empty();
1088 if (fAllClientsDisconnected)
1089 {
1090 LogFlowFunc(("All clients disconnected, cancelling all host commands ...\n"));
1091
1092 /*
1093 * If all clients disconnected we also need to make sure that all buffered
1094 * host commands need to be notified, because Main is waiting a notification
1095 * via a (multi stage) progress object.
1096 */
1097 HostCommand *pCurCmd = RTListGetFirst(&mHostCmdList, HostCommand, Node);
1098 while (pCurCmd)
1099 {
1100 HostCommand *pNext = RTListNodeGetNext(&pCurCmd->Node, HostCommand, Node);
1101 bool fLast = RTListNodeIsLast(&mHostCmdList, &pCurCmd->Node);
1102
1103 int rc2 = cancelHostCmd(pCurCmd->mContextID);
1104 if (RT_FAILURE(rc2))
1105 {
1106 LogFlowFunc(("Cancelling host command with CID=%u (refCount=%RU32) failed with rc=%Rrc\n",
1107 pCurCmd->mContextID, pCurCmd->mRefCount, rc2));
1108 /* Keep going. */
1109 }
1110
1111 while (pCurCmd->Release())
1112 ;
1113 delete pCurCmd;
1114 pCurCmd = NULL;
1115
1116 if (fLast)
1117 break;
1118
1119 pCurCmd = pNext;
1120 }
1121
1122 Assert(RTListIsEmpty(&mHostCmdList));
1123 }
1124
1125 return rc;
1126}
1127
1128/**
1129 * Either fills in parameters from a pending host command into our guest context or
1130 * defer the guest call until we have something from the host.
1131 *
1132 * @return IPRT status code.
1133 * @param u32ClientID The client's ID.
1134 * @param callHandle The client's call handle.
1135 * @param cParms Number of parameters.
1136 * @param paParms Array of parameters.
1137 */
1138int Service::clientGetCommand(uint32_t u32ClientID, VBOXHGCMCALLHANDLE callHandle,
1139 uint32_t cParms, VBOXHGCMSVCPARM paParms[])
1140{
1141 /*
1142 * Lookup client in our list so that we can assign the context ID of
1143 * a command to that client.
1144 */
1145 ClientStateMapIter itClientState = mClientStateMap.find(u32ClientID);
1146 AssertMsg(itClientState != mClientStateMap.end(), ("Client with ID=%RU32 not found when it should be present\n",
1147 u32ClientID));
1148 if (itClientState == mClientStateMap.end())
1149 {
1150 /* Should never happen. Complete the call on the guest side though. */
1151 AssertPtr(mpHelpers);
1152 mpHelpers->pfnCallComplete(callHandle, VERR_NOT_FOUND);
1153
1154 return VERR_NOT_FOUND;
1155 }
1156
1157 ClientState &clientState = itClientState->second;
1158
1159 /* Use the current (inbound) connection. */
1160 ClientConnection thisCon;
1161 thisCon.mHandle = callHandle;
1162 thisCon.mNumParms = cParms;
1163 thisCon.mParms = paParms;
1164
1165 return clientState.RunCurrent(&thisCon);
1166}
1167
1168int Service::clientSetMsgFilterSet(uint32_t u32ClientID, VBOXHGCMCALLHANDLE callHandle,
1169 uint32_t cParms, VBOXHGCMSVCPARM paParms[])
1170{
1171 /*
1172 * Lookup client in our list so that we can assign the context ID of
1173 * a command to that client.
1174 */
1175 ClientStateMapIter itClientState = mClientStateMap.find(u32ClientID);
1176 AssertMsg(itClientState != mClientStateMap.end(), ("Client with ID=%RU32 not found when it should be present\n",
1177 u32ClientID));
1178 if (itClientState == mClientStateMap.end())
1179 return VERR_NOT_FOUND; /* Should never happen. */
1180
1181 if (cParms != 4)
1182 return VERR_INVALID_PARAMETER;
1183
1184 uint32_t uValue, uMaskAdd, uMaskRemove;
1185 int rc = paParms[0].getUInt32(&uValue);
1186 if (RT_SUCCESS(rc))
1187 rc = paParms[1].getUInt32(&uMaskAdd);
1188 if (RT_SUCCESS(rc))
1189 rc = paParms[2].getUInt32(&uMaskRemove);
1190 /** @todo paParm[3] (flags) not used yet. */
1191 if (RT_SUCCESS(rc))
1192 {
1193 ClientState &clientState = itClientState->second;
1194
1195 clientState.mFlags |= CLIENTSTATE_FLAG_CONTEXTFILTER;
1196 if (uMaskAdd)
1197 clientState.mFilterMask |= uMaskAdd;
1198 if (uMaskRemove)
1199 clientState.mFilterMask &= ~uMaskRemove;
1200
1201 clientState.mFilterValue = uValue;
1202
1203 LogFlowFunc(("[Client %RU32] Setting message filterMask=0x%x, filterVal=%RU32 set (flags=0x%x, maskAdd=0x%x, maskRemove=0x%x)\n",
1204 u32ClientID, clientState.mFilterMask, clientState.mFilterValue,
1205 clientState.mFlags, uMaskAdd, uMaskRemove));
1206 }
1207
1208 return rc;
1209}
1210
1211int Service::clientSetMsgFilterUnset(uint32_t u32ClientID, VBOXHGCMCALLHANDLE callHandle,
1212 uint32_t cParms, VBOXHGCMSVCPARM paParms[])
1213{
1214 /*
1215 * Lookup client in our list so that we can assign the context ID of
1216 * a command to that client.
1217 */
1218 ClientStateMapIter itClientState = mClientStateMap.find(u32ClientID);
1219 AssertMsg(itClientState != mClientStateMap.end(), ("Client with ID=%RU32 not found when it should be present\n",
1220 u32ClientID));
1221 if (itClientState == mClientStateMap.end())
1222 return VERR_NOT_FOUND; /* Should never happen. */
1223
1224 if (cParms != 1)
1225 return VERR_INVALID_PARAMETER;
1226
1227 ClientState &clientState = itClientState->second;
1228
1229 clientState.mFlags &= ~CLIENTSTATE_FLAG_CONTEXTFILTER;
1230 clientState.mFilterMask = 0;
1231 clientState.mFilterValue = 0;
1232
1233 LogFlowFunc(("[Client %RU32} Unset message filter\n", u32ClientID));
1234 return VINF_SUCCESS;
1235}
1236
1237int Service::clientSkipMsg(uint32_t u32ClientID, VBOXHGCMCALLHANDLE callHandle,
1238 uint32_t cParms, VBOXHGCMSVCPARM paParms[])
1239{
1240 /*
1241 * Lookup client in our list so that we can assign the context ID of
1242 * a command to that client.
1243 */
1244 ClientStateMapIter itClientState = mClientStateMap.find(u32ClientID);
1245 AssertMsg(itClientState != mClientStateMap.end(), ("Client ID=%RU32 not found when it should be present\n",
1246 u32ClientID));
1247 if (itClientState == mClientStateMap.end())
1248 return VERR_NOT_FOUND; /* Should never happen. */
1249
1250 if (cParms != 1)
1251 return VERR_INVALID_PARAMETER;
1252
1253 LogFlowFunc(("[Client %RU32] Skipping current message ...\n", u32ClientID));
1254
1255 itClientState->second.DequeueCurrent();
1256
1257 return VINF_SUCCESS;
1258}
1259
1260/**
1261 * Cancels a buffered host command to unblock waiting on Main side
1262 * via callbacks.
1263 *
1264 * @return IPRT status code.
1265 * @param u32ContextID Context ID of host command to cancel.
1266 */
1267int Service::cancelHostCmd(uint32_t u32ContextID)
1268{
1269 Assert(mpfnHostCallback);
1270
1271 LogFlowFunc(("Cancelling CID=%u ...\n", u32ContextID));
1272
1273 uint32_t cParms = 0;
1274 VBOXHGCMSVCPARM arParms[2];
1275 arParms[cParms++].setUInt32(u32ContextID);
1276
1277 return hostCallback(GUEST_DISCONNECTED, cParms, arParms);
1278}
1279
1280/**
1281 * Client asks itself (in another thread) to cancel all pending waits which are blocking the client
1282 * from shutting down / doing something else.
1283 *
1284 * @return IPRT status code.
1285 * @param u32ClientID The client's ID.
1286 * @param rcPending Result code for completing pending operation.
1287 */
1288int Service::cancelPendingWaits(uint32_t u32ClientID, int rcPending)
1289{
1290 ClientStateMapIter itClientState = mClientStateMap.find(u32ClientID);
1291 if (itClientState != mClientStateMap.end())
1292 return itClientState->second.CancelWaiting(rcPending);
1293
1294 return VINF_SUCCESS;
1295}
1296
1297/**
1298 * Notifies the host (using low-level HGCM callbacks) about an event
1299 * which was sent from the client.
1300 *
1301 * @return IPRT status code.
1302 * @param eFunction Function (event) that occured.
1303 * @param cParms Number of parameters.
1304 * @param paParms Array of parameters.
1305 */
1306int Service::hostCallback(uint32_t eFunction, uint32_t cParms, VBOXHGCMSVCPARM paParms[])
1307{
1308 LogFlowFunc(("eFunction=%ld, cParms=%ld, paParms=%p\n",
1309 eFunction, cParms, paParms));
1310
1311 int rc;
1312 if (mpfnHostCallback)
1313 {
1314 VBOXGUESTCTRLHOSTCALLBACK data(cParms, paParms);
1315 rc = mpfnHostCallback(mpvHostData, eFunction,
1316 (void *)(&data), sizeof(data));
1317 }
1318 else
1319 rc = VERR_NOT_SUPPORTED;
1320
1321 LogFlowFunc(("Returning rc=%Rrc\n", rc));
1322 return rc;
1323}
1324
1325/**
1326 * Processes a command received from the host side and re-routes it to
1327 * a connect client on the guest.
1328 *
1329 * @return IPRT status code.
1330 * @param eFunction Function code to process.
1331 * @param cParms Number of parameters.
1332 * @param paParms Array of parameters.
1333 */
1334int Service::hostProcessCommand(uint32_t eFunction, uint32_t cParms, VBOXHGCMSVCPARM paParms[])
1335{
1336 /*
1337 * If no client is connected at all we don't buffer any host commands
1338 * and immediately return an error to the host. This avoids the host
1339 * waiting for a response from the guest side in case VBoxService on
1340 * the guest is not running/system is messed up somehow.
1341 */
1342 if (mClientStateMap.empty())
1343 return VERR_NOT_FOUND;
1344
1345 int rc;
1346
1347 HostCommand *pHostCmd = NULL;
1348 try
1349 {
1350 pHostCmd = new HostCommand();
1351 rc = pHostCmd->Allocate(eFunction, cParms, paParms);
1352 if (RT_SUCCESS(rc))
1353 /* rc = */ RTListAppend(&mHostCmdList, &pHostCmd->Node);
1354 }
1355 catch (std::bad_alloc)
1356 {
1357 rc = VERR_NO_MEMORY;
1358 }
1359
1360 if (RT_SUCCESS(rc))
1361 {
1362 LogFlowFunc(("Handling host command CID=%RU32, eFunction=%RU32, cParms=%RU32, paParms=%p, numClients=%zu\n",
1363 pHostCmd->mContextID, eFunction, cParms, paParms, mClientStateMap.size()));
1364
1365 /*
1366 * Wake up all pending clients which are interested in this
1367 * host command.
1368 */
1369#ifdef DEBUG
1370 uint32_t uClientsWokenUp = 0;
1371#endif
1372 ClientStateMapIter itClientState = mClientStateMap.begin();
1373 AssertMsg(itClientState != mClientStateMap.end(), ("Client state map is empty when it should not\n"));
1374 while (itClientState != mClientStateMap.end())
1375 {
1376 ClientState &clientState = itClientState->second;
1377
1378 /* If a client indicates that it it wants the new host command,
1379 * add a reference to not delete it.*/
1380 if (clientState.WantsHostCommand(pHostCmd))
1381 {
1382 clientState.EnqueueCommand(pHostCmd);
1383
1384 int rc2 = clientState.Wakeup();
1385 if (RT_FAILURE(rc2))
1386 LogFlowFunc(("Waking up client ID=%RU32 failed with rc=%Rrc\n",
1387 itClientState->first, rc2));
1388#ifdef DEBUG
1389 uClientsWokenUp++;
1390#endif
1391 }
1392
1393 ++itClientState;
1394 }
1395
1396#ifdef DEBUG
1397 LogFlowFunc(("%RU32 clients have been woken up\n", uClientsWokenUp));
1398#endif
1399 }
1400
1401 return rc;
1402}
1403
1404/**
1405 * @interface_method_impl{VBOXHGCMSVCFNTABLE,pfnCall}
1406 *
1407 * @note All functions which do not involve an unreasonable delay will be
1408 * handled synchronously. If needed, we will add a request handler
1409 * thread in future for those which do.
1410 *
1411 * @thread HGCM
1412 */
1413void Service::call(VBOXHGCMCALLHANDLE callHandle, uint32_t u32ClientID,
1414 void * /* pvClient */, uint32_t eFunction, uint32_t cParms,
1415 VBOXHGCMSVCPARM paParms[])
1416{
1417 int rc = VINF_SUCCESS;
1418 LogFlowFunc(("[Client %RU32] eFunction=%RU32, cParms=%RU32, paParms=0x%p\n",
1419 u32ClientID, eFunction, cParms, paParms));
1420 try
1421 {
1422 /*
1423 * The guest asks the host for the next message to process.
1424 */
1425 if (eFunction == GUEST_MSG_WAIT)
1426 {
1427 LogFlowFunc(("[Client %RU32] GUEST_MSG_GET\n", u32ClientID));
1428 rc = clientGetCommand(u32ClientID, callHandle, cParms, paParms);
1429 }
1430 else
1431 {
1432 switch (eFunction)
1433 {
1434 /*
1435 * A client wants to shut down and asks us (this service) to cancel
1436 * all blocking/pending waits (VINF_HGCM_ASYNC_EXECUTE) so that the
1437 * client can gracefully shut down.
1438 */
1439 case GUEST_CANCEL_PENDING_WAITS:
1440 LogFlowFunc(("[Client %RU32] GUEST_CANCEL_PENDING_WAITS\n", u32ClientID));
1441 rc = cancelPendingWaits(u32ClientID, VINF_SUCCESS /* Pending result */);
1442 break;
1443
1444 /*
1445 * The guest only wants certain messages set by the filter mask(s).
1446 * Since VBox 4.3+.
1447 */
1448 case GUEST_MSG_FILTER_SET:
1449 LogFlowFunc(("[Client %RU32] GUEST_MSG_FILTER_SET\n", u32ClientID));
1450 rc = clientSetMsgFilterSet(u32ClientID, callHandle, cParms, paParms);
1451 break;
1452
1453 /*
1454 * Unsetting the message filter flag.
1455 */
1456 case GUEST_MSG_FILTER_UNSET:
1457 LogFlowFunc(("[Client %RU32] GUEST_MSG_FILTER_UNSET\n", u32ClientID));
1458 rc = clientSetMsgFilterUnset(u32ClientID, callHandle, cParms, paParms);
1459 break;
1460
1461 /*
1462 * The guest only wants skip the currently assigned messages. Neded
1463 * for dropping its assigned reference of the current assigned host
1464 * command in queue.
1465 * Since VBox 4.3+.
1466 */
1467 case GUEST_MSG_SKIP:
1468 LogFlowFunc(("[Client %RU32] GUEST_MSG_SKIP\n", u32ClientID));
1469 rc = clientSkipMsg(u32ClientID, callHandle, cParms, paParms);
1470 break;
1471
1472 /*
1473 * The guest wants to close specific guest session. This is handy for
1474 * shutting down dedicated guest session processes from another process.
1475 */
1476 case GUEST_SESSION_CLOSE:
1477 LogFlowFunc(("[Client %RU32] GUEST_SESSION_CLOSE\n", u32ClientID));
1478 rc = sessionClose(u32ClientID, callHandle, cParms, paParms);
1479 break;
1480
1481 /*
1482 * For all other regular commands we call our hostCallback
1483 * function. If the current command does not support notifications,
1484 * notifyHost will return VERR_NOT_SUPPORTED.
1485 */
1486 default:
1487 rc = hostCallback(eFunction, cParms, paParms);
1488 break;
1489 }
1490
1491 if (rc != VINF_HGCM_ASYNC_EXECUTE)
1492 {
1493 /* Tell the client that the call is complete (unblocks waiting). */
1494 AssertPtr(mpHelpers);
1495 mpHelpers->pfnCallComplete(callHandle, rc);
1496 }
1497 }
1498 }
1499 catch (std::bad_alloc)
1500 {
1501 rc = VERR_NO_MEMORY;
1502 }
1503}
1504
1505/**
1506 * Service call handler for the host.
1507 * @interface_method_impl{VBOXHGCMSVCFNTABLE,pfnHostCall}
1508 * @thread hgcm
1509 */
1510int Service::hostCall(uint32_t eFunction, uint32_t cParms, VBOXHGCMSVCPARM paParms[])
1511{
1512 int rc = VERR_NOT_SUPPORTED;
1513 LogFlowFunc(("fn=%RU32, cParms=%RU32, paParms=0x%p\n",
1514 eFunction, cParms, paParms));
1515 try
1516 {
1517 switch (eFunction)
1518 {
1519 /**
1520 * Host
1521 */
1522 case HOST_CANCEL_PENDING_WAITS:
1523 {
1524 LogFlowFunc(("HOST_CANCEL_PENDING_WAITS\n"));
1525 ClientStateMapIter itClientState = mClientStateMap.begin();
1526 while (itClientState != mClientStateMap.end())
1527 {
1528 int rc2 = itClientState->second.CancelWaiting(VINF_SUCCESS /* Pending rc. */);
1529 if (RT_FAILURE(rc2))
1530 LogFlowFunc(("Cancelling waiting for client ID=%RU32 failed with rc=%Rrc",
1531 itClientState->first, rc2));
1532 ++itClientState;
1533 }
1534 rc = VINF_SUCCESS;
1535 break;
1536 }
1537
1538 default:
1539 rc = hostProcessCommand(eFunction, cParms, paParms);
1540 break;
1541 }
1542 }
1543 catch (std::bad_alloc)
1544 {
1545 rc = VERR_NO_MEMORY;
1546 }
1547
1548 return rc;
1549}
1550
1551/**
1552 * Client asks another client (guest) session to close.
1553 *
1554 * @return IPRT status code.
1555 * @param u32ClientID The client's ID.
1556 * @param callHandle The client's call handle.
1557 * @param cParms Number of parameters.
1558 * @param paParms Array of parameters.
1559 */
1560int Service::sessionClose(uint32_t u32ClientID, VBOXHGCMCALLHANDLE callHandle, uint32_t cParms, VBOXHGCMSVCPARM paParms[])
1561{
1562 if (cParms < 2)
1563 return VERR_INVALID_PARAMETER;
1564
1565 uint32_t uContextID, uFlags;
1566 int rc = paParms[0].getUInt32(&uContextID);
1567 if (RT_SUCCESS(rc))
1568 rc = paParms[1].getUInt32(&uFlags);
1569
1570 uint32_t uSessionID = VBOX_GUESTCTRL_CONTEXTID_GET_SESSION(uContextID);
1571
1572 if (RT_SUCCESS(rc))
1573 rc = hostProcessCommand(HOST_SESSION_CLOSE, cParms, paParms);
1574
1575 LogFlowFunc(("Closing guest session ID=%RU32 (from client ID=%RU32) returned with rc=%Rrc\n",
1576 uSessionID, u32ClientID, rc));
1577 return rc;
1578}
1579
1580int Service::uninit(void)
1581{
1582 return VINF_SUCCESS;
1583}
1584
1585} /* namespace guestControl */
1586
1587using guestControl::Service;
1588
1589/**
1590 * @copydoc VBOXHGCMSVCLOAD
1591 */
1592extern "C" DECLCALLBACK(DECLEXPORT(int)) VBoxHGCMSvcLoad(VBOXHGCMSVCFNTABLE *pTable)
1593{
1594 int rc = VINF_SUCCESS;
1595
1596 LogFlowFunc(("pTable=%p\n", pTable));
1597
1598 if (!VALID_PTR(pTable))
1599 {
1600 rc = VERR_INVALID_PARAMETER;
1601 }
1602 else
1603 {
1604 LogFlowFunc(("pTable->cbSize=%d, pTable->u32Version=0x%08X\n", pTable->cbSize, pTable->u32Version));
1605
1606 if ( pTable->cbSize != sizeof (VBOXHGCMSVCFNTABLE)
1607 || pTable->u32Version != VBOX_HGCM_SVC_VERSION)
1608 {
1609 rc = VERR_VERSION_MISMATCH;
1610 }
1611 else
1612 {
1613 Service *pService = NULL;
1614 /* No exceptions may propagate outside. */
1615 try
1616 {
1617 pService = new Service(pTable->pHelpers);
1618 }
1619 catch (int rcThrown)
1620 {
1621 rc = rcThrown;
1622 }
1623 catch(std::bad_alloc &)
1624 {
1625 rc = VERR_NO_MEMORY;
1626 }
1627
1628 if (RT_SUCCESS(rc))
1629 {
1630 /*
1631 * We don't need an additional client data area on the host,
1632 * because we're a class which can have members for that :-).
1633 */
1634 pTable->cbClient = 0;
1635
1636 /* Register functions. */
1637 pTable->pfnUnload = Service::svcUnload;
1638 pTable->pfnConnect = Service::svcConnect;
1639 pTable->pfnDisconnect = Service::svcDisconnect;
1640 pTable->pfnCall = Service::svcCall;
1641 pTable->pfnHostCall = Service::svcHostCall;
1642 pTable->pfnSaveState = NULL; /* The service is stateless, so the normal */
1643 pTable->pfnLoadState = NULL; /* construction done before restoring suffices */
1644 pTable->pfnRegisterExtension = Service::svcRegisterExtension;
1645
1646 /* Service specific initialization. */
1647 pTable->pvService = pService;
1648 }
1649 else
1650 {
1651 if (pService)
1652 {
1653 delete pService;
1654 pService = NULL;
1655 }
1656 }
1657 }
1658 }
1659
1660 LogFlowFunc(("Returning %Rrc\n", rc));
1661 return rc;
1662}
1663
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette