VirtualBox

source: vbox/trunk/src/VBox/Main/hgcm/HGCM.cpp@ 10466

Last change on this file since 10466 was 9705, checked in by vboxsync, 17 years ago

Main/hgcm: fixed a segfault affecting services with no client data

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 63.6 KB
Line 
1/** @file
2 *
3 * HGCM (Host-Guest Communication Manager)
4 */
5
6/*
7 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22#define LOG_GROUP_MAIN_OVERRIDE LOG_GROUP_HGCM
23#include "Logging.h"
24
25#include "hgcm/HGCM.h"
26#include "hgcm/HGCMThread.h"
27
28#include <VBox/err.h>
29#include <VBox/hgcmsvc.h>
30
31#include <iprt/alloc.h>
32#include <iprt/alloca.h>
33#include <iprt/avl.h>
34#include <iprt/critsect.h>
35#include <iprt/asm.h>
36#include <iprt/ldr.h>
37#include <iprt/param.h>
38#include <iprt/path.h>
39#include <iprt/string.h>
40#include <iprt/semaphore.h>
41#include <iprt/thread.h>
42
43#include <VBox/VBoxGuest.h>
44
45/**
46 * A service gets one thread, which synchronously delivers messages to
47 * the service. This is good for serialization.
48 *
49 * Some services may want to process messages asynchronously, and will want
50 * a next message to be delivered, while a previous message is still being
51 * processed.
52 *
53 * The dedicated service thread delivers a next message when service
54 * returns after fetching a previous one. The service will call a message
55 * completion callback when message is actually processed. So returning
56 * from the service call means only that the service is processing message.
57 *
58 * 'Message processed' condition is indicated by service, which call the
59 * callback, even if the callback is called synchronously in the dedicated
60 * thread.
61 *
62 * This message completion callback is only valid for Call requests.
63 * Connect and Disconnect are processed synchronously by the service.
64 */
65
66
67/* The maximum allowed size of a service name in bytes. */
68#define VBOX_HGCM_SVC_NAME_MAX_BYTES 1024
69
70struct _HGCMSVCEXTHANDLEDATA
71{
72 char *pszServiceName;
73 /* The service name follows. */
74};
75
76/** Internal helper service object. HGCM code would use it to
77 * hold information about services and communicate with services.
78 * The HGCMService is an (in future) abstract class that implements
79 * common functionality. There will be derived classes for specific
80 * service types.
81 */
82
83class HGCMService
84{
85 private:
86 VBOXHGCMSVCHELPERS m_svcHelpers;
87
88 static HGCMService *sm_pSvcListHead;
89 static HGCMService *sm_pSvcListTail;
90
91 static int sm_cServices;
92
93 HGCMTHREADHANDLE m_thread;
94 friend DECLCALLBACK(void) hgcmServiceThread (HGCMTHREADHANDLE ThreadHandle, void *pvUser);
95
96 uint32_t volatile m_u32RefCnt;
97
98 HGCMService *m_pSvcNext;
99 HGCMService *m_pSvcPrev;
100
101 char *m_pszSvcName;
102 char *m_pszSvcLibrary;
103
104 RTLDRMOD m_hLdrMod;
105 PFNVBOXHGCMSVCLOAD m_pfnLoad;
106
107 VBOXHGCMSVCFNTABLE m_fntable;
108
109 int m_cClients;
110 int m_cClientsAllocated;
111
112 uint32_t *m_paClientIds;
113
114 HGCMSVCEXTHANDLE m_hExtension;
115
116 int loadServiceDLL (void);
117 void unloadServiceDLL (void);
118
119 /*
120 * Main HGCM thread methods.
121 */
122 int instanceCreate (const char *pszServiceLibrary, const char *pszServiceName);
123 void instanceDestroy (void);
124
125 int saveClientState(uint32_t u32ClientId, PSSMHANDLE pSSM);
126 int loadClientState(uint32_t u32ClientId, PSSMHANDLE pSSM);
127
128 HGCMService ();
129 ~HGCMService () {};
130
131 static DECLCALLBACK(void) svcHlpCallComplete (VBOXHGCMCALLHANDLE callHandle, int32_t rc);
132 static DECLCALLBACK(void) svcHlpDisconnectClient (void *pvInstance, uint32_t u32ClientId);
133
134 public:
135
136 /*
137 * Main HGCM thread methods.
138 */
139 static int LoadService (const char *pszServiceLibrary, const char *pszServiceName);
140 void UnloadService (void);
141
142 static void UnloadAll (void);
143
144 static int ResolveService (HGCMService **ppsvc, const char *pszServiceName);
145 void ReferenceService (void);
146 void ReleaseService (void);
147
148 static void Reset (void);
149
150 static int SaveState (PSSMHANDLE pSSM);
151 static int LoadState (PSSMHANDLE pSSM);
152
153 int CreateAndConnectClient (uint32_t *pu32ClientIdOut, uint32_t u32ClientIdIn);
154 int DisconnectClient (uint32_t u32ClientId, bool fFromService);
155
156 int HostCall (uint32_t u32Function, uint32_t cParms, VBOXHGCMSVCPARM *paParms);
157
158 uint32_t SizeOfClient (void) { return m_fntable.cbClient; };
159
160 int RegisterExtension (HGCMSVCEXTHANDLE handle, PFNHGCMSVCEXT pfnExtension, void *pvExtension);
161 void UnregisterExtension (HGCMSVCEXTHANDLE handle);
162
163 /*
164 * The service thread methods.
165 */
166
167 int GuestCall (PPDMIHGCMPORT pHGCMPort, PVBOXHGCMCMD pCmd, uint32_t u32ClientId, uint32_t u32Function, uint32_t cParms, VBOXHGCMSVCPARM aParms[]);
168};
169
170
171class HGCMClient: public HGCMObject
172{
173 public:
174 HGCMClient () : HGCMObject(HGCMOBJ_CLIENT), pService(NULL),
175 pvData(NULL) {};
176 ~HGCMClient ();
177
178 int Init (HGCMService *pSvc);
179
180 /** Service that the client is connected to. */
181 HGCMService *pService;
182
183 /** Client specific data. */
184 void *pvData;
185};
186
187HGCMClient::~HGCMClient ()
188{
189 if (pService->SizeOfClient () > 0)
190 RTMemFree (pvData);
191}
192
193int HGCMClient::Init (HGCMService *pSvc)
194{
195 pService = pSvc;
196
197 if (pService->SizeOfClient () > 0)
198 {
199 pvData = RTMemAllocZ (pService->SizeOfClient ());
200
201 if (!pvData)
202 {
203 return VERR_NO_MEMORY;
204 }
205 }
206
207 return VINF_SUCCESS;
208}
209
210
211#define HGCM_CLIENT_DATA(pService, pClient) (pClient->pvData)
212
213
214
215HGCMService *HGCMService::sm_pSvcListHead = NULL;
216HGCMService *HGCMService::sm_pSvcListTail = NULL;
217int HGCMService::sm_cServices = 0;
218
219HGCMService::HGCMService ()
220 :
221 m_thread (0),
222 m_u32RefCnt (0),
223 m_pSvcNext (NULL),
224 m_pSvcPrev (NULL),
225 m_pszSvcName (NULL),
226 m_pszSvcLibrary (NULL),
227 m_hLdrMod (NIL_RTLDRMOD),
228 m_pfnLoad (NULL),
229 m_cClients (0),
230 m_cClientsAllocated (0),
231 m_paClientIds (NULL),
232 m_hExtension (NULL)
233{
234 memset (&m_fntable, 0, sizeof (m_fntable));
235}
236
237
238static bool g_fResetting = false;
239static bool g_fSaveState = false;
240
241
242/** Helper function to load a local service DLL.
243 *
244 * @return VBox code
245 */
246int HGCMService::loadServiceDLL (void)
247{
248 LogFlowFunc(("m_pszSvcLibrary = %s\n", m_pszSvcLibrary));
249
250 if (m_pszSvcLibrary == NULL)
251 {
252 return VERR_INVALID_PARAMETER;
253 }
254
255 int rc = RTLdrLoadAppSharedLib (m_pszSvcLibrary, &m_hLdrMod);
256
257 if (VBOX_SUCCESS(rc))
258 {
259 LogFlowFunc(("successfully loaded the library.\n"));
260
261 m_pfnLoad = NULL;
262
263 rc = RTLdrGetSymbol (m_hLdrMod, VBOX_HGCM_SVCLOAD_NAME, (void**)&m_pfnLoad);
264
265 if (VBOX_FAILURE (rc) || !m_pfnLoad)
266 {
267 Log(("HGCMService::loadServiceDLL: Error resolving the service entry point %s, rc = %d, m_pfnLoad = %p\n", VBOX_HGCM_SVCLOAD_NAME, rc, m_pfnLoad));
268
269 if (VBOX_SUCCESS(rc))
270 {
271 /* m_pfnLoad was NULL */
272 rc = VERR_SYMBOL_NOT_FOUND;
273 }
274 }
275
276 if (VBOX_SUCCESS(rc))
277 {
278 memset (&m_fntable, 0, sizeof (m_fntable));
279
280 m_fntable.cbSize = sizeof (m_fntable);
281 m_fntable.u32Version = VBOX_HGCM_SVC_VERSION;
282 m_fntable.pHelpers = &m_svcHelpers;
283
284 rc = m_pfnLoad (&m_fntable);
285
286 LogFlowFunc(("m_pfnLoad rc = %Vrc\n", rc));
287
288 if (VBOX_SUCCESS (rc))
289 {
290 if ( m_fntable.pfnUnload == NULL
291 || m_fntable.pfnConnect == NULL
292 || m_fntable.pfnDisconnect == NULL
293 || m_fntable.pfnCall == NULL
294 )
295 {
296 Log(("HGCMService::loadServiceDLL: at least one of function pointers is NULL\n"));
297
298 rc = VERR_INVALID_PARAMETER;
299
300 if (m_fntable.pfnUnload)
301 {
302 m_fntable.pfnUnload (m_fntable.pvService);
303 }
304 }
305 }
306 }
307 }
308 else
309 {
310 LogRel(("HGCM: Failed to load the service library: [%s], rc = %Vrc. The service will be not available.\n", m_pszSvcLibrary, rc));
311 m_hLdrMod = NIL_RTLDRMOD;
312 }
313
314 if (VBOX_FAILURE(rc))
315 {
316 unloadServiceDLL ();
317 }
318
319 return rc;
320}
321
322/** Helper function to free a local service DLL.
323 *
324 * @return VBox code
325 */
326void HGCMService::unloadServiceDLL (void)
327{
328 if (m_hLdrMod)
329 {
330 RTLdrClose (m_hLdrMod);
331 }
332
333 memset (&m_fntable, 0, sizeof (m_fntable));
334 m_pfnLoad = NULL;
335 m_hLdrMod = NIL_RTLDRMOD;
336}
337
338/*
339 * Messages processed by service threads. These threads only call the service entry points.
340 */
341
342#define SVC_MSG_LOAD (0) /* Load the service library and call VBOX_HGCM_SVCLOAD_NAME entry point. */
343#define SVC_MSG_UNLOAD (1) /* call pfnUnload and unload the service library. */
344#define SVC_MSG_CONNECT (2) /* pfnConnect */
345#define SVC_MSG_DISCONNECT (3) /* pfnDisconnect */
346#define SVC_MSG_GUESTCALL (4) /* pfnGuestCall */
347#define SVC_MSG_HOSTCALL (5) /* pfnHostCall */
348#define SVC_MSG_LOADSTATE (6) /* pfnLoadState. */
349#define SVC_MSG_SAVESTATE (7) /* pfnSaveState. */
350#define SVC_MSG_QUIT (8) /* Terminate the thread. */
351#define SVC_MSG_REGEXT (9) /* pfnRegisterExtension */
352#define SVC_MSG_UNREGEXT (10) /* pfnRegisterExtension */
353
354class HGCMMsgSvcLoad: public HGCMMsgCore
355{
356};
357
358class HGCMMsgSvcUnload: public HGCMMsgCore
359{
360};
361
362class HGCMMsgSvcConnect: public HGCMMsgCore
363{
364 public:
365 /* client identifier */
366 uint32_t u32ClientId;
367};
368
369class HGCMMsgSvcDisconnect: public HGCMMsgCore
370{
371 public:
372 /* client identifier */
373 uint32_t u32ClientId;
374};
375
376class HGCMMsgHeader: public HGCMMsgCore
377{
378 public:
379 HGCMMsgHeader () : pCmd (NULL), pHGCMPort (NULL) {};
380
381 /* Command pointer/identifier. */
382 PVBOXHGCMCMD pCmd;
383
384 /* Port to be informed on message completion. */
385 PPDMIHGCMPORT pHGCMPort;
386};
387
388
389class HGCMMsgCall: public HGCMMsgHeader
390{
391 public:
392 /* client identifier */
393 uint32_t u32ClientId;
394
395 /* function number */
396 uint32_t u32Function;
397
398 /* number of parameters */
399 uint32_t cParms;
400
401 VBOXHGCMSVCPARM *paParms;
402};
403
404class HGCMMsgLoadSaveStateClient: public HGCMMsgCore
405{
406 public:
407 uint32_t u32ClientId;
408 PSSMHANDLE pSSM;
409};
410
411class HGCMMsgHostCallSvc: public HGCMMsgCore
412{
413 public:
414 /* function number */
415 uint32_t u32Function;
416
417 /* number of parameters */
418 uint32_t cParms;
419
420 VBOXHGCMSVCPARM *paParms;
421};
422
423class HGCMMsgSvcRegisterExtension: public HGCMMsgCore
424{
425 public:
426 /* Handle of the extension to be registered. */
427 HGCMSVCEXTHANDLE handle;
428 /* The extension entry point. */
429 PFNHGCMSVCEXT pfnExtension;
430 /* The extension pointer. */
431 void *pvExtension;
432};
433
434class HGCMMsgSvcUnregisterExtension: public HGCMMsgCore
435{
436 public:
437 /* Handle of the registered extension. */
438 HGCMSVCEXTHANDLE handle;
439};
440
441static HGCMMsgCore *hgcmMessageAllocSvc (uint32_t u32MsgId)
442{
443 switch (u32MsgId)
444 {
445 case SVC_MSG_LOAD: return new HGCMMsgSvcLoad ();
446 case SVC_MSG_UNLOAD: return new HGCMMsgSvcUnload ();
447 case SVC_MSG_CONNECT: return new HGCMMsgSvcConnect ();
448 case SVC_MSG_DISCONNECT: return new HGCMMsgSvcDisconnect ();
449 case SVC_MSG_HOSTCALL: return new HGCMMsgHostCallSvc ();
450 case SVC_MSG_GUESTCALL: return new HGCMMsgCall ();
451 case SVC_MSG_LOADSTATE:
452 case SVC_MSG_SAVESTATE: return new HGCMMsgLoadSaveStateClient ();
453 case SVC_MSG_REGEXT: return new HGCMMsgSvcRegisterExtension ();
454 case SVC_MSG_UNREGEXT: return new HGCMMsgSvcUnregisterExtension ();
455 default:
456 AssertReleaseMsgFailed(("Msg id = %08X\n", u32MsgId));
457 }
458
459 return NULL;
460}
461
462/*
463 * The service thread. Loads the service library and calls the service entry points.
464 */
465DECLCALLBACK(void) hgcmServiceThread (HGCMTHREADHANDLE ThreadHandle, void *pvUser)
466{
467 HGCMService *pSvc = (HGCMService *)pvUser;
468 AssertRelease(pSvc != NULL);
469
470 bool fQuit = false;
471
472 while (!fQuit)
473 {
474 HGCMMsgCore *pMsgCore;
475 int rc = hgcmMsgGet (ThreadHandle, &pMsgCore);
476
477 if (VBOX_FAILURE (rc))
478 {
479 /* The error means some serious unrecoverable problem in the hgcmMsg/hgcmThread layer. */
480 AssertMsgFailed (("%Vrc\n", rc));
481 break;
482 }
483
484 /* Cache required information to avoid unnecessary pMsgCore access. */
485 uint32_t u32MsgId = pMsgCore->MsgId ();
486
487 switch (u32MsgId)
488 {
489 case SVC_MSG_LOAD:
490 {
491 LogFlowFunc(("SVC_MSG_LOAD\n"));
492 rc = pSvc->loadServiceDLL ();
493 } break;
494
495 case SVC_MSG_UNLOAD:
496 {
497 LogFlowFunc(("SVC_MSG_UNLOAD\n"));
498 if (pSvc->m_fntable.pfnUnload)
499 {
500 pSvc->m_fntable.pfnUnload (pSvc->m_fntable.pvService);
501 }
502
503 pSvc->unloadServiceDLL ();
504 fQuit = true;
505 } break;
506
507 case SVC_MSG_CONNECT:
508 {
509 HGCMMsgSvcConnect *pMsg = (HGCMMsgSvcConnect *)pMsgCore;
510
511 LogFlowFunc(("SVC_MSG_CONNECT u32ClientId = %d\n", pMsg->u32ClientId));
512
513 HGCMClient *pClient = (HGCMClient *)hgcmObjReference (pMsg->u32ClientId, HGCMOBJ_CLIENT);
514
515 if (pClient)
516 {
517 rc = pSvc->m_fntable.pfnConnect (pSvc->m_fntable.pvService, pMsg->u32ClientId, HGCM_CLIENT_DATA(pSvc, pClient));
518
519 hgcmObjDereference (pClient);
520 }
521 else
522 {
523 rc = VERR_HGCM_INVALID_CLIENT_ID;
524 }
525 } break;
526
527 case SVC_MSG_DISCONNECT:
528 {
529 HGCMMsgSvcDisconnect *pMsg = (HGCMMsgSvcDisconnect *)pMsgCore;
530
531 LogFlowFunc(("SVC_MSG_DISCONNECT u32ClientId = %d\n", pMsg->u32ClientId));
532
533 HGCMClient *pClient = (HGCMClient *)hgcmObjReference (pMsg->u32ClientId, HGCMOBJ_CLIENT);
534
535 if (pClient)
536 {
537 rc = pSvc->m_fntable.pfnDisconnect (pSvc->m_fntable.pvService, pMsg->u32ClientId, HGCM_CLIENT_DATA(pSvc, pClient));
538
539 hgcmObjDereference (pClient);
540 }
541 else
542 {
543 rc = VERR_HGCM_INVALID_CLIENT_ID;
544 }
545 } break;
546
547 case SVC_MSG_GUESTCALL:
548 {
549 HGCMMsgCall *pMsg = (HGCMMsgCall *)pMsgCore;
550
551 LogFlowFunc(("SVC_MSG_GUESTCALL u32ClientId = %d, u32Function = %d, cParms = %d, paParms = %p\n",
552 pMsg->u32ClientId, pMsg->u32Function, pMsg->cParms, pMsg->paParms));
553
554 HGCMClient *pClient = (HGCMClient *)hgcmObjReference (pMsg->u32ClientId, HGCMOBJ_CLIENT);
555
556 if (pClient)
557 {
558 pSvc->m_fntable.pfnCall (pSvc->m_fntable.pvService, (VBOXHGCMCALLHANDLE)pMsg, pMsg->u32ClientId, HGCM_CLIENT_DATA(pSvc, pClient), pMsg->u32Function, pMsg->cParms, pMsg->paParms);
559
560 hgcmObjDereference (pClient);
561 }
562 else
563 {
564 rc = VERR_HGCM_INVALID_CLIENT_ID;
565 }
566 } break;
567
568 case SVC_MSG_HOSTCALL:
569 {
570 HGCMMsgHostCallSvc *pMsg = (HGCMMsgHostCallSvc *)pMsgCore;
571
572 LogFlowFunc(("SVC_MSG_HOSTCALL u32Function = %d, cParms = %d, paParms = %p\n", pMsg->u32Function, pMsg->cParms, pMsg->paParms));
573
574 rc = pSvc->m_fntable.pfnHostCall (pSvc->m_fntable.pvService, pMsg->u32Function, pMsg->cParms, pMsg->paParms);
575 } break;
576
577 case SVC_MSG_LOADSTATE:
578 {
579 HGCMMsgLoadSaveStateClient *pMsg = (HGCMMsgLoadSaveStateClient *)pMsgCore;
580
581 LogFlowFunc(("SVC_MSG_LOADSTATE\n"));
582
583 HGCMClient *pClient = (HGCMClient *)hgcmObjReference (pMsg->u32ClientId, HGCMOBJ_CLIENT);
584
585 if (pClient)
586 {
587 if (pSvc->m_fntable.pfnLoadState)
588 {
589 rc = pSvc->m_fntable.pfnLoadState (pSvc->m_fntable.pvService, pMsg->u32ClientId, HGCM_CLIENT_DATA(pSvc, pClient), pMsg->pSSM);
590 }
591
592 hgcmObjDereference (pClient);
593 }
594 else
595 {
596 rc = VERR_HGCM_INVALID_CLIENT_ID;
597 }
598 } break;
599
600 case SVC_MSG_SAVESTATE:
601 {
602 HGCMMsgLoadSaveStateClient *pMsg = (HGCMMsgLoadSaveStateClient *)pMsgCore;
603
604 LogFlowFunc(("SVC_MSG_SAVESTATE\n"));
605
606 HGCMClient *pClient = (HGCMClient *)hgcmObjReference (pMsg->u32ClientId, HGCMOBJ_CLIENT);
607
608 rc = VINF_SUCCESS;
609
610 if (pClient)
611 {
612 if (pSvc->m_fntable.pfnSaveState)
613 {
614 g_fSaveState = true;
615 rc = pSvc->m_fntable.pfnSaveState (pSvc->m_fntable.pvService, pMsg->u32ClientId, HGCM_CLIENT_DATA(pSvc, pClient), pMsg->pSSM);
616 g_fSaveState = false;
617 }
618
619 hgcmObjDereference (pClient);
620 }
621 else
622 {
623 rc = VERR_HGCM_INVALID_CLIENT_ID;
624 }
625 } break;
626
627 case SVC_MSG_REGEXT:
628 {
629 HGCMMsgSvcRegisterExtension *pMsg = (HGCMMsgSvcRegisterExtension *)pMsgCore;
630
631 LogFlowFunc(("SVC_MSG_REGEXT handle = %p, pfn = %p\n", pMsg->handle, pMsg->pfnExtension));
632
633 if (pSvc->m_hExtension)
634 {
635 rc = VERR_NOT_SUPPORTED;
636 }
637 else
638 {
639 if (pSvc->m_fntable.pfnRegisterExtension)
640 {
641 rc = pSvc->m_fntable.pfnRegisterExtension (pSvc->m_fntable.pvService, pMsg->pfnExtension, pMsg->pvExtension);
642 }
643 else
644 {
645 rc = VERR_NOT_SUPPORTED;
646 }
647
648 if (VBOX_SUCCESS (rc))
649 {
650 pSvc->m_hExtension = pMsg->handle;
651 }
652 }
653 } break;
654
655 case SVC_MSG_UNREGEXT:
656 {
657 HGCMMsgSvcUnregisterExtension *pMsg = (HGCMMsgSvcUnregisterExtension *)pMsgCore;
658
659 LogFlowFunc(("SVC_MSG_UNREGEXT handle = %p\n", pMsg->handle));
660
661 if (pSvc->m_hExtension != pMsg->handle)
662 {
663 rc = VERR_NOT_SUPPORTED;
664 }
665 else
666 {
667 if (pSvc->m_fntable.pfnRegisterExtension)
668 {
669 rc = pSvc->m_fntable.pfnRegisterExtension (pSvc->m_fntable.pvService, NULL, NULL);
670 }
671 else
672 {
673 rc = VERR_NOT_SUPPORTED;
674 }
675
676 pSvc->m_hExtension = NULL;
677 }
678 } break;
679
680 default:
681 {
682 AssertMsgFailed(("hgcmServiceThread::Unsupported message number %08X\n", u32MsgId));
683 rc = VERR_NOT_SUPPORTED;
684 } break;
685 }
686
687 if (u32MsgId != SVC_MSG_GUESTCALL)
688 {
689 /* For SVC_MSG_GUESTCALL the service calls the completion helper.
690 * Other messages have to be completed here.
691 */
692 hgcmMsgComplete (pMsgCore, rc);
693 }
694 }
695}
696
697/* static */ DECLCALLBACK(void) HGCMService::svcHlpCallComplete (VBOXHGCMCALLHANDLE callHandle, int32_t rc)
698{
699 HGCMMsgCore *pMsgCore = (HGCMMsgCore *)callHandle;
700
701 if (pMsgCore->MsgId () == SVC_MSG_GUESTCALL)
702 {
703 /* Only call the completion for these messages. The helper
704 * is called by the service, and the service does not get
705 * any other messages.
706 */
707 hgcmMsgComplete (pMsgCore, rc);
708 }
709 else
710 {
711 AssertFailed ();
712 }
713}
714
715/* static */ DECLCALLBACK(void) HGCMService::svcHlpDisconnectClient (void *pvInstance, uint32_t u32ClientId)
716{
717 HGCMService *pService = static_cast <HGCMService *> (pvInstance);
718
719 if (pService)
720 {
721 pService->DisconnectClient (u32ClientId, true);
722 }
723}
724
725static DECLCALLBACK(void) hgcmMsgCompletionCallback (int32_t result, HGCMMsgCore *pMsgCore)
726{
727 /* Call the VMMDev port interface to issue IRQ notification. */
728 HGCMMsgHeader *pMsgHdr = (HGCMMsgHeader *)pMsgCore;
729
730 LogFlow(("MAIN::hgcmMsgCompletionCallback: message %p\n", pMsgCore));
731
732 if (pMsgHdr->pHGCMPort && !g_fResetting)
733 {
734 pMsgHdr->pHGCMPort->pfnCompleted (pMsgHdr->pHGCMPort, g_fSaveState? VINF_HGCM_SAVE_STATE: result, pMsgHdr->pCmd);
735 }
736}
737
738/*
739 * The main HGCM methods of the service.
740 */
741
742int HGCMService::instanceCreate (const char *pszServiceLibrary, const char *pszServiceName)
743{
744 LogFlowFunc(("name %s, lib %s\n", pszServiceName, pszServiceLibrary));
745
746 /* The maximum length of the thread name, allowed by the RT is 15. */
747 char achThreadName[16];
748
749 strncpy (achThreadName, pszServiceName, 15);
750 achThreadName[15] = 0;
751
752 int rc = hgcmThreadCreate (&m_thread, achThreadName, hgcmServiceThread, this);
753
754 if (VBOX_SUCCESS(rc))
755 {
756 m_pszSvcName = RTStrDup (pszServiceName);
757 m_pszSvcLibrary = RTStrDup (pszServiceLibrary);
758
759 if (!m_pszSvcName || !m_pszSvcLibrary)
760 {
761 RTStrFree (m_pszSvcLibrary);
762 m_pszSvcLibrary = NULL;
763
764 RTStrFree (m_pszSvcName);
765 m_pszSvcName = NULL;
766
767 rc = VERR_NO_MEMORY;
768 }
769 else
770 {
771 /* Initialize service helpers table. */
772 m_svcHelpers.pfnCallComplete = svcHlpCallComplete;
773 m_svcHelpers.pvInstance = this;
774 m_svcHelpers.pfnDisconnectClient = svcHlpDisconnectClient;
775
776 /* Execute the load request on the service thread. */
777 HGCMMSGHANDLE hMsg;
778 rc = hgcmMsgAlloc (m_thread, &hMsg, SVC_MSG_LOAD, hgcmMessageAllocSvc);
779
780 if (VBOX_SUCCESS(rc))
781 {
782 rc = hgcmMsgSend (hMsg);
783 }
784 }
785 }
786
787 if (VBOX_FAILURE(rc))
788 {
789 instanceDestroy ();
790 }
791
792 LogFlowFunc(("rc = %Vrc\n", rc));
793 return rc;
794}
795
796void HGCMService::instanceDestroy (void)
797{
798 LogFlowFunc(("%s\n", m_pszSvcName));
799
800 HGCMMSGHANDLE hMsg;
801 int rc = hgcmMsgAlloc (m_thread, &hMsg, SVC_MSG_UNLOAD, hgcmMessageAllocSvc);
802
803 if (VBOX_SUCCESS(rc))
804 {
805 rc = hgcmMsgSend (hMsg);
806
807 if (VBOX_SUCCESS (rc))
808 {
809 hgcmThreadWait (m_thread);
810 }
811 }
812
813 RTStrFree (m_pszSvcLibrary);
814 m_pszSvcLibrary = NULL;
815
816 RTStrFree (m_pszSvcName);
817 m_pszSvcName = NULL;
818}
819
820int HGCMService::saveClientState(uint32_t u32ClientId, PSSMHANDLE pSSM)
821{
822 LogFlowFunc(("%s\n", m_pszSvcName));
823
824 HGCMMSGHANDLE hMsg;
825 int rc = hgcmMsgAlloc (m_thread, &hMsg, SVC_MSG_SAVESTATE, hgcmMessageAllocSvc);
826
827 if (VBOX_SUCCESS(rc))
828 {
829 HGCMMsgLoadSaveStateClient *pMsg = (HGCMMsgLoadSaveStateClient *)hgcmObjReference (hMsg, HGCMOBJ_MSG);
830 AssertRelease(pMsg);
831
832 pMsg->u32ClientId = u32ClientId;
833 pMsg->pSSM = pSSM;
834
835 hgcmObjDereference (pMsg);
836
837 rc = hgcmMsgSend (hMsg);
838 }
839
840 LogFlowFunc(("rc = %Vrc\n", rc));
841 return rc;
842}
843
844int HGCMService::loadClientState (uint32_t u32ClientId, PSSMHANDLE pSSM)
845{
846 LogFlowFunc(("%s\n", m_pszSvcName));
847
848 HGCMMSGHANDLE hMsg;
849 int rc = hgcmMsgAlloc (m_thread, &hMsg, SVC_MSG_LOADSTATE, hgcmMessageAllocSvc);
850
851 if (VBOX_SUCCESS(rc))
852 {
853 HGCMMsgLoadSaveStateClient *pMsg = (HGCMMsgLoadSaveStateClient *)hgcmObjReference (hMsg, HGCMOBJ_MSG);
854
855 AssertRelease(pMsg);
856
857 pMsg->u32ClientId = u32ClientId;
858 pMsg->pSSM = pSSM;
859
860 hgcmObjDereference (pMsg);
861
862 rc = hgcmMsgSend (hMsg);
863 }
864
865 LogFlowFunc(("rc = %Vrc\n", rc));
866 return rc;
867}
868
869
870/** The method creates a service and references it.
871 *
872 * @param pszServcieLibrary The library to be loaded.
873 * @param pszServiceName The name of the service.
874 * @return VBox rc.
875 * @thread main HGCM
876 */
877/* static */ int HGCMService::LoadService (const char *pszServiceLibrary, const char *pszServiceName)
878{
879 LogFlowFunc(("lib %s, name = %s\n", pszServiceLibrary, pszServiceName));
880
881 /* Look at already loaded services to avoid double loading. */
882
883 HGCMService *pSvc;
884 int rc = HGCMService::ResolveService (&pSvc, pszServiceName);
885
886 if (VBOX_SUCCESS (rc))
887 {
888 /* The service is already loaded. */
889 pSvc->ReleaseService ();
890 rc = VERR_HGCM_SERVICE_EXISTS;
891 }
892 else
893 {
894 /* Create the new service. */
895 pSvc = new HGCMService ();
896
897 if (!pSvc)
898 {
899 rc = VERR_NO_MEMORY;
900 }
901 else
902 {
903 /* Load the library and call the initialization entry point. */
904 rc = pSvc->instanceCreate (pszServiceLibrary, pszServiceName);
905
906 if (VBOX_SUCCESS(rc))
907 {
908 /* Insert the just created service to list for future references. */
909 pSvc->m_pSvcNext = sm_pSvcListHead;
910 pSvc->m_pSvcPrev = NULL;
911
912 if (sm_pSvcListHead)
913 {
914 sm_pSvcListHead->m_pSvcPrev = pSvc;
915 }
916 else
917 {
918 sm_pSvcListTail = pSvc;
919 }
920
921 sm_pSvcListHead = pSvc;
922
923 sm_cServices++;
924
925 /* Reference the service (for first time) until it is unloaded on HGCM termination. */
926 AssertRelease (pSvc->m_u32RefCnt == 0);
927 pSvc->ReferenceService ();
928
929 LogFlowFunc(("service %p\n", pSvc));
930 }
931 }
932 }
933
934 LogFlowFunc(("rc = %Vrc\n", rc));
935 return rc;
936}
937
938/** The method unloads a service.
939 *
940 * @thread main HGCM
941 */
942void HGCMService::UnloadService (void)
943{
944 LogFlowFunc(("name = %s\n", m_pszSvcName));
945
946 /* Remove the service from the list. */
947 if (m_pSvcNext)
948 {
949 m_pSvcNext->m_pSvcPrev = m_pSvcPrev;
950 }
951 else
952 {
953 sm_pSvcListTail = m_pSvcPrev;
954 }
955
956 if (m_pSvcPrev)
957 {
958 m_pSvcPrev->m_pSvcNext = m_pSvcNext;
959 }
960 else
961 {
962 sm_pSvcListHead = m_pSvcNext;
963 }
964
965 sm_cServices--;
966
967 /* The service must be unloaded only if all clients were disconnected. */
968 LogFlowFunc(("m_u32RefCnt = %d\n", m_u32RefCnt));
969 AssertRelease (m_u32RefCnt == 1);
970
971 /* Now the service can be released. */
972 ReleaseService ();
973}
974
975/** The method unloads all services.
976 *
977 * @thread main HGCM
978 */
979/* static */ void HGCMService::UnloadAll (void)
980{
981 while (sm_pSvcListHead)
982 {
983 sm_pSvcListHead->UnloadService ();
984 }
985}
986
987/** The method obtains a referenced pointer to the service with
988 * specified name. The caller must call ReleaseService when
989 * the pointer is no longer needed.
990 *
991 * @param ppSvc Where to store the pointer to the service.
992 * @param pszServiceName The name of the service.
993 * @return VBox rc.
994 * @thread main HGCM
995 */
996/* static */ int HGCMService::ResolveService (HGCMService **ppSvc, const char *pszServiceName)
997{
998 LogFlowFunc(("ppSvc = %p name = %s\n",
999 ppSvc, pszServiceName));
1000
1001 if (!ppSvc || !pszServiceName)
1002 {
1003 return VERR_INVALID_PARAMETER;
1004 }
1005
1006 HGCMService *pSvc = sm_pSvcListHead;
1007
1008 while (pSvc)
1009 {
1010 if (strcmp (pSvc->m_pszSvcName, pszServiceName) == 0)
1011 {
1012 break;
1013 }
1014
1015 pSvc = pSvc->m_pSvcNext;
1016 }
1017
1018 LogFlowFunc(("lookup in the list is %p\n", pSvc));
1019
1020 if (pSvc == NULL)
1021 {
1022 return VERR_HGCM_SERVICE_NOT_FOUND;
1023 }
1024
1025 pSvc->ReferenceService ();
1026
1027 *ppSvc = pSvc;
1028
1029 return VINF_SUCCESS;
1030}
1031
1032/** The method increases reference counter.
1033 *
1034 * @thread main HGCM
1035 */
1036void HGCMService::ReferenceService (void)
1037{
1038 ASMAtomicIncU32 (&m_u32RefCnt);
1039 LogFlowFunc(("m_u32RefCnt = %d\n", m_u32RefCnt));
1040}
1041
1042/** The method dereferences a service and deletes it when no more refs.
1043 *
1044 * @thread main HGCM
1045 */
1046void HGCMService::ReleaseService (void)
1047{
1048 LogFlowFunc(("m_u32RefCnt = %d\n", m_u32RefCnt));
1049 uint32_t u32RefCnt = ASMAtomicDecU32 (&m_u32RefCnt);
1050 AssertRelease(u32RefCnt != ~0U);
1051
1052 LogFlowFunc(("u32RefCnt = %d, name %s\n", u32RefCnt, m_pszSvcName));
1053
1054 if (u32RefCnt == 0)
1055 {
1056 instanceDestroy ();
1057 delete this;
1058 }
1059}
1060
1061/** The method is called when the VM is being reset or terminated
1062 * and disconnects all clients from all services.
1063 *
1064 * @thread main HGCM
1065 */
1066/* static */ void HGCMService::Reset (void)
1067{
1068 g_fResetting = true;
1069
1070 HGCMService *pSvc = sm_pSvcListHead;
1071
1072 while (pSvc)
1073 {
1074 while (pSvc->m_cClients && pSvc->m_paClientIds)
1075 {
1076 LogFlowFunc(("handle %d\n", pSvc->m_paClientIds[0]));
1077 pSvc->DisconnectClient (pSvc->m_paClientIds[0], false);
1078 }
1079
1080 pSvc = pSvc->m_pSvcNext;
1081 }
1082
1083 g_fResetting = false;
1084}
1085
1086/** The method saves the HGCM state.
1087 *
1088 * @param pSSM The saved state context.
1089 * @return VBox rc.
1090 * @thread main HGCM
1091 */
1092/* static */ int HGCMService::SaveState (PSSMHANDLE pSSM)
1093{
1094 /* Save the current handle count and restore afterwards to avoid client id conflicts. */
1095 int rc = SSMR3PutU32(pSSM, hgcmObjQueryHandleCount());
1096 AssertRCReturn(rc, rc);
1097
1098 LogFlowFunc(("%d services to be saved:\n", sm_cServices));
1099
1100 /* Save number of services. */
1101 rc = SSMR3PutU32(pSSM, sm_cServices);
1102 AssertRCReturn(rc, rc);
1103
1104 /* Save every service. */
1105 HGCMService *pSvc = sm_pSvcListHead;
1106
1107 while (pSvc)
1108 {
1109 LogFlowFunc(("Saving service [%s]\n", pSvc->m_pszSvcName));
1110
1111 /* Save the length of the service name. */
1112 rc = SSMR3PutU32(pSSM, strlen(pSvc->m_pszSvcName) + 1);
1113 AssertRCReturn(rc, rc);
1114
1115 /* Save the name of the service. */
1116 rc = SSMR3PutStrZ(pSSM, pSvc->m_pszSvcName);
1117 AssertRCReturn(rc, rc);
1118
1119 /* Save the number of clients. */
1120 rc = SSMR3PutU32(pSSM, pSvc->m_cClients);
1121 AssertRCReturn(rc, rc);
1122
1123 /* Call the service for every client. Normally a service must not have
1124 * a global state to be saved: only per client info is relevant.
1125 * The global state of a service is configured during VM startup.
1126 */
1127 int i;
1128
1129 for (i = 0; i < pSvc->m_cClients; i++)
1130 {
1131 uint32_t u32ClientId = pSvc->m_paClientIds[i];
1132
1133 Log(("client id 0x%08X\n", u32ClientId));
1134
1135 /* Save the client id. */
1136 rc = SSMR3PutU32(pSSM, u32ClientId);
1137 AssertRCReturn(rc, rc);
1138
1139 /* Call the service, so the operation is executed by the service thread. */
1140 rc = pSvc->saveClientState (u32ClientId, pSSM);
1141 AssertRCReturn(rc, rc);
1142 }
1143
1144 pSvc = pSvc->m_pSvcNext;
1145 }
1146
1147 return VINF_SUCCESS;
1148}
1149
1150/** The method loads saved HGCM state.
1151 *
1152 * @param pSSM The saved state context.
1153 * @return VBox rc.
1154 * @thread main HGCM
1155 */
1156/* static */ int HGCMService::LoadState (PSSMHANDLE pSSM)
1157{
1158 /* Restore handle count to avoid client id conflicts. */
1159 uint32_t u32;
1160
1161 int rc = SSMR3GetU32(pSSM, &u32);
1162 AssertRCReturn(rc, rc);
1163
1164 hgcmObjSetHandleCount(u32);
1165
1166 /* Get the number of services. */
1167 uint32_t cServices;
1168
1169 rc = SSMR3GetU32(pSSM, &cServices);
1170 AssertRCReturn(rc, rc);
1171
1172 LogFlowFunc(("%d services to be restored:\n", cServices));
1173
1174 while (cServices--)
1175 {
1176 /* Get the length of the service name. */
1177 rc = SSMR3GetU32(pSSM, &u32);
1178 AssertRCReturn(rc, rc);
1179 AssertReturn(u32 <= VBOX_HGCM_SVC_NAME_MAX_BYTES, VERR_SSM_UNEXPECTED_DATA);
1180
1181 char *pszServiceName = (char *)alloca (u32);
1182
1183 /* Get the service name. */
1184 rc = SSMR3GetStrZ(pSSM, pszServiceName, u32);
1185 AssertRCReturn(rc, rc);
1186
1187 LogFlowFunc(("Restoring service [%s]\n", pszServiceName));
1188
1189 /* Resolve the service instance. */
1190 HGCMService *pSvc;
1191 rc = ResolveService (&pSvc, pszServiceName);
1192 AssertReturn(pSvc, VERR_SSM_UNEXPECTED_DATA);
1193
1194 /* Get the number of clients. */
1195 uint32_t cClients;
1196 rc = SSMR3GetU32(pSSM, &cClients);
1197 if (VBOX_FAILURE(rc))
1198 {
1199 pSvc->ReleaseService ();
1200 AssertFailed();
1201 return rc;
1202 }
1203
1204 while (cClients--)
1205 {
1206 /* Get the client id. */
1207 uint32_t u32ClientId;
1208 rc = SSMR3GetU32(pSSM, &u32ClientId);
1209 if (VBOX_FAILURE(rc))
1210 {
1211 pSvc->ReleaseService ();
1212 AssertFailed();
1213 return rc;
1214 }
1215
1216 /* Connect the client. */
1217 rc = pSvc->CreateAndConnectClient (NULL, u32ClientId);
1218 if (VBOX_FAILURE(rc))
1219 {
1220 pSvc->ReleaseService ();
1221 AssertFailed();
1222 return rc;
1223 }
1224
1225 /* Call the service, so the operation is executed by the service thread. */
1226 rc = pSvc->loadClientState (u32ClientId, pSSM);
1227 if (VBOX_FAILURE(rc))
1228 {
1229 pSvc->ReleaseService ();
1230 AssertFailed();
1231 return rc;
1232 }
1233 }
1234
1235 pSvc->ReleaseService ();
1236 }
1237
1238 return VINF_SUCCESS;
1239}
1240
1241/* Create a new client instance and connect it to the service.
1242 *
1243 * @param pu32ClientIdOut If not NULL, then the method must generate a new handle for the client.
1244 * If NULL, use the given 'u32ClientIdIn' handle.
1245 * @param u32ClientIdIn The handle for the client, when 'pu32ClientIdOut' is NULL.
1246 * @return VBox rc.
1247 */
1248int HGCMService::CreateAndConnectClient (uint32_t *pu32ClientIdOut, uint32_t u32ClientIdIn)
1249{
1250 LogFlowFunc(("pu32ClientIdOut = %p, u32ClientIdIn = %d\n", pu32ClientIdOut, u32ClientIdIn));
1251
1252 /* Allocate a client information structure. */
1253 HGCMClient *pClient = new HGCMClient ();
1254
1255 if (!pClient)
1256 {
1257 LogWarningFunc(("Could not allocate HGCMClient!!!\n"));
1258 return VERR_NO_MEMORY;
1259 }
1260
1261 uint32_t handle;
1262
1263 if (pu32ClientIdOut != NULL)
1264 {
1265 handle = hgcmObjGenerateHandle (pClient);
1266 }
1267 else
1268 {
1269 handle = hgcmObjAssignHandle (pClient, u32ClientIdIn);
1270 }
1271
1272 LogFlowFunc(("client id = %d\n", handle));
1273
1274 AssertRelease(handle);
1275
1276 /* Initialize the HGCM part of the client. */
1277 int rc = pClient->Init (this);
1278
1279 if (VBOX_SUCCESS(rc))
1280 {
1281 /* Call the service. */
1282 HGCMMSGHANDLE hMsg;
1283
1284 rc = hgcmMsgAlloc (m_thread, &hMsg, SVC_MSG_CONNECT, hgcmMessageAllocSvc);
1285
1286 if (VBOX_SUCCESS(rc))
1287 {
1288 HGCMMsgSvcConnect *pMsg = (HGCMMsgSvcConnect *)hgcmObjReference (hMsg, HGCMOBJ_MSG);
1289 AssertRelease(pMsg);
1290
1291 pMsg->u32ClientId = handle;
1292
1293 hgcmObjDereference (pMsg);
1294
1295 rc = hgcmMsgSend (hMsg);
1296
1297 if (VBOX_SUCCESS (rc))
1298 {
1299 /* Add the client Id to the array. */
1300 if (m_cClients == m_cClientsAllocated)
1301 {
1302 m_paClientIds = (uint32_t *)RTMemRealloc (m_paClientIds, (m_cClientsAllocated + 64) * sizeof (m_paClientIds[0]));
1303 Assert(m_paClientIds);
1304 m_cClientsAllocated += 64;
1305 }
1306
1307 m_paClientIds[m_cClients] = handle;
1308 m_cClients++;
1309 }
1310 }
1311 }
1312
1313 if (VBOX_FAILURE(rc))
1314 {
1315 hgcmObjDeleteHandle (handle);
1316 }
1317 else
1318 {
1319 if (pu32ClientIdOut != NULL)
1320 {
1321 *pu32ClientIdOut = handle;
1322 }
1323
1324 ReferenceService ();
1325 }
1326
1327 LogFlowFunc(("rc = %Vrc\n", rc));
1328 return rc;
1329}
1330
1331/* Disconnect the client from the service and delete the client handle.
1332 *
1333 * @param u32ClientId The handle of the client.
1334 * @return VBox rc.
1335 */
1336int HGCMService::DisconnectClient (uint32_t u32ClientId, bool fFromService)
1337{
1338 int rc = VINF_SUCCESS;
1339
1340 LogFlowFunc(("client id = %d, fFromService = %d\n", u32ClientId, fFromService));
1341
1342 if (!fFromService)
1343 {
1344 /* Call the service. */
1345 HGCMMSGHANDLE hMsg;
1346
1347 rc = hgcmMsgAlloc (m_thread, &hMsg, SVC_MSG_DISCONNECT, hgcmMessageAllocSvc);
1348
1349 if (VBOX_SUCCESS(rc))
1350 {
1351 HGCMMsgSvcDisconnect *pMsg = (HGCMMsgSvcDisconnect *)hgcmObjReference (hMsg, HGCMOBJ_MSG);
1352 AssertRelease(pMsg);
1353
1354 pMsg->u32ClientId = u32ClientId;
1355
1356 hgcmObjDereference (pMsg);
1357
1358 rc = hgcmMsgSend (hMsg);
1359 }
1360 }
1361
1362 if (VBOX_SUCCESS (rc))
1363 {
1364 /* Remove the client id from the array in any case. */
1365 int i;
1366
1367 for (i = 0; i < m_cClients; i++)
1368 {
1369 if (m_paClientIds[i] == u32ClientId)
1370 {
1371 m_cClients--;
1372
1373 if (m_cClients > i)
1374 {
1375 memmove (&m_paClientIds[i], &m_paClientIds[i + 1], m_cClients - i);
1376 }
1377
1378 break;
1379 }
1380 }
1381
1382 /* Delete the client handle. */
1383 hgcmObjDeleteHandle (u32ClientId);
1384
1385 /* The service must be released. */
1386 ReleaseService ();
1387 }
1388
1389 LogFlowFunc(("rc = %Vrc\n", rc));
1390 return rc;
1391}
1392
1393int HGCMService::RegisterExtension (HGCMSVCEXTHANDLE handle,
1394 PFNHGCMSVCEXT pfnExtension,
1395 void *pvExtension)
1396{
1397 LogFlowFunc(("%s\n", handle->pszServiceName));
1398
1399 /* Forward the message to the service thread. */
1400 HGCMMSGHANDLE hMsg = 0;
1401 int rc = hgcmMsgAlloc (m_thread, &hMsg, SVC_MSG_REGEXT, hgcmMessageAllocSvc);
1402
1403 if (VBOX_SUCCESS(rc))
1404 {
1405 HGCMMsgSvcRegisterExtension *pMsg = (HGCMMsgSvcRegisterExtension *)hgcmObjReference (hMsg, HGCMOBJ_MSG);
1406 AssertRelease(pMsg);
1407
1408 pMsg->handle = handle;
1409 pMsg->pfnExtension = pfnExtension;
1410 pMsg->pvExtension = pvExtension;
1411
1412 hgcmObjDereference (pMsg);
1413
1414 rc = hgcmMsgSend (hMsg);
1415 }
1416
1417 LogFlowFunc(("rc = %Vrc\n", rc));
1418 return rc;
1419}
1420
1421void HGCMService::UnregisterExtension (HGCMSVCEXTHANDLE handle)
1422{
1423 /* Forward the message to the service thread. */
1424 HGCMMSGHANDLE hMsg = 0;
1425 int rc = hgcmMsgAlloc (m_thread, &hMsg, SVC_MSG_UNREGEXT, hgcmMessageAllocSvc);
1426
1427 if (VBOX_SUCCESS(rc))
1428 {
1429 HGCMMsgSvcUnregisterExtension *pMsg = (HGCMMsgSvcUnregisterExtension *)hgcmObjReference (hMsg, HGCMOBJ_MSG);
1430 AssertRelease(pMsg);
1431
1432 pMsg->handle = handle;
1433
1434 hgcmObjDereference (pMsg);
1435
1436 rc = hgcmMsgSend (hMsg);
1437 }
1438
1439 LogFlowFunc(("rc = %Vrc\n", rc));
1440}
1441
1442/* Perform a guest call to the service.
1443 *
1444 * @param pHGCMPort The port to be used for completion confirmation.
1445 * @param pCmd The VBox HGCM context.
1446 * @param u32ClientId The client handle to be disconnected and deleted.
1447 * @param u32Function The function number.
1448 * @param cParms Number of parameters.
1449 * @param paParms Pointer to array of parameters.
1450 * @return VBox rc.
1451 */
1452int HGCMService::GuestCall (PPDMIHGCMPORT pHGCMPort, PVBOXHGCMCMD pCmd, uint32_t u32ClientId, uint32_t u32Function, uint32_t cParms, VBOXHGCMSVCPARM paParms[])
1453{
1454 HGCMMSGHANDLE hMsg = 0;
1455
1456 LogFlow(("MAIN::HGCMService::Call\n"));
1457
1458 int rc = hgcmMsgAlloc (m_thread, &hMsg, SVC_MSG_GUESTCALL, hgcmMessageAllocSvc);
1459
1460 if (VBOX_SUCCESS(rc))
1461 {
1462 HGCMMsgCall *pMsg = (HGCMMsgCall *)hgcmObjReference (hMsg, HGCMOBJ_MSG);
1463
1464 AssertRelease(pMsg);
1465
1466 pMsg->pCmd = pCmd;
1467 pMsg->pHGCMPort = pHGCMPort;
1468
1469 pMsg->u32ClientId = u32ClientId;
1470 pMsg->u32Function = u32Function;
1471 pMsg->cParms = cParms;
1472 pMsg->paParms = paParms;
1473
1474 hgcmObjDereference (pMsg);
1475
1476 rc = hgcmMsgPost (hMsg, hgcmMsgCompletionCallback);
1477 }
1478 else
1479 {
1480 Log(("MAIN::HGCMService::Call: Message allocation failed: %Vrc\n", rc));
1481 }
1482
1483 LogFlowFunc(("rc = %Vrc\n", rc));
1484 return rc;
1485}
1486
1487/* Perform a host call the service.
1488 *
1489 * @param u32Function The function number.
1490 * @param cParms Number of parameters.
1491 * @param paParms Pointer to array of parameters.
1492 * @return VBox rc.
1493 */
1494int HGCMService::HostCall (uint32_t u32Function, uint32_t cParms, VBOXHGCMSVCPARM *paParms)
1495{
1496 LogFlowFunc(("%s u32Function = %d, cParms = %d, paParms = %p\n",
1497 m_pszSvcName, u32Function, cParms, paParms));
1498
1499 HGCMMSGHANDLE hMsg = 0;
1500 int rc = hgcmMsgAlloc (m_thread, &hMsg, SVC_MSG_HOSTCALL, hgcmMessageAllocSvc);
1501
1502 if (VBOX_SUCCESS(rc))
1503 {
1504 HGCMMsgHostCallSvc *pMsg = (HGCMMsgHostCallSvc *)hgcmObjReference (hMsg, HGCMOBJ_MSG);
1505 AssertRelease(pMsg);
1506
1507 pMsg->u32Function = u32Function;
1508 pMsg->cParms = cParms;
1509 pMsg->paParms = paParms;
1510
1511 hgcmObjDereference (pMsg);
1512
1513 rc = hgcmMsgSend (hMsg);
1514 }
1515
1516 LogFlowFunc(("rc = %Vrc\n", rc));
1517 return rc;
1518}
1519
1520
1521/*
1522 * Main HGCM thread that manages services.
1523 */
1524
1525/* Messages processed by the main HGCM thread. */
1526#define HGCM_MSG_CONNECT (10) /* Connect a client to a service. */
1527#define HGCM_MSG_DISCONNECT (11) /* Disconnect the specified client id. */
1528#define HGCM_MSG_LOAD (12) /* Load the service. */
1529#define HGCM_MSG_HOSTCALL (13) /* Call the service. */
1530#define HGCM_MSG_LOADSTATE (14) /* Load saved state for the specified service. */
1531#define HGCM_MSG_SAVESTATE (15) /* Save state for the specified service. */
1532#define HGCM_MSG_RESET (16) /* Disconnect all clients from the specified service. */
1533#define HGCM_MSG_QUIT (17) /* Unload all services and terminate the thread. */
1534#define HGCM_MSG_REGEXT (18) /* Register a service extension. */
1535#define HGCM_MSG_UNREGEXT (19) /* Unregister a service extension. */
1536
1537class HGCMMsgMainConnect: public HGCMMsgHeader
1538{
1539 public:
1540 /* Service name. */
1541 const char *pszServiceName;
1542 /* Where to store the client handle. */
1543 uint32_t *pu32ClientId;
1544};
1545
1546class HGCMMsgMainDisconnect: public HGCMMsgHeader
1547{
1548 public:
1549 /* Handle of the client to be disconnected. */
1550 uint32_t u32ClientId;
1551};
1552
1553class HGCMMsgMainLoad: public HGCMMsgCore
1554{
1555 public:
1556 /* Name of the library to be loaded. */
1557 const char *pszServiceLibrary;
1558 /* Name to be assigned to the service. */
1559 const char *pszServiceName;
1560};
1561
1562class HGCMMsgMainHostCall: public HGCMMsgCore
1563{
1564 public:
1565 /* Which service to call. */
1566 const char *pszServiceName;
1567 /* Function number. */
1568 uint32_t u32Function;
1569 /* Number of the function parameters. */
1570 uint32_t cParms;
1571 /* Pointer to array of the function parameters. */
1572 VBOXHGCMSVCPARM *paParms;
1573};
1574
1575class HGCMMsgMainLoadSaveState: public HGCMMsgCore
1576{
1577 public:
1578 /* SSM context. */
1579 PSSMHANDLE pSSM;
1580};
1581
1582class HGCMMsgMainReset: public HGCMMsgCore
1583{
1584};
1585
1586class HGCMMsgMainQuit: public HGCMMsgCore
1587{
1588};
1589
1590class HGCMMsgMainRegisterExtension: public HGCMMsgCore
1591{
1592 public:
1593 /* Returned handle to be used in HGCMMsgMainUnregisterExtension. */
1594 HGCMSVCEXTHANDLE *pHandle;
1595 /* Name of the service. */
1596 const char *pszServiceName;
1597 /* The extension entry point. */
1598 PFNHGCMSVCEXT pfnExtension;
1599 /* The extension pointer. */
1600 void *pvExtension;
1601};
1602
1603class HGCMMsgMainUnregisterExtension: public HGCMMsgCore
1604{
1605 public:
1606 /* Handle of the registered extension. */
1607 HGCMSVCEXTHANDLE handle;
1608};
1609
1610static HGCMMsgCore *hgcmMainMessageAlloc (uint32_t u32MsgId)
1611{
1612 switch (u32MsgId)
1613 {
1614 case HGCM_MSG_CONNECT: return new HGCMMsgMainConnect ();
1615 case HGCM_MSG_DISCONNECT: return new HGCMMsgMainDisconnect ();
1616 case HGCM_MSG_LOAD: return new HGCMMsgMainLoad ();
1617 case HGCM_MSG_HOSTCALL: return new HGCMMsgMainHostCall ();
1618 case HGCM_MSG_LOADSTATE:
1619 case HGCM_MSG_SAVESTATE: return new HGCMMsgMainLoadSaveState ();
1620 case HGCM_MSG_RESET: return new HGCMMsgMainReset ();
1621 case HGCM_MSG_QUIT: return new HGCMMsgMainQuit ();
1622 case HGCM_MSG_REGEXT: return new HGCMMsgMainRegisterExtension ();
1623 case HGCM_MSG_UNREGEXT: return new HGCMMsgMainUnregisterExtension ();
1624 default:
1625 AssertReleaseMsgFailed(("Msg id = %08X\n", u32MsgId));
1626 }
1627
1628 return NULL;
1629}
1630
1631
1632/* The main HGCM thread handler. */
1633static DECLCALLBACK(void) hgcmThread (HGCMTHREADHANDLE ThreadHandle, void *pvUser)
1634{
1635 LogFlowFunc(("ThreadHandle = %p, pvUser = %p\n",
1636 ThreadHandle, pvUser));
1637
1638 NOREF(pvUser);
1639
1640 bool fQuit = false;
1641
1642 while (!fQuit)
1643 {
1644 HGCMMsgCore *pMsgCore;
1645 int rc = hgcmMsgGet (ThreadHandle, &pMsgCore);
1646
1647 if (VBOX_FAILURE (rc))
1648 {
1649 /* The error means some serious unrecoverable problem in the hgcmMsg/hgcmThread layer. */
1650 AssertMsgFailed (("%Vrc\n", rc));
1651 break;
1652 }
1653
1654 uint32_t u32MsgId = pMsgCore->MsgId ();
1655
1656 switch (u32MsgId)
1657 {
1658 case HGCM_MSG_CONNECT:
1659 {
1660 HGCMMsgMainConnect *pMsg = (HGCMMsgMainConnect *)pMsgCore;
1661
1662 LogFlowFunc(("HGCM_MSG_CONNECT pszServiceName %s, pu32ClientId %p\n",
1663 pMsg->pszServiceName, pMsg->pu32ClientId));
1664
1665 /* Resolve the service name to the pointer to service instance.
1666 */
1667 HGCMService *pService;
1668 rc = HGCMService::ResolveService (&pService, pMsg->pszServiceName);
1669
1670 if (VBOX_SUCCESS (rc))
1671 {
1672 /* Call the service instance method. */
1673 rc = pService->CreateAndConnectClient (pMsg->pu32ClientId, 0);
1674
1675 /* Release the service after resolve. */
1676 pService->ReleaseService ();
1677 }
1678 } break;
1679
1680 case HGCM_MSG_DISCONNECT:
1681 {
1682 HGCMMsgMainDisconnect *pMsg = (HGCMMsgMainDisconnect *)pMsgCore;
1683
1684 LogFlowFunc(("HGCM_MSG_DISCONNECT u32ClientId = %d\n",
1685 pMsg->u32ClientId));
1686
1687 HGCMClient *pClient = (HGCMClient *)hgcmObjReference (pMsg->u32ClientId, HGCMOBJ_CLIENT);
1688
1689 if (!pClient)
1690 {
1691 rc = VERR_HGCM_INVALID_CLIENT_ID;
1692 break;
1693 }
1694
1695 /* The service the client belongs to. */
1696 HGCMService *pService = pClient->pService;
1697
1698 /* Call the service instance to disconnect the client. */
1699 rc = pService->DisconnectClient (pMsg->u32ClientId, false);
1700
1701 hgcmObjDereference (pClient);
1702 } break;
1703
1704 case HGCM_MSG_LOAD:
1705 {
1706 HGCMMsgMainLoad *pMsg = (HGCMMsgMainLoad *)pMsgCore;
1707
1708 LogFlowFunc(("HGCM_MSG_LOAD pszServiceName = %s, pMsg->pszServiceLibrary = %s\n",
1709 pMsg->pszServiceName, pMsg->pszServiceLibrary));
1710
1711 rc = HGCMService::LoadService (pMsg->pszServiceLibrary, pMsg->pszServiceName);
1712 } break;
1713
1714 case HGCM_MSG_HOSTCALL:
1715 {
1716 HGCMMsgMainHostCall *pMsg = (HGCMMsgMainHostCall *)pMsgCore;
1717
1718 LogFlowFunc(("HGCM_MSG_HOSTCALL pszServiceName %s, u32Function %d, cParms %d, paParms %p\n",
1719 pMsg->pszServiceName, pMsg->u32Function, pMsg->cParms, pMsg->paParms));
1720
1721 /* Resolve the service name to the pointer to service instance. */
1722 HGCMService *pService;
1723 rc = HGCMService::ResolveService (&pService, pMsg->pszServiceName);
1724
1725 if (VBOX_SUCCESS (rc))
1726 {
1727 rc = pService->HostCall (pMsg->u32Function, pMsg->cParms, pMsg->paParms);
1728
1729 pService->ReleaseService ();
1730 }
1731 } break;
1732
1733 case HGCM_MSG_RESET:
1734 {
1735 LogFlowFunc(("HGCM_MSG_RESET\n"));
1736
1737 HGCMService::Reset ();
1738 } break;
1739
1740 case HGCM_MSG_LOADSTATE:
1741 {
1742 HGCMMsgMainLoadSaveState *pMsg = (HGCMMsgMainLoadSaveState *)pMsgCore;
1743
1744 LogFlowFunc(("HGCM_MSG_LOADSTATE\n"));
1745
1746 rc = HGCMService::LoadState (pMsg->pSSM);
1747 } break;
1748
1749 case HGCM_MSG_SAVESTATE:
1750 {
1751 HGCMMsgMainLoadSaveState *pMsg = (HGCMMsgMainLoadSaveState *)pMsgCore;
1752
1753 LogFlowFunc(("HGCM_MSG_SAVESTATE\n"));
1754
1755 rc = HGCMService::SaveState (pMsg->pSSM);
1756 } break;
1757
1758 case HGCM_MSG_QUIT:
1759 {
1760 LogFlowFunc(("HGCM_MSG_QUIT\n"));
1761
1762 HGCMService::UnloadAll ();
1763
1764 fQuit = true;
1765 } break;
1766
1767 case HGCM_MSG_REGEXT:
1768 {
1769 HGCMMsgMainRegisterExtension *pMsg = (HGCMMsgMainRegisterExtension *)pMsgCore;
1770
1771 LogFlowFunc(("HGCM_MSG_REGEXT\n"));
1772
1773 /* Allocate the handle data. */
1774 HGCMSVCEXTHANDLE handle = (HGCMSVCEXTHANDLE)RTMemAllocZ (sizeof (struct _HGCMSVCEXTHANDLEDATA)
1775 + strlen (pMsg->pszServiceName)
1776 + sizeof (char));
1777
1778 if (handle == NULL)
1779 {
1780 rc = VERR_NO_MEMORY;
1781 }
1782 else
1783 {
1784 handle->pszServiceName = (char *)((uint8_t *)handle + sizeof (struct _HGCMSVCEXTHANDLEDATA));
1785 strcpy (handle->pszServiceName, pMsg->pszServiceName);
1786
1787 HGCMService *pService;
1788 rc = HGCMService::ResolveService (&pService, handle->pszServiceName);
1789
1790 if (VBOX_SUCCESS (rc))
1791 {
1792 pService->RegisterExtension (handle, pMsg->pfnExtension, pMsg->pvExtension);
1793
1794 pService->ReleaseService ();
1795 }
1796
1797 if (VBOX_FAILURE (rc))
1798 {
1799 RTMemFree (handle);
1800 }
1801 else
1802 {
1803 *pMsg->pHandle = handle;
1804 }
1805 }
1806 } break;
1807
1808 case HGCM_MSG_UNREGEXT:
1809 {
1810 HGCMMsgMainUnregisterExtension *pMsg = (HGCMMsgMainUnregisterExtension *)pMsgCore;
1811
1812 LogFlowFunc(("HGCM_MSG_UNREGEXT\n"));
1813
1814 HGCMService *pService;
1815 rc = HGCMService::ResolveService (&pService, pMsg->handle->pszServiceName);
1816
1817 if (VBOX_SUCCESS (rc))
1818 {
1819 pService->UnregisterExtension (pMsg->handle);
1820
1821 pService->ReleaseService ();
1822 }
1823
1824 RTMemFree (pMsg->handle);
1825 } break;
1826
1827 default:
1828 {
1829 AssertMsgFailed(("hgcmThread: Unsupported message number %08X!!!\n", u32MsgId));
1830 rc = VERR_NOT_SUPPORTED;
1831 } break;
1832 }
1833
1834 /* Complete the message processing. */
1835 hgcmMsgComplete (pMsgCore, rc);
1836
1837 LogFlowFunc(("message processed %Vrc\n", rc));
1838 }
1839}
1840
1841
1842/*
1843 * The HGCM API.
1844 */
1845
1846/* The main hgcm thread. */
1847static HGCMTHREADHANDLE g_hgcmThread = 0;
1848
1849/*
1850 * Public HGCM functions.
1851 *
1852 * hgcmGuest* - called as a result of the guest HGCM requests.
1853 * hgcmHost* - called by the host.
1854 */
1855
1856/* Load a HGCM service from the specified library.
1857 * Assign the specified name to the service.
1858 *
1859 * @param pszServiceLibrary The library to be loaded.
1860 * @param pszServiceName The name to be assigned to the service.
1861 * @return VBox rc.
1862 */
1863int HGCMHostLoad (const char *pszServiceLibrary,
1864 const char *pszServiceName)
1865{
1866 LogFlowFunc(("lib = %s, name = %s\n", pszServiceLibrary, pszServiceName));
1867
1868 if (!pszServiceLibrary || !pszServiceName)
1869 {
1870 return VERR_INVALID_PARAMETER;
1871 }
1872
1873 /* Forward the request to the main hgcm thread. */
1874 HGCMMSGHANDLE hMsg = 0;
1875
1876 int rc = hgcmMsgAlloc (g_hgcmThread, &hMsg, HGCM_MSG_LOAD, hgcmMainMessageAlloc);
1877
1878 if (VBOX_SUCCESS(rc))
1879 {
1880 /* Initialize the message. Since the message is synchronous, use the supplied pointers. */
1881 HGCMMsgMainLoad *pMsg = (HGCMMsgMainLoad *)hgcmObjReference (hMsg, HGCMOBJ_MSG);
1882 AssertRelease(pMsg);
1883
1884 pMsg->pszServiceLibrary = pszServiceLibrary;
1885 pMsg->pszServiceName = pszServiceName;
1886
1887 hgcmObjDereference (pMsg);
1888
1889 rc = hgcmMsgSend (hMsg);
1890 }
1891
1892 LogFlowFunc(("rc = %Vrc\n", rc));
1893 return rc;
1894}
1895
1896/* Register a HGCM service extension.
1897 *
1898 * @param pHandle Returned handle for the registered extension.
1899 * @param pszServiceName The name of the service.
1900 * @param pfnExtension The extension callback.
1901 * @return VBox rc.
1902 */
1903int HGCMHostRegisterServiceExtension (HGCMSVCEXTHANDLE *pHandle,
1904 const char *pszServiceName,
1905 PFNHGCMSVCEXT pfnExtension,
1906 void *pvExtension)
1907{
1908 LogFlowFunc(("pHandle = %p, name = %s, pfn = %p, rv = %p\n", pHandle, pszServiceName, pfnExtension, pvExtension));
1909
1910 if (!pHandle || !pszServiceName || !pfnExtension)
1911 {
1912 return VERR_INVALID_PARAMETER;
1913 }
1914
1915 /* Forward the request to the main hgcm thread. */
1916 HGCMMSGHANDLE hMsg = 0;
1917
1918 int rc = hgcmMsgAlloc (g_hgcmThread, &hMsg, HGCM_MSG_REGEXT, hgcmMainMessageAlloc);
1919
1920 if (VBOX_SUCCESS(rc))
1921 {
1922 /* Initialize the message. Since the message is synchronous, use the supplied pointers. */
1923 HGCMMsgMainRegisterExtension *pMsg = (HGCMMsgMainRegisterExtension *)hgcmObjReference (hMsg, HGCMOBJ_MSG);
1924 AssertRelease(pMsg);
1925
1926 pMsg->pHandle = pHandle;
1927 pMsg->pszServiceName = pszServiceName;
1928 pMsg->pfnExtension = pfnExtension;
1929 pMsg->pvExtension = pvExtension;
1930
1931 hgcmObjDereference (pMsg);
1932
1933 rc = hgcmMsgSend (hMsg);
1934 }
1935
1936 LogFlowFunc(("*pHandle = %p, rc = %Vrc\n", *pHandle, rc));
1937 return rc;
1938}
1939
1940void HGCMHostUnregisterServiceExtension (HGCMSVCEXTHANDLE handle)
1941{
1942 LogFlowFunc(("handle = %p\n", handle));
1943
1944 /* Forward the request to the main hgcm thread. */
1945 HGCMMSGHANDLE hMsg = 0;
1946
1947 int rc = hgcmMsgAlloc (g_hgcmThread, &hMsg, HGCM_MSG_UNREGEXT, hgcmMainMessageAlloc);
1948
1949 if (VBOX_SUCCESS(rc))
1950 {
1951 /* Initialize the message. */
1952 HGCMMsgMainUnregisterExtension *pMsg = (HGCMMsgMainUnregisterExtension *)hgcmObjReference (hMsg, HGCMOBJ_MSG);
1953 AssertRelease(pMsg);
1954
1955 pMsg->handle = handle;
1956
1957 hgcmObjDereference (pMsg);
1958
1959 rc = hgcmMsgSend (hMsg);
1960 }
1961
1962 LogFlowFunc(("rc = %Vrc\n", rc));
1963 return;
1964}
1965
1966/* Find a service and inform it about a client connection, create a client handle.
1967 *
1968 * @param pHGCMPort The port to be used for completion confirmation.
1969 * @param pCmd The VBox HGCM context.
1970 * @param pszServiceName The name of the service to be connected to.
1971 * @param pu32ClientId Where the store the created client handle.
1972 * @return VBox rc.
1973 */
1974int HGCMGuestConnect (PPDMIHGCMPORT pHGCMPort,
1975 PVBOXHGCMCMD pCmd,
1976 const char *pszServiceName,
1977 uint32_t *pu32ClientId)
1978{
1979 LogFlowFunc(("pHGCMPort = %p, pCmd = %p, name = %s, pu32ClientId = %p\n",
1980 pHGCMPort, pCmd, pszServiceName, pu32ClientId));
1981
1982 if (pHGCMPort == NULL || pCmd == NULL || pszServiceName == NULL || pu32ClientId == NULL)
1983 {
1984 return VERR_INVALID_PARAMETER;
1985 }
1986
1987 /* Forward the request to the main hgcm thread. */
1988 HGCMMSGHANDLE hMsg = 0;
1989
1990 int rc = hgcmMsgAlloc (g_hgcmThread, &hMsg, HGCM_MSG_CONNECT, hgcmMainMessageAlloc);
1991
1992 if (VBOX_SUCCESS(rc))
1993 {
1994 /* Initialize the message. Since 'pszServiceName' and 'pu32ClientId'
1995 * will not be deallocated by the caller until the message is completed,
1996 * use the supplied pointers.
1997 */
1998 HGCMMsgMainConnect *pMsg = (HGCMMsgMainConnect *)hgcmObjReference (hMsg, HGCMOBJ_MSG);
1999 AssertRelease(pMsg);
2000
2001 pMsg->pHGCMPort = pHGCMPort;
2002 pMsg->pCmd = pCmd;
2003 pMsg->pszServiceName = pszServiceName;
2004 pMsg->pu32ClientId = pu32ClientId;
2005
2006 hgcmObjDereference (pMsg);
2007
2008 rc = hgcmMsgPost (hMsg, hgcmMsgCompletionCallback);
2009 }
2010
2011 LogFlowFunc(("rc = %Vrc\n", rc));
2012 return rc;
2013}
2014
2015/* Tell a service that the client is disconnecting, destroy the client handle.
2016 *
2017 * @param pHGCMPort The port to be used for completion confirmation.
2018 * @param pCmd The VBox HGCM context.
2019 * @param u32ClientId The client handle to be disconnected and deleted.
2020 * @return VBox rc.
2021 */
2022int HGCMGuestDisconnect (PPDMIHGCMPORT pHGCMPort,
2023 PVBOXHGCMCMD pCmd,
2024 uint32_t u32ClientId)
2025{
2026 LogFlowFunc(("pHGCMPort = %p, pCmd = %p, u32ClientId = %d\n",
2027 pHGCMPort, pCmd, u32ClientId));
2028
2029 if (!pHGCMPort || !pCmd || !u32ClientId)
2030 {
2031 return VERR_INVALID_PARAMETER;
2032 }
2033
2034 /* Forward the request to the main hgcm thread. */
2035 HGCMMSGHANDLE hMsg = 0;
2036
2037 int rc = hgcmMsgAlloc (g_hgcmThread, &hMsg, HGCM_MSG_DISCONNECT, hgcmMainMessageAlloc);
2038
2039 if (VBOX_SUCCESS(rc))
2040 {
2041 /* Initialize the message. */
2042 HGCMMsgMainDisconnect *pMsg = (HGCMMsgMainDisconnect *)hgcmObjReference (hMsg, HGCMOBJ_MSG);
2043 AssertRelease(pMsg);
2044
2045 pMsg->pCmd = pCmd;
2046 pMsg->pHGCMPort = pHGCMPort;
2047 pMsg->u32ClientId = u32ClientId;
2048
2049 hgcmObjDereference (pMsg);
2050
2051 rc = hgcmMsgPost (hMsg, hgcmMsgCompletionCallback);
2052 }
2053
2054 LogFlowFunc(("rc = %Vrc\n", rc));
2055 return rc;
2056}
2057
2058/* Helper to send either HGCM_MSG_SAVESTATE or HGCM_MSG_LOADSTATE messages to the main HGCM thread.
2059 *
2060 * @param pSSM The SSM handle.
2061 * @param u32MsgId The message to be sent: HGCM_MSG_SAVESTATE or HGCM_MSG_LOADSTATE.
2062 * @return VBox rc.
2063 */
2064static int hgcmHostLoadSaveState (PSSMHANDLE pSSM,
2065 uint32_t u32MsgId)
2066{
2067 LogFlowFunc(("pSSM = %p, u32MsgId = %d\n", pSSM, u32MsgId));
2068
2069 HGCMMSGHANDLE hMsg = 0;
2070
2071 int rc = hgcmMsgAlloc (g_hgcmThread, &hMsg, u32MsgId, hgcmMainMessageAlloc);
2072
2073 if (VBOX_SUCCESS(rc))
2074 {
2075 HGCMMsgMainLoadSaveState *pMsg = (HGCMMsgMainLoadSaveState *)hgcmObjReference (hMsg, HGCMOBJ_MSG);
2076 AssertRelease(pMsg);
2077
2078 pMsg->pSSM = pSSM;
2079
2080 hgcmObjDereference (pMsg);
2081
2082 rc = hgcmMsgSend (hMsg);
2083 }
2084
2085 LogFlowFunc(("rc = %Vrc\n", rc));
2086 return rc;
2087}
2088
2089/* Save the state of services.
2090 *
2091 * @param pSSM The SSM handle.
2092 * @return VBox rc.
2093 */
2094int HGCMHostSaveState (PSSMHANDLE pSSM)
2095{
2096 return hgcmHostLoadSaveState (pSSM, HGCM_MSG_SAVESTATE);
2097}
2098
2099/* Load the state of services.
2100 *
2101 * @param pSSM The SSM handle.
2102 * @return VBox rc.
2103 */
2104int HGCMHostLoadState (PSSMHANDLE pSSM)
2105{
2106 return hgcmHostLoadSaveState (pSSM, HGCM_MSG_LOADSTATE);
2107}
2108
2109/* The guest calls the service.
2110 *
2111 * @param pHGCMPort The port to be used for completion confirmation.
2112 * @param pCmd The VBox HGCM context.
2113 * @param u32ClientId The client handle to be disconnected and deleted.
2114 * @param u32Function The function number.
2115 * @param cParms Number of parameters.
2116 * @param paParms Pointer to array of parameters.
2117 * @return VBox rc.
2118 */
2119int HGCMGuestCall (PPDMIHGCMPORT pHGCMPort,
2120 PVBOXHGCMCMD pCmd,
2121 uint32_t u32ClientId,
2122 uint32_t u32Function,
2123 uint32_t cParms,
2124 VBOXHGCMSVCPARM *paParms)
2125{
2126 LogFlowFunc(("pHGCMPort = %p, pCmd = %p, u32ClientId = %d, u32Function = %d, cParms = %d, paParms = %p\n",
2127 pHGCMPort, pCmd, u32ClientId, u32Function, cParms, paParms));
2128
2129 if (!pHGCMPort || !pCmd || u32ClientId == 0)
2130 {
2131 return VERR_INVALID_PARAMETER;
2132 }
2133
2134 int rc = VERR_HGCM_INVALID_CLIENT_ID;
2135
2136 /* Resolve the client handle to the client instance pointer. */
2137 HGCMClient *pClient = (HGCMClient *)hgcmObjReference (u32ClientId, HGCMOBJ_CLIENT);
2138
2139 if (pClient)
2140 {
2141 AssertRelease(pClient->pService);
2142
2143 /* Forward the message to the service thread. */
2144 rc = pClient->pService->GuestCall (pHGCMPort, pCmd, u32ClientId, u32Function, cParms, paParms);
2145
2146 hgcmObjDereference (pClient);
2147 }
2148
2149 LogFlowFunc(("rc = %Vrc\n", rc));
2150 return rc;
2151}
2152
2153/* The host calls the service.
2154 *
2155 * @param pszServiceName The service name to be called.
2156 * @param u32Function The function number.
2157 * @param cParms Number of parameters.
2158 * @param paParms Pointer to array of parameters.
2159 * @return VBox rc.
2160 */
2161int HGCMHostCall (const char *pszServiceName,
2162 uint32_t u32Function,
2163 uint32_t cParms,
2164 VBOXHGCMSVCPARM *paParms)
2165{
2166 LogFlowFunc(("name = %s, u32Function = %d, cParms = %d, paParms = %p\n",
2167 pszServiceName, u32Function, cParms, paParms));
2168
2169 if (!pszServiceName)
2170 {
2171 return VERR_INVALID_PARAMETER;
2172 }
2173
2174 HGCMMSGHANDLE hMsg = 0;
2175
2176 /* Host calls go to main HGCM thread that resolves the service name to the
2177 * service instance pointer and then, using the service pointer, forwards
2178 * the message to the service thread.
2179 * So it is slow but host calls are intended mostly for configuration and
2180 * other non-time-critical functions.
2181 */
2182 int rc = hgcmMsgAlloc (g_hgcmThread, &hMsg, HGCM_MSG_HOSTCALL, hgcmMainMessageAlloc);
2183
2184 if (VBOX_SUCCESS(rc))
2185 {
2186 HGCMMsgMainHostCall *pMsg = (HGCMMsgMainHostCall *)hgcmObjReference (hMsg, HGCMOBJ_MSG);
2187 AssertRelease(pMsg);
2188
2189 pMsg->pszServiceName = (char *)pszServiceName;
2190 pMsg->u32Function = u32Function;
2191 pMsg->cParms = cParms;
2192 pMsg->paParms = paParms;
2193
2194 hgcmObjDereference (pMsg);
2195
2196 rc = hgcmMsgSend (hMsg);
2197 }
2198
2199 LogFlowFunc(("rc = %Vrc\n", rc));
2200 return rc;
2201}
2202
2203int HGCMHostReset (void)
2204{
2205 LogFlowFunc(("\n"));
2206
2207 /* Disconnect all clients.
2208 */
2209
2210 HGCMMSGHANDLE hMsg = 0;
2211
2212 int rc = hgcmMsgAlloc (g_hgcmThread, &hMsg, HGCM_MSG_RESET, hgcmMainMessageAlloc);
2213
2214 if (VBOX_SUCCESS(rc))
2215 {
2216 rc = hgcmMsgSend (hMsg);
2217 }
2218
2219 LogFlowFunc(("rc = %Vrc\n", rc));
2220 return rc;
2221}
2222
2223int HGCMHostInit (void)
2224{
2225 LogFlowFunc(("\n"));
2226
2227 int rc = hgcmThreadInit ();
2228
2229 if (VBOX_SUCCESS(rc))
2230 {
2231 /*
2232 * Start main HGCM thread.
2233 */
2234
2235 rc = hgcmThreadCreate (&g_hgcmThread, "MainHGCMthread", hgcmThread, NULL);
2236
2237 if (VBOX_FAILURE (rc))
2238 {
2239 LogRel(("Failed to start HGCM thread. HGCM services will be unavailable!!! rc = %Vrc\n", rc));
2240 }
2241 }
2242
2243 LogFlowFunc(("rc = %Vrc\n", rc));
2244 return rc;
2245}
2246
2247int HGCMHostShutdown (void)
2248{
2249 LogFlowFunc(("\n"));
2250
2251 /*
2252 * Do HGCMReset and then unload all services.
2253 */
2254
2255 int rc = HGCMHostReset ();
2256
2257 if (VBOX_SUCCESS (rc))
2258 {
2259 /* Send the quit message to the main hgcmThread. */
2260 HGCMMSGHANDLE hMsg = 0;
2261
2262 rc = hgcmMsgAlloc (g_hgcmThread, &hMsg, HGCM_MSG_QUIT, hgcmMainMessageAlloc);
2263
2264 if (VBOX_SUCCESS(rc))
2265 {
2266 rc = hgcmMsgSend (hMsg);
2267
2268 if (VBOX_SUCCESS (rc))
2269 {
2270 /* Wait for the thread termination. */
2271 hgcmThreadWait (g_hgcmThread);
2272 g_hgcmThread = 0;
2273
2274 hgcmThreadUninit ();
2275 }
2276 }
2277 }
2278
2279 LogFlowFunc(("rc = %Vrc\n", rc));
2280 return rc;
2281}
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