VirtualBox

source: vbox/trunk/include/VBox/HostServices/Service.h@ 58110

Last change on this file since 58110 was 58106, checked in by vboxsync, 9 years ago

include,misc: Corrected a bunch of doxygen errors.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 17.2 KB
Line 
1/** @file
2 * Base class for an host-guest service.
3 */
4
5/*
6 * Copyright (C) 2011-2015 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 *
16 * The contents of this file may alternatively be used under the terms
17 * of the Common Development and Distribution License Version 1.0
18 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
19 * VirtualBox OSE distribution, in which case the provisions of the
20 * CDDL are applicable instead of those of the GPL.
21 *
22 * You may elect to license modified versions of this file under the
23 * terms and conditions of either the GPL or the CDDL or both.
24 */
25
26
27#ifndef ___VBox_HostService_Service_h
28#define ___VBox_HostService_Service_h
29
30#include <VBox/log.h>
31#include <VBox/hgcmsvc.h>
32#include <iprt/assert.h>
33#include <iprt/alloc.h>
34#include <iprt/cpp/utils.h>
35
36#include <memory> /* for auto_ptr */
37
38/** @todo document the poor classes. */
39namespace HGCM
40{
41
42class Message
43{
44 /* Contains a copy of HGCM parameters. */
45public:
46 Message(uint32_t uMsg, uint32_t cParms, VBOXHGCMSVCPARM aParms[])
47 : m_uMsg(0)
48 , m_cParms(0)
49 , m_paParms(0)
50 {
51 initData(uMsg, cParms, aParms);
52 }
53
54 virtual ~Message(void)
55 {
56 cleanup();
57 }
58
59 uint32_t message() const { return m_uMsg; }
60 uint32_t paramsCount() const { return m_cParms; }
61
62 int getData(uint32_t uMsg, uint32_t cParms, VBOXHGCMSVCPARM aParms[]) const
63 {
64 if (m_uMsg != uMsg)
65 {
66 LogFlowFunc(("Message type does not match (%u (buffer), %u (guest))\n",
67 m_uMsg, uMsg));
68 return VERR_INVALID_PARAMETER;
69 }
70 if (m_cParms != cParms)
71 {
72 LogFlowFunc(("Parameter count does not match (%u (buffer), %u (guest))\n",
73 m_cParms, cParms));
74 return VERR_INVALID_PARAMETER;
75 }
76
77 int rc = copyParmsInternal(cParms, m_paParms, &aParms[0], false /* fCreatePtrs */);
78
79// if (RT_FAILURE(rc))
80// cleanup(aParms);
81 return rc;
82 }
83
84 int getParmU32Info(uint32_t iParm, uint32_t *pu32Info) const
85 {
86 AssertPtrNullReturn(pu32Info, VERR_INVALID_PARAMETER);
87 AssertReturn(iParm < m_cParms, VERR_INVALID_PARAMETER);
88 AssertReturn(m_paParms[iParm].type == VBOX_HGCM_SVC_PARM_32BIT, VERR_INVALID_PARAMETER);
89
90 *pu32Info = m_paParms[iParm].u.uint32;
91
92 return VINF_SUCCESS;
93 }
94
95 int getParmU64Info(uint32_t iParm, uint64_t *pu64Info) const
96 {
97 AssertPtrNullReturn(pu64Info, VERR_INVALID_PARAMETER);
98 AssertReturn(iParm < m_cParms, VERR_INVALID_PARAMETER);
99 AssertReturn(m_paParms[iParm].type == VBOX_HGCM_SVC_PARM_64BIT, VERR_INVALID_PARAMETER);
100
101 *pu64Info = m_paParms[iParm].u.uint64;
102
103 return VINF_SUCCESS;
104 }
105
106 int getParmPtrInfo(uint32_t iParm, void **ppvAddr, uint32_t *pcSize) const
107 {
108 AssertPtrNullReturn(ppvAddr, VERR_INVALID_PARAMETER);
109 AssertPtrNullReturn(pcSize, VERR_INVALID_PARAMETER);
110 AssertReturn(iParm < m_cParms, VERR_INVALID_PARAMETER);
111 AssertReturn(m_paParms[iParm].type == VBOX_HGCM_SVC_PARM_PTR, VERR_INVALID_PARAMETER);
112
113 *ppvAddr = m_paParms[iParm].u.pointer.addr;
114 *pcSize = m_paParms[iParm].u.pointer.size;
115
116 return VINF_SUCCESS;
117 }
118
119 static int copyParms(uint32_t cParms, PVBOXHGCMSVCPARM paParmsSrc, PVBOXHGCMSVCPARM paParmsDst)
120 {
121 return copyParmsInternal(cParms, paParmsSrc, paParmsDst, false /* fCreatePtrs */);
122 }
123
124private:
125
126 uint32_t m_uMsg;
127 uint32_t m_cParms;
128 PVBOXHGCMSVCPARM m_paParms;
129
130 int initData(uint32_t uMsg, uint32_t cParms, VBOXHGCMSVCPARM aParms[])
131 {
132 AssertReturn(cParms < 256, VERR_INVALID_PARAMETER);
133 AssertPtrNullReturn(aParms, VERR_INVALID_PARAMETER);
134
135 /* Cleanup old messages. */
136 cleanup();
137
138 m_uMsg = uMsg;
139 m_cParms = cParms;
140
141 if (cParms > 0)
142 {
143 m_paParms = (VBOXHGCMSVCPARM*)RTMemAllocZ(sizeof(VBOXHGCMSVCPARM) * m_cParms);
144 if (!m_paParms)
145 return VERR_NO_MEMORY;
146 }
147
148 int rc = copyParmsInternal(cParms, &aParms[0], m_paParms, true /* fCreatePtrs */);
149
150 if (RT_FAILURE(rc))
151 cleanup();
152
153 return rc;
154 }
155
156 static int copyParmsInternal(uint32_t cParms, PVBOXHGCMSVCPARM paParmsSrc, PVBOXHGCMSVCPARM paParmsDst, bool fCreatePtrs)
157 {
158 int rc = VINF_SUCCESS;
159 for (uint32_t i = 0; i < cParms; ++i)
160 {
161 paParmsDst[i].type = paParmsSrc[i].type;
162 switch (paParmsSrc[i].type)
163 {
164 case VBOX_HGCM_SVC_PARM_32BIT:
165 {
166 paParmsDst[i].u.uint32 = paParmsSrc[i].u.uint32;
167 break;
168 }
169 case VBOX_HGCM_SVC_PARM_64BIT:
170 {
171 paParmsDst[i].u.uint64 = paParmsSrc[i].u.uint64;
172 break;
173 }
174 case VBOX_HGCM_SVC_PARM_PTR:
175 {
176 /* Do we have to recreate the memory? */
177 if (fCreatePtrs)
178 {
179 /* Yes, do so. */
180 paParmsDst[i].u.pointer.size = paParmsSrc[i].u.pointer.size;
181 if (paParmsDst[i].u.pointer.size > 0)
182 {
183 paParmsDst[i].u.pointer.addr = RTMemAlloc(paParmsDst[i].u.pointer.size);
184 if (!paParmsDst[i].u.pointer.addr)
185 {
186 rc = VERR_NO_MEMORY;
187 break;
188 }
189 }
190 }
191 else
192 {
193 /* No, but we have to check if there is enough room. */
194 if (paParmsDst[i].u.pointer.size < paParmsSrc[i].u.pointer.size)
195 {
196 rc = VERR_BUFFER_OVERFLOW;
197 break;
198 }
199 }
200
201 if (paParmsSrc[i].u.pointer.size)
202 {
203 if ( paParmsDst[i].u.pointer.addr
204 && paParmsDst[i].u.pointer.size)
205 {
206 memcpy(paParmsDst[i].u.pointer.addr,
207 paParmsSrc[i].u.pointer.addr,
208 RT_MIN(paParmsDst[i].u.pointer.size, paParmsSrc[i].u.pointer.size));
209 }
210 else
211 rc = VERR_INVALID_POINTER;
212 }
213 break;
214 }
215 default:
216 {
217 AssertMsgFailed(("Unknown HGCM type %u\n", paParmsSrc[i].type));
218 rc = VERR_INVALID_PARAMETER;
219 break;
220 }
221 }
222 if (RT_FAILURE(rc))
223 break;
224 }
225 return rc;
226 }
227
228 void cleanup()
229 {
230 if (m_paParms)
231 {
232 for (uint32_t i = 0; i < m_cParms; ++i)
233 {
234 switch (m_paParms[i].type)
235 {
236 case VBOX_HGCM_SVC_PARM_PTR:
237 if (m_paParms[i].u.pointer.size)
238 RTMemFree(m_paParms[i].u.pointer.addr);
239 break;
240 }
241 }
242 RTMemFree(m_paParms);
243 m_paParms = 0;
244 }
245 m_cParms = 0;
246 m_uMsg = 0;
247 }
248};
249
250class Client
251{
252public:
253 Client(uint32_t uClientId, VBOXHGCMCALLHANDLE hHandle = NULL,
254 uint32_t uMsg = 0, uint32_t cParms = 0, VBOXHGCMSVCPARM aParms[] = NULL)
255 : m_uClientId(uClientId)
256 , m_uProtocol(0)
257 , m_hHandle(hHandle)
258 , m_uMsg(uMsg)
259 , m_cParms(cParms)
260 , m_paParms(aParms) {}
261
262public:
263
264 VBOXHGCMCALLHANDLE handle(void) const { return m_hHandle; }
265 uint32_t message(void) const { return m_uMsg; }
266 uint32_t clientId(void) const { return m_uClientId; }
267 uint32_t protocol(void) const { return m_uProtocol; }
268
269public:
270
271 int setProtocol(uint32_t uProtocol) { m_uProtocol = uProtocol; return VINF_SUCCESS; }
272
273public:
274
275 int addMessageInfo(uint32_t uMsg, uint32_t cParms)
276 {
277 if (m_cParms != 3)
278 return VERR_INVALID_PARAMETER;
279
280 m_paParms[0].setUInt32(uMsg);
281 m_paParms[1].setUInt32(cParms);
282
283 return VINF_SUCCESS;
284 }
285 int addMessageInfo(const Message *pMessage)
286 {
287 AssertPtrReturn(pMessage, VERR_INVALID_POINTER);
288 if (m_cParms != 3)
289 return VERR_INVALID_PARAMETER;
290
291 m_paParms[0].setUInt32(pMessage->message());
292 m_paParms[1].setUInt32(pMessage->paramsCount());
293
294 return VINF_SUCCESS;
295 }
296 int addMessage(const Message *pMessage)
297 {
298 AssertPtrReturn(pMessage, VERR_INVALID_POINTER);
299 return pMessage->getData(m_uMsg, m_cParms, m_paParms);
300 }
301
302private:
303
304 uint32_t m_uClientId;
305 /** Optional protocol version the client uses. */
306 uint32_t m_uProtocol;
307 VBOXHGCMCALLHANDLE m_hHandle;
308 uint32_t m_uMsg;
309 uint32_t m_cParms;
310 PVBOXHGCMSVCPARM m_paParms;
311};
312
313template <class T>
314class AbstractService: public RTCNonCopyable
315{
316public:
317 /**
318 * @copydoc VBOXHGCMSVCLOAD
319 */
320 static DECLCALLBACK(int) svcLoad(VBOXHGCMSVCFNTABLE *pTable)
321 {
322 LogFlowFunc(("ptable = %p\n", pTable));
323 int rc = VINF_SUCCESS;
324
325 if (!VALID_PTR(pTable))
326 rc = VERR_INVALID_PARAMETER;
327 else
328 {
329 LogFlowFunc(("ptable->cbSize = %d, ptable->u32Version = 0x%08X\n", pTable->cbSize, pTable->u32Version));
330
331 if ( pTable->cbSize != sizeof (VBOXHGCMSVCFNTABLE)
332 || pTable->u32Version != VBOX_HGCM_SVC_VERSION)
333 rc = VERR_VERSION_MISMATCH;
334 else
335 {
336 std::auto_ptr<AbstractService> apService;
337 /* No exceptions may propagate outside. */
338 try
339 {
340 apService = std::auto_ptr<AbstractService>(new T(pTable->pHelpers));
341 } catch (int rcThrown)
342 {
343 rc = rcThrown;
344 } catch (...)
345 {
346 rc = VERR_UNRESOLVED_ERROR;
347 }
348
349 if (RT_SUCCESS(rc))
350 {
351 /*
352 * We don't need an additional client data area on the host,
353 * because we're a class which can have members for that :-).
354 */
355 pTable->cbClient = 0;
356
357 /* These functions are mandatory */
358 pTable->pfnUnload = svcUnload;
359 pTable->pfnConnect = svcConnect;
360 pTable->pfnDisconnect = svcDisconnect;
361 pTable->pfnCall = svcCall;
362 /* Clear obligatory functions. */
363 pTable->pfnHostCall = NULL;
364 pTable->pfnSaveState = NULL;
365 pTable->pfnLoadState = NULL;
366 pTable->pfnRegisterExtension = NULL;
367
368 /* Let the service itself initialize. */
369 rc = apService->init(pTable);
370
371 /* Only on success stop the auto release of the auto_ptr. */
372 if (RT_SUCCESS(rc))
373 pTable->pvService = apService.release();
374 }
375 }
376 }
377
378 LogFlowFunc(("returning %Rrc\n", rc));
379 return rc;
380 }
381 virtual ~AbstractService() {};
382
383protected:
384 explicit AbstractService(PVBOXHGCMSVCHELPERS pHelpers)
385 : m_pHelpers(pHelpers)
386 , m_pfnHostCallback(NULL)
387 , m_pvHostData(NULL)
388 {}
389 virtual int init(VBOXHGCMSVCFNTABLE *ptable) { return VINF_SUCCESS; }
390 virtual int uninit() { return VINF_SUCCESS; }
391 virtual int clientConnect(uint32_t u32ClientID, void *pvClient) = 0;
392 virtual int clientDisconnect(uint32_t u32ClientID, void *pvClient) = 0;
393 virtual void guestCall(VBOXHGCMCALLHANDLE callHandle, uint32_t u32ClientID, void *pvClient, uint32_t eFunction, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) = 0;
394 virtual int hostCall(uint32_t eFunction, uint32_t cParms, VBOXHGCMSVCPARM paParms[]) { return VINF_SUCCESS; }
395
396 /** Type definition for use in callback functions. */
397 typedef AbstractService SELF;
398 /** HGCM helper functions. */
399 PVBOXHGCMSVCHELPERS m_pHelpers;
400 /*
401 * Callback function supplied by the host for notification of updates
402 * to properties.
403 */
404 PFNHGCMSVCEXT m_pfnHostCallback;
405 /** User data pointer to be supplied to the host callback function. */
406 void *m_pvHostData;
407
408 /**
409 * @copydoc VBOXHGCMSVCFNTABLE::pfnUnload
410 * Simply deletes the service object
411 */
412 static DECLCALLBACK(int) svcUnload(void *pvService)
413 {
414 AssertLogRelReturn(VALID_PTR(pvService), VERR_INVALID_PARAMETER);
415 SELF *pSelf = reinterpret_cast<SELF *>(pvService);
416 int rc = pSelf->uninit();
417 AssertRC(rc);
418 if (RT_SUCCESS(rc))
419 delete pSelf;
420 return rc;
421 }
422
423 /**
424 * @copydoc VBOXHGCMSVCFNTABLE::pfnConnect
425 * Stub implementation of pfnConnect and pfnDisconnect.
426 */
427 static DECLCALLBACK(int) svcConnect(void *pvService,
428 uint32_t u32ClientID,
429 void *pvClient)
430 {
431 AssertLogRelReturn(VALID_PTR(pvService), VERR_INVALID_PARAMETER);
432 LogFlowFunc(("pvService=%p, u32ClientID=%u, pvClient=%p\n", pvService, u32ClientID, pvClient));
433 SELF *pSelf = reinterpret_cast<SELF *>(pvService);
434 int rc = pSelf->clientConnect(u32ClientID, pvClient);
435 LogFlowFunc(("rc=%Rrc\n", rc));
436 return rc;
437 }
438
439 /**
440 * @copydoc VBOXHGCMSVCFNTABLE::pfnConnect
441 * Stub implementation of pfnConnect and pfnDisconnect.
442 */
443 static DECLCALLBACK(int) svcDisconnect(void *pvService,
444 uint32_t u32ClientID,
445 void *pvClient)
446 {
447 AssertLogRelReturn(VALID_PTR(pvService), VERR_INVALID_PARAMETER);
448 LogFlowFunc(("pvService=%p, u32ClientID=%u, pvClient=%p\n", pvService, u32ClientID, pvClient));
449 SELF *pSelf = reinterpret_cast<SELF *>(pvService);
450 int rc = pSelf->clientDisconnect(u32ClientID, pvClient);
451 LogFlowFunc(("rc=%Rrc\n", rc));
452 return rc;
453 }
454
455 /**
456 * @copydoc VBOXHGCMSVCFNTABLE::pfnCall
457 * Wraps to the call member function
458 */
459 static DECLCALLBACK(void) svcCall(void * pvService,
460 VBOXHGCMCALLHANDLE callHandle,
461 uint32_t u32ClientID,
462 void *pvClient,
463 uint32_t u32Function,
464 uint32_t cParms,
465 VBOXHGCMSVCPARM paParms[])
466 {
467 AssertLogRelReturnVoid(VALID_PTR(pvService));
468 LogFlowFunc(("pvService=%p, callHandle=%p, u32ClientID=%u, pvClient=%p, u32Function=%u, cParms=%u, paParms=%p\n", pvService, callHandle, u32ClientID, pvClient, u32Function, cParms, paParms));
469 SELF *pSelf = reinterpret_cast<SELF *>(pvService);
470 pSelf->guestCall(callHandle, u32ClientID, pvClient, u32Function, cParms, paParms);
471 LogFlowFunc(("returning\n"));
472 }
473
474 /**
475 * @copydoc VBOXHGCMSVCFNTABLE::pfnHostCall
476 * Wraps to the hostCall member function
477 */
478 static DECLCALLBACK(int) svcHostCall(void *pvService,
479 uint32_t u32Function,
480 uint32_t cParms,
481 VBOXHGCMSVCPARM paParms[])
482 {
483 AssertLogRelReturn(VALID_PTR(pvService), VERR_INVALID_PARAMETER);
484 LogFlowFunc(("pvService=%p, u32Function=%u, cParms=%u, paParms=%p\n", pvService, u32Function, cParms, paParms));
485 SELF *pSelf = reinterpret_cast<SELF *>(pvService);
486 int rc = pSelf->hostCall(u32Function, cParms, paParms);
487 LogFlowFunc(("rc=%Rrc\n", rc));
488 return rc;
489 }
490
491 /**
492 * @copydoc VBOXHGCMSVCFNTABLE::pfnRegisterExtension
493 * Installs a host callback for notifications of property changes.
494 */
495 static DECLCALLBACK(int) svcRegisterExtension(void *pvService,
496 PFNHGCMSVCEXT pfnExtension,
497 void *pvExtension)
498 {
499 AssertLogRelReturn(VALID_PTR(pvService), VERR_INVALID_PARAMETER);
500 LogFlowFunc(("pvService=%p, pfnExtension=%p, pvExtention=%p\n", pvService, pfnExtension, pvExtension));
501 SELF *pSelf = reinterpret_cast<SELF *>(pvService);
502 pSelf->m_pfnHostCallback = pfnExtension;
503 pSelf->m_pvHostData = pvExtension;
504 return VINF_SUCCESS;
505 }
506};
507
508}
509
510#endif /* !___VBox_HostService_Service_h */
511
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