VirtualBox

source: vbox/trunk/src/VBox/GuestHost/OpenGL/util/vboxhgsmi.c@ 66295

Last change on this file since 66295 was 65381, checked in by vboxsync, 8 years ago

bugref:8282: Additions/linux: submit DRM driver to the Linux kernel: move all graphics device-related header files to a separate sub-directory and add that to the include path where they are needed. The intention is too be able to remove the VBox/ include folder in the DRM driver package.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 48.2 KB
Line 
1/* $Id: vboxhgsmi.c 65381 2017-01-20 09:23:53Z vboxsync $ */
2/** @file
3 * VBox HGCM connection
4 */
5
6/*
7 * Copyright (C) 2008-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//#define IN_GUEST
19#ifdef IN_GUEST /* entire file */
20
21#ifdef RT_OS_WINDOWS
22# include <iprt/win/windows.h>
23# include <ddraw.h>
24#else
25# include <sys/ioctl.h>
26# include <errno.h>
27# include <fcntl.h>
28# include <string.h>
29# include <unistd.h>
30#endif
31
32#include "cr_error.h"
33#include "cr_net.h"
34#include "cr_bufpool.h"
35#include "cr_mem.h"
36#include "cr_string.h"
37#include "cr_endian.h"
38#include "cr_threads.h"
39#include "net_internals.h"
40#include "cr_process.h"
41
42#include <iprt/thread.h>
43#include <iprt/assert.h>
44
45#include <VBoxCrHgsmi.h>
46#if 1 /** @todo Try use the Vbgl interface instead of talking directly to the driver? */
47# include <VBox/VBoxGuest.h>
48#else
49# include <VBox/VBoxGuestLib.h>
50#endif
51#include <VBox/HostServices/VBoxCrOpenGLSvc.h>
52
53#ifndef IN_GUEST
54# error "Hgsmi is supported for guest lib only!!"
55#endif
56
57#ifdef DEBUG_misha
58# ifdef CRASSERT
59# undef CRASSERT
60# endif
61# define CRASSERT Assert
62#endif
63
64#define VBOX_WITH_CRHGSMIPROFILE
65#ifdef VBOX_WITH_CRHGSMIPROFILE
66#include <iprt/time.h>
67#include <stdio.h>
68
69typedef struct VBOXCRHGSMIPROFILE
70{
71 uint64_t cStartTime;
72 uint64_t cStepsTime;
73 uint64_t cSteps;
74} VBOXCRHGSMIPROFILE, *PVBOXCRHGSMIPROFILE;
75
76#define VBOXCRHGSMIPROFILE_GET_TIME_NANO() RTTimeNanoTS()
77#define VBOXCRHGSMIPROFILE_GET_TIME_MILLI() RTTimeMilliTS()
78
79/* 10 sec */
80#define VBOXCRHGSMIPROFILE_LOG_STEP_TIME (10000000000.)
81
82DECLINLINE(void) vboxCrHgsmiProfileStart(PVBOXCRHGSMIPROFILE pProfile)
83{
84 pProfile->cStepsTime = 0;
85 pProfile->cSteps = 0;
86 pProfile->cStartTime = VBOXCRHGSMIPROFILE_GET_TIME_NANO();
87}
88
89DECLINLINE(void) vboxCrHgsmiProfileStep(PVBOXCRHGSMIPROFILE pProfile, uint64_t cStepTime)
90{
91 pProfile->cStepsTime += cStepTime;
92 ++pProfile->cSteps;
93}
94
95typedef struct VBOXCRHGSMIPROFILE_SCOPE
96{
97 uint64_t cStartTime;
98// bool bDisable;
99} VBOXCRHGSMIPROFILE_SCOPE, *PVBOXCRHGSMIPROFILE_SCOPE;
100
101static VBOXCRHGSMIPROFILE g_VBoxProfile;
102
103static void vboxCrHgsmiLog(char * szString, ...)
104{
105 char szBuffer[4096] = {0};
106 va_list pArgList;
107 va_start(pArgList, szString);
108 _vsnprintf(szBuffer, sizeof(szBuffer) / sizeof(szBuffer[0]), szString, pArgList);
109 va_end(pArgList);
110
111 VBoxCrHgsmiLog(szBuffer);
112}
113
114DECLINLINE(void) vboxCrHgsmiProfileLog(PVBOXCRHGSMIPROFILE pProfile, uint64_t cTime)
115{
116 uint64_t profileTime = cTime - pProfile->cStartTime;
117 double percent = ((double)100.0) * pProfile->cStepsTime / profileTime;
118 double cps = ((double)1000000000.) * pProfile->cSteps / profileTime;
119 vboxCrHgsmiLog("hgsmi: cps: %.1f, host %.1f%%\n", cps, percent);
120}
121
122DECLINLINE(void) vboxCrHgsmiProfileScopeEnter(PVBOXCRHGSMIPROFILE_SCOPE pScope)
123{
124// pScope->bDisable = false;
125 pScope->cStartTime = VBOXCRHGSMIPROFILE_GET_TIME_NANO();
126}
127
128DECLINLINE(void) vboxCrHgsmiProfileScopeExit(PVBOXCRHGSMIPROFILE_SCOPE pScope)
129{
130// if (!pScope->bDisable)
131 {
132 uint64_t cTime = VBOXCRHGSMIPROFILE_GET_TIME_NANO();
133 vboxCrHgsmiProfileStep(&g_VBoxProfile, cTime - pScope->cStartTime);
134 if (VBOXCRHGSMIPROFILE_LOG_STEP_TIME < cTime - g_VBoxProfile.cStartTime)
135 {
136 vboxCrHgsmiProfileLog(&g_VBoxProfile, cTime);
137 vboxCrHgsmiProfileStart(&g_VBoxProfile);
138 }
139 }
140}
141
142
143#define VBOXCRHGSMIPROFILE_INIT() vboxCrHgsmiProfileStart(&g_VBoxProfile)
144#define VBOXCRHGSMIPROFILE_TERM() do {} while (0)
145
146#define VBOXCRHGSMIPROFILE_FUNC_PROLOGUE() \
147 VBOXCRHGSMIPROFILE_SCOPE __vboxCrHgsmiProfileScope; \
148 vboxCrHgsmiProfileScopeEnter(&__vboxCrHgsmiProfileScope);
149
150#define VBOXCRHGSMIPROFILE_FUNC_EPILOGUE() \
151 vboxCrHgsmiProfileScopeExit(&__vboxCrHgsmiProfileScope); \
152
153
154#else
155#define VBOXCRHGSMIPROFILE_INIT() do {} while (0)
156#define VBOXCRHGSMIPROFILE_TERM() do {} while (0)
157#define VBOXCRHGSMIPROFILE_FUNC_PROLOGUE() do {} while (0)
158#define VBOXCRHGSMIPROFILE_FUNC_EPILOGUE() do {} while (0)
159#endif
160
161
162typedef struct {
163 int initialized;
164 int num_conns;
165 CRConnection **conns;
166#ifdef CHROMIUM_THREADSAFE
167 CRmutex mutex;
168 CRmutex recvmutex;
169#endif
170 CRNetReceiveFuncList *recv_list;
171 CRNetCloseFuncList *close_list;
172 CRBufferPool *mempool;
173#ifdef RT_OS_WINDOWS
174 HANDLE hGuestDrv;
175#else
176 int iGuestDrv;
177#endif
178} CRVBOXHGSMIDATA;
179
180#define CR_VBOXHGSMI_BUFFER_MAGIC 0xEDCBA123
181
182typedef struct CRVBOXHGSMIBUFFER {
183 uint32_t magic;
184 union
185 {
186 struct
187 {
188 uint32_t cbLock : 31;
189 uint32_t bIsBuf : 1;
190 };
191 uint32_t Value;
192 };
193 union
194 {
195 PVBOXUHGSMI_BUFFER pBuffer;
196 uint32_t u32Len; /* <- sys mem buf specific */
197 uint64_t u64Align;
198 };
199} CRVBOXHGSMIBUFFER;
200
201typedef struct CRVBOXHGSMI_CLIENT {
202 PVBOXUHGSMI pHgsmi;
203 PVBOXUHGSMI_BUFFER pCmdBuffer;
204 PVBOXUHGSMI_BUFFER pHGBuffer;
205 void *pvHGBuffer;
206 CRBufferPool *bufpool;
207} CRVBOXHGSMI_CLIENT, *PCRVBOXHGSMI_CLIENT;
208
209
210static CRVBOXHGSMIDATA g_crvboxhgsmi = {0,};
211
212DECLINLINE(PCRVBOXHGSMI_CLIENT) _crVBoxHGSMIClientGet()
213{
214 PCRVBOXHGSMI_CLIENT pClient = (PCRVBOXHGSMI_CLIENT)VBoxCrHgsmiQueryClient();
215 Assert(pClient);
216 return pClient;
217}
218
219#ifndef RT_OS_WINDOWS
220 #define TRUE true
221 #define FALSE false
222 #define INVALID_HANDLE_VALUE (-1)
223#endif
224
225/* Some forward declarations */
226static void _crVBoxHGSMIReceiveMessage(CRConnection *conn, PCRVBOXHGSMI_CLIENT pClient);
227
228/* we still use hgcm for establishing the connection
229 * @todo: join the common code of HGSMI & HGCM */
230/**
231 * Send an HGCM request
232 *
233 * @return VBox status code
234 * @param pvData Data pointer
235 * @param cbData Data size
236 */
237/** @todo use vbglR3DoIOCtl here instead */
238static int crVBoxHGCMCall(void *pvData, unsigned cbData)
239{
240#ifdef IN_GUEST
241
242# ifdef RT_OS_WINDOWS
243 DWORD cbReturned;
244
245 if (DeviceIoControl (g_crvboxhgsmi.hGuestDrv,
246 VBOXGUEST_IOCTL_HGCM_CALL(cbData),
247 pvData, cbData,
248 pvData, cbData,
249 &cbReturned,
250 NULL))
251 {
252 return VINF_SUCCESS;
253 }
254 Assert(0);
255 crDebug("vboxCall failed with %x\n", GetLastError());
256 return VERR_NOT_SUPPORTED;
257# else
258 int rc;
259# if defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
260 VBGLBIGREQ Hdr;
261 Hdr.u32Magic = VBGLBIGREQ_MAGIC;
262 Hdr.cbData = cbData;
263 Hdr.pvDataR3 = pvData;
264# if HC_ARCH_BITS == 32
265 Hdr.u32Padding = 0;
266# endif
267 rc = ioctl(g_crvboxhgsmi.iGuestDrv, VBOXGUEST_IOCTL_HGCM_CALL(cbData), &Hdr);
268# else
269 rc = ioctl(g_crvboxhgsmi.iGuestDrv, VBOXGUEST_IOCTL_HGCM_CALL(cbData), pvData);
270# endif
271# ifdef RT_OS_LINUX
272 if (rc == 0)
273# else
274 if (rc >= 0)
275# endif
276 {
277 return VINF_SUCCESS;
278 }
279# ifdef RT_OS_LINUX
280 if (rc >= 0) /* positive values are negated VBox error status codes. */
281 {
282 crWarning("vboxCall failed with VBox status code %d\n", -rc);
283 if (rc==VINF_INTERRUPTED)
284 {
285 RTMSINTERVAL sl;
286 int i;
287
288 for (i=0, sl=50; i<6; i++, sl=sl*2)
289 {
290 RTThreadSleep(sl);
291 rc = ioctl(g_crvboxhgsmi.iGuestDrv, VBOXGUEST_IOCTL_HGCM_CALL(cbData), pvData);
292 if (rc==0)
293 {
294 crWarning("vboxCall retry(%i) succeeded", i+1);
295 return VINF_SUCCESS;
296 }
297 else if (rc==VINF_INTERRUPTED)
298 {
299 continue;
300 }
301 else
302 {
303 crWarning("vboxCall retry(%i) failed with VBox status code %d", i+1, -rc);
304 break;
305 }
306 }
307 }
308 }
309 else
310# endif
311 crWarning("vboxCall failed with %x\n", errno);
312 return VERR_NOT_SUPPORTED;
313# endif /*#ifdef RT_OS_WINDOWS*/
314
315#else /*#ifdef IN_GUEST*/
316 crError("crVBoxHGCMCall called on host side!");
317 CRASSERT(FALSE);
318 return VERR_NOT_SUPPORTED;
319#endif
320}
321
322
323/* add sizeof header + page align */
324#define CRVBOXHGSMI_PAGE_ALIGN(_s) (((_s) + 0xfff) & ~0xfff)
325#define CRVBOXHGSMI_BUF_HDR_SIZE() (sizeof (CRVBOXHGSMIBUFFER))
326#define CRVBOXHGSMI_BUF_SIZE(_s) CRVBOXHGSMI_PAGE_ALIGN((_s) + CRVBOXHGSMI_BUF_HDR_SIZE())
327#define CRVBOXHGSMI_BUF_LOCK_SIZE(_s) ((_s) + CRVBOXHGSMI_BUF_HDR_SIZE())
328#define CRVBOXHGSMI_BUF_DATA(_p) ((void*)(((CRVBOXHGSMIBUFFER*)(_p)) + 1))
329#define CRVBOXHGSMI_BUF_HDR(_p) (((CRVBOXHGSMIBUFFER*)(_p)) - 1)
330#define CRVBOXHGSMI_BUF_OFFSET(_st2, _st1) ((uint32_t)(((uint8_t*)(_st2)) - ((uint8_t*)(_st1))))
331
332static CRVBOXHGSMIHDR *_crVBoxHGSMICmdBufferLock(PCRVBOXHGSMI_CLIENT pClient, uint32_t cbBuffer)
333{
334 /* in theory it is OK to use one cmd buffer for async cmd submission
335 * because bDiscard flag should result in allocating a new memory backend if the
336 * allocation is still in use.
337 * However, NOTE: since one and the same semaphore sync event is used for completion notification,
338 * for the notification mechanism working as expected
339 * 1. host must complete commands in the same order as it receives them
340 * (to avoid situation when guest receives notification for another command completion)
341 * 2. guest must eventually wait for command completion unless he specified bDoNotSignalCompletion
342 * 3. guest must wait for command completion in the same order as it submits them
343 * in case we can not satisfy any of the above, we should introduce multiple command buffers */
344 CRVBOXHGSMIHDR * pHdr;
345 VBOXUHGSMI_BUFFER_LOCK_FLAGS fFlags;
346 int rc;
347 fFlags.Value = 0;
348 fFlags.bDiscard = 1;
349 rc = pClient->pCmdBuffer->pfnLock(pClient->pCmdBuffer, 0, cbBuffer, fFlags, (void**)&pHdr);
350 AssertRC(rc);
351 if (RT_SUCCESS(rc))
352 return pHdr;
353
354 crWarning("Failed to Lock the command buffer of size(%d), rc(%d)\n", cbBuffer, rc);
355 return NULL;
356}
357
358static CRVBOXHGSMIHDR *_crVBoxHGSMICmdBufferLockRo(PCRVBOXHGSMI_CLIENT pClient, uint32_t cbBuffer)
359{
360 /* in theory it is OK to use one cmd buffer for async cmd submission
361 * because bDiscard flag should result in allocating a new memory backend if the
362 * allocation is still in use.
363 * However, NOTE: since one and the same semaphore sync event is used for completion notification,
364 * for the notification mechanism working as expected
365 * 1. host must complete commands in the same order as it receives them
366 * (to avoid situation when guest receives notification for another command completion)
367 * 2. guest must eventually wait for command completion unless he specified bDoNotSignalCompletion
368 * 3. guest must wait for command completion in the same order as it submits them
369 * in case we can not satisfy any of the above, we should introduce multiple command buffers */
370 CRVBOXHGSMIHDR * pHdr;
371 VBOXUHGSMI_BUFFER_LOCK_FLAGS fFlags;
372 int rc;
373 fFlags.Value = 0;
374 fFlags.bReadOnly = 1;
375 rc = pClient->pCmdBuffer->pfnLock(pClient->pCmdBuffer, 0, cbBuffer, fFlags, (void**)&pHdr);
376 AssertRC(rc);
377 if (RT_FAILURE(rc))
378 crWarning("Failed to Lock the command buffer of size(%d), rc(%d)\n", cbBuffer, rc);
379 return pHdr;
380}
381
382static void _crVBoxHGSMICmdBufferUnlock(PCRVBOXHGSMI_CLIENT pClient)
383{
384 int rc = pClient->pCmdBuffer->pfnUnlock(pClient->pCmdBuffer);
385 AssertRC(rc);
386 if (RT_FAILURE(rc))
387 crWarning("Failed to Unlock the command buffer rc(%d)\n", rc);
388}
389
390static int32_t _crVBoxHGSMICmdBufferGetRc(PCRVBOXHGSMI_CLIENT pClient)
391{
392 CRVBOXHGSMIHDR * pHdr;
393 VBOXUHGSMI_BUFFER_LOCK_FLAGS fFlags;
394 int rc;
395
396 fFlags.Value = 0;
397 fFlags.bReadOnly = 1;
398 rc = pClient->pCmdBuffer->pfnLock(pClient->pCmdBuffer, 0, sizeof (*pHdr), fFlags, (void**)&pHdr);
399 AssertRC(rc);
400 if (RT_FAILURE(rc))
401 {
402 crWarning("Failed to Lock the command buffer of size(%d), rc(%d)\n", sizeof (*pHdr), rc);
403 return rc;
404 }
405
406 rc = pHdr->result;
407 AssertRC(rc);
408 pClient->pCmdBuffer->pfnUnlock(pClient->pCmdBuffer);
409
410 return rc;
411}
412
413DECLINLINE(PVBOXUHGSMI_BUFFER) _crVBoxHGSMIRecvBufGet(PCRVBOXHGSMI_CLIENT pClient)
414{
415 if (pClient->pvHGBuffer)
416 {
417 int rc = pClient->pHGBuffer->pfnUnlock(pClient->pHGBuffer);
418 if (RT_FAILURE(rc))
419 {
420 return NULL;
421 }
422 pClient->pvHGBuffer = NULL;
423 }
424 return pClient->pHGBuffer;
425}
426
427DECLINLINE(void*) _crVBoxHGSMIRecvBufData(PCRVBOXHGSMI_CLIENT pClient, uint32_t cbBuffer)
428{
429 VBOXUHGSMI_BUFFER_LOCK_FLAGS fFlags;
430 int rc;
431 Assert(!pClient->pvHGBuffer);
432 fFlags.Value = 0;
433 rc = pClient->pHGBuffer->pfnLock(pClient->pHGBuffer, 0, cbBuffer, fFlags, &pClient->pvHGBuffer);
434 AssertRC(rc);
435 if (RT_SUCCESS(rc))
436 {
437 return pClient->pvHGBuffer;
438 }
439 return NULL;
440}
441
442
443static PVBOXUHGSMI_BUFFER _crVBoxHGSMIBufFromMemPtr(void *pvBuf)
444{
445 CRVBOXHGSMIBUFFER *pHdr = CRVBOXHGSMI_BUF_HDR(pvBuf);
446 PVBOXUHGSMI_BUFFER pBuf;
447 int rc;
448
449 CRASSERT(pHdr->magic == CR_VBOXHGSMI_BUFFER_MAGIC);
450 pBuf = pHdr->pBuffer;
451 rc = pBuf->pfnUnlock(pBuf);
452 AssertRC(rc);
453 if (RT_FAILURE(rc))
454 {
455 return NULL;
456 }
457 return pBuf;
458}
459
460static PVBOXUHGSMI_BUFFER _crVBoxHGSMIBufAlloc(PCRVBOXHGSMI_CLIENT pClient, uint32_t cbSize)
461{
462 PVBOXUHGSMI_BUFFER buf;
463 int rc;
464
465 buf = (PVBOXUHGSMI_BUFFER ) crBufferPoolPop(pClient->bufpool, cbSize);
466
467 if (!buf)
468 {
469 crDebug("Buffer pool %p was empty; allocating new %d byte buffer.",
470 (void *) pClient->bufpool,
471 cbSize);
472 rc = pClient->pHgsmi->pfnBufferCreate(pClient->pHgsmi, cbSize,
473 VBOXUHGSMI_SYNCHOBJECT_TYPE_NONE, NULL,
474 &buf);
475 AssertRC(rc);
476 if (RT_FAILURE(rc))
477 crWarning("Failed to Create a buffer of size(%d), rc(%d)\n", cbSize, rc);
478 }
479 return buf;
480}
481
482static void _crVBoxHGSMIBufFree(PCRVBOXHGSMI_CLIENT pClient, PVBOXUHGSMI_BUFFER pBuf)
483{
484 crBufferPoolPush(pClient->bufpool, pBuf, pBuf->cbBuffer);
485}
486
487static CRVBOXHGSMIBUFFER* _crVBoxHGSMISysMemAlloc(uint32_t cbSize)
488{
489 CRVBOXHGSMIBUFFER *pBuf;
490#ifdef CHROMIUM_THREADSAFE
491 crLockMutex(&g_crvboxhgsmi.mutex);
492#endif
493 pBuf = crBufferPoolPop(g_crvboxhgsmi.mempool, cbSize);
494 if (!pBuf)
495 {
496 pBuf = (CRVBOXHGSMIBUFFER*)crAlloc(CRVBOXHGSMI_BUF_HDR_SIZE() + cbSize);
497 Assert(pBuf);
498 if (pBuf)
499 {
500 pBuf->magic = CR_VBOXHGSMI_BUFFER_MAGIC;
501 pBuf->cbLock = cbSize;
502 pBuf->bIsBuf = false;
503 pBuf->u32Len = 0;
504 }
505 }
506#ifdef CHROMIUM_THREADSAFE
507 crUnlockMutex(&g_crvboxhgsmi.mutex);
508#endif
509 return pBuf;
510}
511
512static void *_crVBoxHGSMIDoAlloc(PCRVBOXHGSMI_CLIENT pClient, uint32_t cbSize)
513{
514 PVBOXUHGSMI_BUFFER buf;
515 CRVBOXHGSMIBUFFER *pData = NULL;
516 int rc;
517
518 buf = _crVBoxHGSMIBufAlloc(pClient, CRVBOXHGSMI_BUF_SIZE(cbSize));
519 Assert(buf);
520 if (buf)
521 {
522 VBOXUHGSMI_BUFFER_LOCK_FLAGS fFlags;
523 buf->pvUserData = pClient;
524 fFlags.Value = 0;
525 fFlags.bDiscard = 1;
526 rc = buf->pfnLock(buf, 0, CRVBOXHGSMI_BUF_LOCK_SIZE(cbSize), fFlags, (void**)&pData);
527 if (RT_SUCCESS(rc))
528 {
529 pData->magic = CR_VBOXHGSMI_BUFFER_MAGIC;
530 pData->cbLock = CRVBOXHGSMI_BUF_LOCK_SIZE(cbSize);
531 pData->bIsBuf = true;
532 pData->pBuffer = buf;
533 }
534 else
535 {
536 crWarning("Failed to Lock the buffer, rc(%d)\n", rc);
537 }
538 }
539
540 return CRVBOXHGSMI_BUF_DATA(pData);
541
542}
543static void *crVBoxHGSMIAlloc(CRConnection *conn)
544{
545 PCRVBOXHGSMI_CLIENT pClient;
546 void *pvBuf;
547
548 VBOXCRHGSMIPROFILE_FUNC_PROLOGUE();
549
550 pClient = _crVBoxHGSMIClientGet();
551 pvBuf = _crVBoxHGSMIDoAlloc(pClient, conn->buffer_size);
552 Assert(pvBuf);
553
554 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
555
556 return pvBuf;
557}
558
559DECLINLINE(void) _crVBoxHGSMIFillCmd(VBOXUHGSMI_BUFFER_SUBMIT *pSubm, PCRVBOXHGSMI_CLIENT pClient, uint32_t cbData)
560{
561 pSubm->pBuf = pClient->pCmdBuffer;
562 pSubm->offData = 0;
563 pSubm->cbData = cbData;
564 pSubm->fFlags.Value = 0;
565 pSubm->fFlags.bDoNotRetire = 1;
566// pSubm->fFlags.bDoNotSignalCompletion = 1; /* <- we do not need that actually since
567// * in case we want completion,
568// * we will block in _crVBoxHGSMICmdBufferGetRc (when locking the buffer)
569// * which is needed for getting the result */
570}
571
572#ifdef RT_OS_WINDOWS
573#define CRVBOXHGSMI_BUF_WAIT(_pBub) WaitForSingleObject((_pBub)->hSynch, INFINITE);
574#else
575# error "Port Me!!"
576#endif
577
578DECLINLINE(void) _crVBoxHGSMIWaitCmd(PCRVBOXHGSMI_CLIENT pClient)
579{
580 int rc = CRVBOXHGSMI_BUF_WAIT(pClient->pCmdBuffer);
581 Assert(rc == 0);
582}
583
584static void _crVBoxHGSMIWriteExact(CRConnection *conn, PCRVBOXHGSMI_CLIENT pClient, PVBOXUHGSMI_BUFFER pBuf, uint32_t offStart, unsigned int len)
585{
586 int rc;
587 int32_t callRes;
588 VBOXUHGSMI_BUFFER_SUBMIT aSubmit[2];
589
590#ifdef IN_GUEST
591 if (conn->u32InjectClientID)
592 {
593 CRVBOXHGSMIINJECT *parms = (CRVBOXHGSMIINJECT *)_crVBoxHGSMICmdBufferLock(pClient, sizeof (*parms));
594 Assert(parms);
595 if (!parms)
596 {
597 return;
598 }
599
600 parms->hdr.result = VERR_WRONG_ORDER;
601 parms->hdr.u32ClientID = conn->u32ClientID;
602 parms->hdr.u32Function = SHCRGL_GUEST_FN_INJECT;
603// parms->hdr.u32Reserved = 0;
604
605 parms->u32ClientID = conn->u32InjectClientID;
606
607 parms->iBuffer = 1;
608 _crVBoxHGSMICmdBufferUnlock(pClient);
609
610 _crVBoxHGSMIFillCmd(&aSubmit[0], pClient, sizeof (*parms));
611
612 aSubmit[1].pBuf = pBuf;
613 aSubmit[1].offData = offStart;
614 aSubmit[1].cbData = len;
615 aSubmit[1].fFlags.Value = 0;
616 aSubmit[1].fFlags.bHostReadOnly = 1;
617
618 rc = pClient->pHgsmi->pfnBufferSubmitAsynch(pClient->pHgsmi, aSubmit, 2);
619 AssertRC(rc);
620 if (RT_SUCCESS(rc))
621 {
622 _crVBoxHGSMIWaitCmd(pClient);
623 /* @todo: do we need to wait for completion actually?
624 * NOTE: in case we do not need completion,
625 * we MUST specify bDoNotSignalCompletion flag for the command buffer */
626// CRVBOXHGSMI_BUF_WAIT(pClient->pCmdBuffer);
627
628 callRes = _crVBoxHGSMICmdBufferGetRc(pClient);
629 }
630 }
631 else
632#endif
633 {
634 CRVBOXHGSMIWRITE * parms = (CRVBOXHGSMIWRITE *)_crVBoxHGSMICmdBufferLock(pClient, sizeof (*parms));;
635
636 parms->hdr.result = VERR_WRONG_ORDER;
637 parms->hdr.u32ClientID = conn->u32ClientID;
638 parms->hdr.u32Function = SHCRGL_GUEST_FN_WRITE;
639// parms->hdr.u32Reserved = 0;
640
641 parms->iBuffer = 1;
642 _crVBoxHGSMICmdBufferUnlock(pClient);
643
644 _crVBoxHGSMIFillCmd(&aSubmit[0], pClient, sizeof (*parms));
645
646 aSubmit[1].pBuf = pBuf;
647 aSubmit[1].offData = offStart;
648 aSubmit[1].cbData = len;
649 aSubmit[1].fFlags.Value = 0;
650 aSubmit[1].fFlags.bHostReadOnly = 1;
651
652 rc = pClient->pHgsmi->pfnBufferSubmitAsynch(pClient->pHgsmi, aSubmit, 2);
653 AssertRC(rc);
654 if (RT_SUCCESS(rc))
655 {
656 _crVBoxHGSMIWaitCmd(pClient);
657 /* @todo: do we need to wait for completion actually?
658 * NOTE: in case we do not need completion,
659 * we MUST specify bDoNotSignalCompletion flag for the command buffer */
660// CRVBOXHGSMI_BUF_WAIT(pClient->pCmdBuffer);
661
662 callRes = _crVBoxHGSMICmdBufferGetRc(pClient);
663 }
664 }
665
666 if (RT_FAILURE(rc) || RT_FAILURE(callRes))
667 {
668 crWarning("SHCRGL_GUEST_FN_WRITE failed with %x %x\n", rc, callRes);
669 }
670}
671
672static void crVBoxHGSMIWriteExact(CRConnection *conn, const void *buf, unsigned int len)
673{
674 VBOXCRHGSMIPROFILE_FUNC_PROLOGUE();
675 Assert(0);
676
677 CRASSERT(0);
678// PCRVBOXHGSMI_CLIENT pClient;
679// PVBOXUHGSMI_BUFFER pBuf = _crVBoxHGSMIBufFromMemPtr(buf);
680// if (!pBuf)
681// return;
682// pClient = (PCRVBOXHGSMI_CLIENT)pBuf->pvUserData;
683// _crVBoxHGSMIWriteExact(conn, pClient, pBuf, 0, len);
684// _crVBoxHGSMIBufFree(pClient, pBuf);
685 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
686}
687
688static void _crVBoxHGSMIPollHost(CRConnection *conn, PCRVBOXHGSMI_CLIENT pClient)
689{
690 CRVBOXHGSMIREAD *parms = (CRVBOXHGSMIREAD *)_crVBoxHGSMICmdBufferLock(pClient, sizeof (*parms));
691 int rc;
692 VBOXUHGSMI_BUFFER_SUBMIT aSubmit[2];
693 PVBOXUHGSMI_BUFFER pRecvBuffer;
694 uint32_t cbBuffer;
695
696 Assert(parms);
697
698 parms->hdr.result = VERR_WRONG_ORDER;
699 parms->hdr.u32ClientID = conn->u32ClientID;
700 parms->hdr.u32Function = SHCRGL_GUEST_FN_READ;
701// parms->hdr.u32Reserved = 0;
702
703 CRASSERT(!conn->pBuffer); //make sure there's no data to process
704 parms->iBuffer = 1;
705 parms->cbBuffer = 0;
706
707 _crVBoxHGSMICmdBufferUnlock(pClient);
708
709 pRecvBuffer = _crVBoxHGSMIRecvBufGet(pClient);
710 Assert(pRecvBuffer);
711 if (!pRecvBuffer)
712 return;
713
714 _crVBoxHGSMIFillCmd(&aSubmit[0], pClient, sizeof (*parms));
715
716 aSubmit[1].pBuf = pRecvBuffer;
717 aSubmit[1].offData = 0;
718 aSubmit[1].cbData = pRecvBuffer->cbBuffer;
719 aSubmit[1].fFlags.Value = 0;
720 aSubmit[1].fFlags.bHostWriteOnly = 1;
721
722 rc = pClient->pHgsmi->pfnBufferSubmitAsynch(pClient->pHgsmi, aSubmit, 2);
723 AssertRC(rc);
724 if (RT_FAILURE(rc))
725 {
726 crWarning("pfnBufferSubmitAsynch failed with %d \n", rc);
727 return;
728 }
729
730 _crVBoxHGSMIWaitCmd(pClient);
731
732 parms = (CRVBOXHGSMIREAD *)_crVBoxHGSMICmdBufferLockRo(pClient, sizeof (*parms));
733 Assert(parms);
734 if (!parms)
735 {
736 crWarning("_crVBoxHGSMICmdBufferLockRo failed\n");
737 return;
738 }
739
740 if (RT_SUCCESS(parms->hdr.result))
741 cbBuffer = parms->cbBuffer;
742 else
743 cbBuffer = 0;
744
745 _crVBoxHGSMICmdBufferUnlock(pClient);
746
747 if (cbBuffer)
748 {
749 void *pvData = _crVBoxHGSMIRecvBufData(pClient, cbBuffer);
750 Assert(pvData);
751 if (pvData)
752 {
753 conn->pBuffer = pvData;
754 conn->cbBuffer = cbBuffer;
755 }
756 }
757}
758
759static void _crVBoxHGSMIReadExact(CRConnection *conn, PCRVBOXHGSMI_CLIENT pClient)
760{
761 _crVBoxHGSMIPollHost(conn, pClient);
762
763 if (conn->cbBuffer)
764 _crVBoxHGSMIReceiveMessage(conn, pClient);
765}
766
767/* Same as crVBoxHGCMWriteExact, but combined with read of writeback data.
768 * This halves the number of HGCM calls we do,
769 * most likely crVBoxHGCMPollHost shouldn't be called at all now.
770 */
771static void
772_crVBoxHGSMIWriteReadExact(CRConnection *conn, PCRVBOXHGSMI_CLIENT pClient, void *buf, uint32_t offBuffer, unsigned int len, bool bIsBuffer)
773{
774 CRVBOXHGSMIWRITEREAD *parms = (CRVBOXHGSMIWRITEREAD*)_crVBoxHGSMICmdBufferLock(pClient, sizeof (*parms));
775 int rc;
776 VBOXUHGSMI_BUFFER_SUBMIT aSubmit[3];
777 PVBOXUHGSMI_BUFFER pBuf = NULL;
778 VBOXUHGSMI_BUFFER_LOCK_FLAGS fFlags;
779// uint32_t cbBuffer;
780
781 parms->hdr.result = VERR_WRONG_ORDER;
782 parms->hdr.u32ClientID = conn->u32ClientID;
783 parms->hdr.u32Function = SHCRGL_GUEST_FN_WRITE_READ;
784// parms->hdr.u32Reserved = 0;
785
786 parms->iBuffer = 1;
787
788 CRASSERT(!conn->pBuffer); //make sure there's no data to process
789 parms->iWriteback = 2;
790 parms->cbWriteback = 0;
791
792 _crVBoxHGSMICmdBufferUnlock(pClient);
793
794 if (!bIsBuffer)
795 {
796 void *pvBuf;
797 pBuf = _crVBoxHGSMIBufAlloc(pClient, len);
798 Assert(pBuf);
799 if (!pBuf)
800 return;
801
802 Assert(!offBuffer);
803
804 offBuffer = 0;
805 fFlags.Value = 0;
806 fFlags.bDiscard = 1;
807 fFlags.bWriteOnly = 1;
808 rc = pBuf->pfnLock(pBuf, 0, len, fFlags, &pvBuf);
809 AssertRC(rc);
810 if (RT_SUCCESS(rc))
811 {
812 memcpy(pvBuf, buf, len);
813 rc = pBuf->pfnUnlock(pBuf);
814 AssertRC(rc);
815 CRASSERT(RT_SUCCESS(rc));
816 }
817 else
818 {
819 _crVBoxHGSMIBufFree(pClient, pBuf);
820 return;
821 }
822 }
823 else
824 {
825 pBuf = (PVBOXUHGSMI_BUFFER)buf;
826 }
827
828 do
829 {
830 PVBOXUHGSMI_BUFFER pRecvBuffer = _crVBoxHGSMIRecvBufGet(pClient);
831 Assert(pRecvBuffer);
832 if (!pRecvBuffer)
833 return;
834
835 _crVBoxHGSMIFillCmd(&aSubmit[0], pClient, sizeof (*parms));
836
837 aSubmit[1].pBuf = pBuf;
838 aSubmit[1].offData = offBuffer;
839 aSubmit[1].cbData = len;
840 aSubmit[1].fFlags.Value = 0;
841 aSubmit[1].fFlags.bHostReadOnly = 1;
842
843 aSubmit[2].pBuf = pRecvBuffer;
844 aSubmit[2].offData = 0;
845 aSubmit[2].cbData = pRecvBuffer->cbBuffer;
846 aSubmit[2].fFlags.Value = 0;
847
848 rc = pClient->pHgsmi->pfnBufferSubmitAsynch(pClient->pHgsmi, aSubmit, 3);
849 AssertRC(rc);
850 if (RT_FAILURE(rc))
851 {
852 crWarning("pfnBufferSubmitAsynch failed with %d \n", rc);
853 break;
854 }
855
856 _crVBoxHGSMIWaitCmd(pClient);
857
858 parms = (CRVBOXHGSMIWRITEREAD *)_crVBoxHGSMICmdBufferLockRo(pClient, sizeof (*parms));
859 Assert(parms);
860 if (parms)
861 {
862 uint32_t cbWriteback = parms->cbWriteback;
863 rc = parms->hdr.result;
864 _crVBoxHGSMICmdBufferUnlock(pClient);
865#ifdef DEBUG
866 parms = NULL;
867#endif
868 if (RT_SUCCESS(rc))
869 {
870 if (cbWriteback)
871 {
872 void *pvData = _crVBoxHGSMIRecvBufData(pClient, cbWriteback);
873 Assert(pvData);
874 if (pvData)
875 {
876 conn->pBuffer = pvData;
877 conn->cbBuffer = cbWriteback;
878 _crVBoxHGSMIReceiveMessage(conn, pClient);
879 }
880 }
881 }
882 else if (VERR_BUFFER_OVERFLOW == rc)
883 {
884 PVBOXUHGSMI_BUFFER pOldBuf = pClient->pHGBuffer;
885 Assert(!pClient->pvHGBuffer);
886 CRASSERT(cbWriteback>pClient->pHGBuffer->cbBuffer);
887 crDebug("Reallocating host buffer from %d to %d bytes", conn->cbHostBufferAllocated, cbWriteback);
888
889 rc = pClient->pHgsmi->pfnBufferCreate(pClient->pHgsmi, CRVBOXHGSMI_PAGE_ALIGN(cbWriteback),
890 VBOXUHGSMI_SYNCHOBJECT_TYPE_NONE, NULL, &pClient->pHGBuffer);
891 AssertRC(rc);
892 CRASSERT(RT_SUCCESS(rc));
893 if (RT_SUCCESS(rc))
894 {
895 rc = pOldBuf->pfnDestroy(pOldBuf);
896 CRASSERT(RT_SUCCESS(rc));
897
898 _crVBoxHGSMIReadExact(conn, pClient/*, cbWriteback*/);
899 }
900 }
901 else
902 {
903 crWarning("SHCRGL_GUEST_FN_WRITE_READ (%i) failed with %x \n", len, rc);
904 }
905 }
906 else
907 {
908 crWarning("_crVBoxHGSMICmdBufferLockRo failed\n");
909 break;
910 }
911 } while (0);
912
913 if (!bIsBuffer)
914 _crVBoxHGSMIBufFree(pClient, pBuf);
915}
916
917static void crVBoxHGSMISend(CRConnection *conn, void **bufp,
918 const void *start, unsigned int len)
919{
920 PCRVBOXHGSMI_CLIENT pClient;
921 PVBOXUHGSMI_BUFFER pBuf;
922
923 VBOXCRHGSMIPROFILE_FUNC_PROLOGUE();
924
925 if (!bufp) /* We're sending a user-allocated buffer. */
926 {
927 pClient = _crVBoxHGSMIClientGet();
928#ifndef IN_GUEST
929 //@todo remove temp buffer allocation in unpacker
930 /* we're at the host side, so just store data until guest polls us */
931 _crVBoxHGCMWriteBytes(conn, start, len);
932#else
933 CRASSERT(!conn->u32InjectClientID);
934 crDebug("SHCRGL: sending userbuf with %d bytes\n", len);
935 _crVBoxHGSMIWriteReadExact(conn, pClient, (void*)start, 0, len, false);
936#endif
937 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
938 return;
939 }
940
941 /* The region [start .. start + len + 1] lies within a buffer that
942 * was allocated with crVBoxHGCMAlloc() and can be put into the free
943 * buffer pool when we're done sending it.
944 */
945
946 pBuf = _crVBoxHGSMIBufFromMemPtr(*bufp);
947 Assert(pBuf);
948 if (!pBuf)
949 {
950 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
951 return;
952 }
953
954 pClient = (PCRVBOXHGSMI_CLIENT)pBuf->pvUserData;
955
956 /* Length would be passed as part of HGCM pointer description
957 * No need to prepend it to the buffer
958 */
959#ifdef IN_GUEST
960 if (conn->u32InjectClientID)
961 {
962 _crVBoxHGSMIWriteExact(conn, pClient, pBuf, CRVBOXHGSMI_BUF_OFFSET(start, *bufp) + CRVBOXHGSMI_BUF_HDR_SIZE(), len);
963 }
964 else
965#endif
966 {
967 _crVBoxHGSMIWriteReadExact(conn, pClient, pBuf, CRVBOXHGSMI_BUF_OFFSET(start, *bufp) + CRVBOXHGSMI_BUF_HDR_SIZE(), len, true);
968 }
969
970 /* Reclaim this pointer for reuse */
971 _crVBoxHGSMIBufFree(pClient, pBuf);
972 /* Since the buffer's now in the 'free' buffer pool, the caller can't
973 * use it any more. Setting bufp to NULL will make sure the caller
974 * doesn't try to re-use the buffer.
975 */
976 *bufp = NULL;
977
978 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
979}
980
981static void crVBoxHGSMISingleRecv(CRConnection *conn, void *buf, unsigned int len)
982{
983 PCRVBOXHGSMI_CLIENT pClient;
984 PVBOXUHGSMI_BUFFER pBuf;
985 VBOXCRHGSMIPROFILE_FUNC_PROLOGUE();
986 pBuf = _crVBoxHGSMIBufFromMemPtr(buf);
987 Assert(pBuf);
988 Assert(0);
989 CRASSERT(0);
990 if (!pBuf)
991 {
992 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
993 return;
994 }
995 pClient = (PCRVBOXHGSMI_CLIENT)pBuf->pvUserData;
996 _crVBoxHGSMIReadExact(conn, pClient);
997 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
998}
999
1000static void _crVBoxHGSMIFree(CRConnection *conn, void *buf)
1001{
1002 CRVBOXHGSMIBUFFER *hgsmi_buffer = (CRVBOXHGSMIBUFFER *) buf - 1;
1003
1004 CRASSERT(hgsmi_buffer->magic == CR_VBOXHGSMI_BUFFER_MAGIC);
1005
1006 if (hgsmi_buffer->bIsBuf)
1007 {
1008 PVBOXUHGSMI_BUFFER pBuf = hgsmi_buffer->pBuffer;
1009 PCRVBOXHGSMI_CLIENT pClient = (PCRVBOXHGSMI_CLIENT)pBuf->pvUserData;
1010 pBuf->pfnUnlock(pBuf);
1011 _crVBoxHGSMIBufFree(pClient, pBuf);
1012 }
1013 else
1014 {
1015 /*@todo wrong len for redir buffers*/
1016 conn->recv_credits += hgsmi_buffer->u32Len;
1017
1018#ifdef CHROMIUM_THREADSAFE
1019 crLockMutex(&g_crvboxhgsmi.mutex);
1020#endif
1021 crBufferPoolPush(g_crvboxhgsmi.mempool, hgsmi_buffer, hgsmi_buffer->cbLock);
1022#ifdef CHROMIUM_THREADSAFE
1023 crUnlockMutex(&g_crvboxhgsmi.mutex);
1024#endif
1025 }
1026}
1027static void crVBoxHGSMIFree(CRConnection *conn, void *buf)
1028{
1029 VBOXCRHGSMIPROFILE_FUNC_PROLOGUE();
1030 _crVBoxHGSMIFree(conn, buf);
1031 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
1032}
1033
1034static void _crVBoxHGSMIReceiveMessage(CRConnection *conn, PCRVBOXHGSMI_CLIENT pClient)
1035{
1036 uint32_t len;
1037 CRVBOXHGSMIBUFFER* pSysBuf;
1038 CRMessage *msg;
1039 CRMessageType cached_type;
1040
1041 len = conn->cbBuffer;
1042 Assert(len > 0);
1043 Assert(conn->pBuffer);
1044 CRASSERT(len > 0);
1045 CRASSERT(conn->pBuffer);
1046
1047#ifndef IN_GUEST
1048 if (conn->allow_redir_ptr)
1049 {
1050#endif //IN_GUEST
1051 CRASSERT(conn->buffer_size >= sizeof(CRMessageRedirPtr));
1052
1053 pSysBuf = _crVBoxHGSMISysMemAlloc( conn->buffer_size );
1054 pSysBuf->u32Len = sizeof(CRMessageRedirPtr);
1055
1056 msg = (CRMessage *)CRVBOXHGSMI_BUF_DATA(pSysBuf);
1057
1058 msg->header.type = CR_MESSAGE_REDIR_PTR;
1059 msg->redirptr.pMessage = (CRMessageHeader*) (conn->pBuffer);
1060 msg->header.conn_id = msg->redirptr.pMessage->conn_id;
1061
1062 cached_type = msg->redirptr.pMessage->type;
1063
1064 conn->cbBuffer = 0;
1065 conn->pBuffer = NULL;
1066#ifndef IN_GUEST
1067 }
1068 else
1069 {
1070 if ( len <= conn->buffer_size )
1071 {
1072 /* put in pre-allocated buffer */
1073 hgcm_buffer = (CRVBOXHGCMBUFFER *) crVBoxHGCMAlloc( conn ) - 1;
1074 }
1075 else
1076 {
1077 /* allocate new buffer,
1078 * not using pool here as it's most likely one time transfer of huge texture
1079 */
1080 hgcm_buffer = (CRVBOXHGCMBUFFER *) crAlloc( sizeof(CRVBOXHGCMBUFFER) + len );
1081 hgcm_buffer->magic = CR_VBOXHGCM_BUFFER_MAGIC;
1082 hgcm_buffer->kind = CR_VBOXHGCM_MEMORY_BIG;
1083 hgcm_buffer->allocated = sizeof(CRVBOXHGCMBUFFER) + len;
1084# ifdef RT_OS_WINDOWS
1085 hgcm_buffer->pDDS = NULL;
1086# endif
1087 }
1088
1089 hgcm_buffer->len = len;
1090
1091 _crVBoxHGCMReadBytes(conn, hgcm_buffer + 1, len);
1092
1093 msg = (CRMessage *) (hgcm_buffer + 1);
1094 cached_type = msg->header.type;
1095 }
1096#endif //IN_GUEST
1097
1098 conn->recv_credits -= len;
1099 conn->total_bytes_recv += len;
1100 conn->recv_count++;
1101
1102 crNetDispatchMessage( g_crvboxhgsmi.recv_list, conn, msg, len );
1103
1104 /* CR_MESSAGE_OPCODES is freed in crserverlib/server_stream.c with crNetFree.
1105 * OOB messages are the programmer's problem. -- Humper 12/17/01
1106 */
1107 if (cached_type != CR_MESSAGE_OPCODES
1108 && cached_type != CR_MESSAGE_OOB
1109 && cached_type != CR_MESSAGE_GATHER)
1110 {
1111 _crVBoxHGSMIFree(conn, msg);
1112 }
1113}
1114
1115static void crVBoxHGSMIReceiveMessage(CRConnection *conn)
1116{
1117 PCRVBOXHGSMI_CLIENT pClient;
1118 VBOXCRHGSMIPROFILE_FUNC_PROLOGUE();
1119
1120 pClient = _crVBoxHGSMIClientGet();
1121
1122 Assert(pClient);
1123 Assert(0);
1124
1125 _crVBoxHGSMIReceiveMessage(conn, pClient);
1126 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
1127}
1128/*
1129 * Called on host side only, to "accept" client connection
1130 */
1131static void crVBoxHGSMIAccept( CRConnection *conn, const char *hostname, unsigned short port )
1132{
1133 VBOXCRHGSMIPROFILE_FUNC_PROLOGUE();
1134 Assert(0);
1135
1136 CRASSERT(conn && conn->pHostBuffer);
1137#ifdef IN_GUEST
1138 CRASSERT(FALSE);
1139#endif
1140 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
1141}
1142
1143static int crVBoxHGSMISetVersion(CRConnection *conn, unsigned int vMajor, unsigned int vMinor)
1144{
1145 CRVBOXHGCMSETVERSION parms;
1146 int rc;
1147
1148 parms.hdr.result = VERR_WRONG_ORDER;
1149 parms.hdr.u32ClientID = conn->u32ClientID;
1150 parms.hdr.u32Function = SHCRGL_GUEST_FN_SET_VERSION;
1151 parms.hdr.cParms = SHCRGL_CPARMS_SET_VERSION;
1152
1153 parms.vMajor.type = VMMDevHGCMParmType_32bit;
1154 parms.vMajor.u.value32 = CR_PROTOCOL_VERSION_MAJOR;
1155 parms.vMinor.type = VMMDevHGCMParmType_32bit;
1156 parms.vMinor.u.value32 = CR_PROTOCOL_VERSION_MINOR;
1157
1158 rc = crVBoxHGCMCall(&parms, sizeof(parms));
1159
1160 AssertRC(rc);
1161
1162 if (RT_FAILURE(rc) || RT_FAILURE(parms.hdr.result))
1163 {
1164 crWarning("Host doesn't accept our version %d.%d. Make sure you have appropriate additions installed!",
1165 parms.vMajor.u.value32, parms.vMinor.u.value32);
1166 return FALSE;
1167 }
1168
1169 conn->vMajor = CR_PROTOCOL_VERSION_MAJOR;
1170 conn->vMinor = CR_PROTOCOL_VERSION_MINOR;
1171
1172 return TRUE;
1173}
1174
1175static int crVBoxHGSMISetPID(CRConnection *conn, unsigned long long pid)
1176{
1177 CRVBOXHGCMSETPID parms;
1178 int rc;
1179
1180 parms.hdr.result = VERR_WRONG_ORDER;
1181 parms.hdr.u32ClientID = conn->u32ClientID;
1182 parms.hdr.u32Function = SHCRGL_GUEST_FN_SET_PID;
1183 parms.hdr.cParms = SHCRGL_CPARMS_SET_PID;
1184
1185 parms.u64PID.type = VMMDevHGCMParmType_64bit;
1186 parms.u64PID.u.value64 = pid;
1187
1188 rc = crVBoxHGCMCall(&parms, sizeof(parms));
1189
1190 if (RT_FAILURE(rc) || RT_FAILURE(parms.hdr.result))
1191 {
1192 Assert(0);
1193
1194 crWarning("SHCRGL_GUEST_FN_SET_PID failed!");
1195 return FALSE;
1196 }
1197
1198 return TRUE;
1199}
1200
1201/**
1202 * The function that actually connects. This should only be called by clients,
1203 * guests in vbox case.
1204 * Servers go through crVBoxHGCMAccept;
1205 */
1206/*@todo use vbglR3Something here */
1207static int crVBoxHGSMIDoConnect( CRConnection *conn )
1208{
1209#ifdef IN_GUEST
1210 VBoxGuestHGCMConnectInfo info;
1211
1212#ifdef RT_OS_WINDOWS
1213 DWORD cbReturned;
1214 VBOXCRHGSMIPROFILE_FUNC_PROLOGUE();
1215
1216 if (g_crvboxhgsmi.hGuestDrv == INVALID_HANDLE_VALUE)
1217 {
1218 /* open VBox guest driver */
1219 g_crvboxhgsmi.hGuestDrv = CreateFile(VBOXGUEST_DEVICE_NAME,
1220 GENERIC_READ | GENERIC_WRITE,
1221 FILE_SHARE_READ | FILE_SHARE_WRITE,
1222 NULL,
1223 OPEN_EXISTING,
1224 FILE_ATTRIBUTE_NORMAL,
1225 NULL);
1226
1227 /* @todo check if we could rollback to softwareopengl */
1228 if (g_crvboxhgsmi.hGuestDrv == INVALID_HANDLE_VALUE)
1229 {
1230 Assert(0);
1231 crDebug("could not open VBox Guest Additions driver! rc = %d\n", GetLastError());
1232 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
1233 return FALSE;
1234 }
1235 }
1236#else
1237 VBOXCRHGSMIPROFILE_FUNC_PROLOGUE();
1238
1239 if (g_crvboxhgsmi.iGuestDrv == INVALID_HANDLE_VALUE)
1240 {
1241 g_crvboxhgsmi.iGuestDrv = open(VBOXGUEST_USER_DEVICE_NAME, O_RDWR, 0);
1242 if (g_crvboxhgsmi.iGuestDrv == INVALID_HANDLE_VALUE)
1243 {
1244 crDebug("could not open Guest Additions kernel module! rc = %d\n", errno);
1245 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
1246 return FALSE;
1247 }
1248 }
1249#endif
1250
1251 memset (&info, 0, sizeof (info));
1252 info.Loc.type = VMMDevHGCMLoc_LocalHost_Existing;
1253 strcpy (info.Loc.u.host.achName, "VBoxSharedCrOpenGL");
1254
1255#ifdef RT_OS_WINDOWS
1256 if (DeviceIoControl(g_crvboxhgsmi.hGuestDrv,
1257 VBOXGUEST_IOCTL_HGCM_CONNECT,
1258 &info, sizeof (info),
1259 &info, sizeof (info),
1260 &cbReturned,
1261 NULL))
1262#elif defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
1263 VBGLBIGREQ Hdr;
1264 Hdr.u32Magic = VBGLBIGREQ_MAGIC;
1265 Hdr.cbData = sizeof(info);
1266 Hdr.pvDataR3 = &info;
1267# if HC_ARCH_BITS == 32
1268 Hdr.u32Padding = 0;
1269# endif
1270 if (ioctl(g_crvboxhgsmi.iGuestDrv, VBOXGUEST_IOCTL_HGCM_CONNECT, &Hdr) >= 0)
1271#else
1272 if (ioctl(g_crvboxhgsmi.iGuestDrv, VBOXGUEST_IOCTL_HGCM_CONNECT, &info, sizeof (info)) >= 0)
1273#endif
1274 {
1275 if (info.result == VINF_SUCCESS)
1276 {
1277 int rc;
1278 conn->u32ClientID = info.u32ClientID;
1279 crDebug("HGCM connect was successful: client id =0x%x\n", conn->u32ClientID);
1280
1281 rc = crVBoxHGSMISetVersion(conn, CR_PROTOCOL_VERSION_MAJOR, CR_PROTOCOL_VERSION_MINOR);
1282 if (!rc)
1283 {
1284 return rc;
1285 }
1286#ifdef RT_OS_WINDOWS
1287 rc = crVBoxHGCMSetPID(conn, GetCurrentProcessId());
1288#else
1289 rc = crVBoxHGCMSetPID(conn, crGetPID());
1290#endif
1291 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
1292 return rc;
1293 }
1294 else
1295 {
1296 Assert(0);
1297 crDebug("HGCM connect failed with rc=0x%x\n", info.result);
1298 }
1299 }
1300 else
1301 {
1302#ifdef RT_OS_WINDOWS
1303 crDebug("IOCTL for HGCM connect failed with rc=0x%x\n", GetLastError());
1304#else
1305 crDebug("IOCTL for HGCM connect failed with rc=0x%x\n", errno);
1306#endif
1307 }
1308
1309 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
1310
1311 return FALSE;
1312#else /*#ifdef IN_GUEST*/
1313 crError("crVBoxHGSMIDoConnect called on host side!");
1314 CRASSERT(FALSE);
1315 return FALSE;
1316#endif
1317}
1318
1319/*@todo same, replace DeviceIoControl with vbglR3DoIOCtl */
1320static void crVBoxHGSMIDoDisconnect( CRConnection *conn )
1321{
1322#ifdef IN_GUEST
1323 VBoxGuestHGCMDisconnectInfo info;
1324# ifdef RT_OS_WINDOWS
1325 DWORD cbReturned;
1326# endif
1327 int i;
1328#endif
1329 VBOXCRHGSMIPROFILE_FUNC_PROLOGUE();
1330
1331 if (conn->pHostBuffer)
1332 {
1333 crFree(conn->pHostBuffer);
1334 conn->pHostBuffer = NULL;
1335 conn->cbHostBuffer = 0;
1336 conn->cbHostBufferAllocated = 0;
1337 }
1338
1339 conn->pBuffer = NULL;
1340 conn->cbBuffer = 0;
1341
1342 //@todo hold lock here?
1343 if (conn->type == CR_VBOXHGCM)
1344 {
1345 --g_crvboxhgsmi.num_conns;
1346
1347 if (conn->index < g_crvboxhgsmi.num_conns)
1348 {
1349 g_crvboxhgsmi.conns[conn->index] = g_crvboxhgsmi.conns[g_crvboxhgsmi.num_conns];
1350 g_crvboxhgsmi.conns[conn->index]->index = conn->index;
1351 }
1352 else g_crvboxhgsmi.conns[conn->index] = NULL;
1353
1354 conn->type = CR_NO_CONNECTION;
1355 }
1356
1357#ifndef IN_GUEST
1358#else /* IN_GUEST */
1359 if (conn->u32ClientID)
1360 {
1361 memset (&info, 0, sizeof (info));
1362 info.u32ClientID = conn->u32ClientID;
1363
1364# ifdef RT_OS_WINDOWS
1365 if ( !DeviceIoControl(g_crvboxhgsmi.hGuestDrv,
1366 VBOXGUEST_IOCTL_HGCM_DISCONNECT,
1367 &info, sizeof (info),
1368 &info, sizeof (info),
1369 &cbReturned,
1370 NULL) )
1371 {
1372 crDebug("Disconnect failed with %x\n", GetLastError());
1373 }
1374# elif defined(RT_OS_SOLARIS) || defined(RT_OS_FREEBSD)
1375 VBGLBIGREQ Hdr;
1376 Hdr.u32Magic = VBGLBIGREQ_MAGIC;
1377 Hdr.cbData = sizeof(info);
1378 Hdr.pvDataR3 = &info;
1379# if HC_ARCH_BITS == 32
1380 Hdr.u32Padding = 0;
1381# endif
1382 if (ioctl(g_crvboxhgsmi.iGuestDrv, VBOXGUEST_IOCTL_HGCM_DISCONNECT, &Hdr) >= 0)
1383# else
1384 if (ioctl(g_crvboxhgsmi.iGuestDrv, VBOXGUEST_IOCTL_HGCM_DISCONNECT, &info, sizeof (info)) < 0)
1385 {
1386 crDebug("Disconnect failed with %x\n", errno);
1387 }
1388# endif
1389
1390 conn->u32ClientID = 0;
1391 }
1392
1393 /* see if any connections remain */
1394 for (i = 0; i < g_crvboxhgsmi.num_conns; i++)
1395 if (g_crvboxhgsmi.conns[i] && g_crvboxhgsmi.conns[i]->type != CR_NO_CONNECTION)
1396 break;
1397
1398 /* close guest additions driver*/
1399 if (i>=g_crvboxhgsmi.num_conns)
1400 {
1401# ifdef RT_OS_WINDOWS
1402 CloseHandle(g_crvboxhgsmi.hGuestDrv);
1403 g_crvboxhgsmi.hGuestDrv = INVALID_HANDLE_VALUE;
1404# else
1405 close(g_crvboxhgsmi.iGuestDrv);
1406 g_crvboxhgsmi.iGuestDrv = INVALID_HANDLE_VALUE;
1407# endif
1408 }
1409 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
1410#endif /* IN_GUEST */
1411}
1412
1413static void crVBoxHGSMIInstantReclaim(CRConnection *conn, CRMessage *mess)
1414{
1415 VBOXCRHGSMIPROFILE_FUNC_PROLOGUE();
1416 Assert(0);
1417
1418 _crVBoxHGSMIFree(conn, mess);
1419 CRASSERT(FALSE);
1420
1421 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
1422}
1423
1424static void crVBoxHGSMIHandleNewMessage( CRConnection *conn, CRMessage *msg, unsigned int len )
1425{
1426 VBOXCRHGSMIPROFILE_FUNC_PROLOGUE();
1427 Assert(0);
1428
1429 CRASSERT(FALSE);
1430 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
1431}
1432
1433static DECLCALLBACK(HVBOXCRHGSMI_CLIENT) _crVBoxHGSMIClientCreate(PVBOXUHGSMI pHgsmi)
1434{
1435 PCRVBOXHGSMI_CLIENT pClient = crAlloc(sizeof (CRVBOXHGSMI_CLIENT));
1436
1437 if (pClient)
1438 {
1439 int rc;
1440 pClient->pHgsmi = pHgsmi;
1441 rc = pHgsmi->pfnBufferCreate(pHgsmi, CRVBOXHGSMI_PAGE_ALIGN(1),
1442 VBOXUHGSMI_SYNCHOBJECT_TYPE_EVENT,
1443 NULL,
1444 &pClient->pCmdBuffer);
1445 AssertRC(rc);
1446 if (RT_SUCCESS(rc))
1447 {
1448 rc = pHgsmi->pfnBufferCreate(pHgsmi, CRVBOXHGSMI_PAGE_ALIGN(0x800000),
1449 VBOXUHGSMI_SYNCHOBJECT_TYPE_EVENT,
1450 NULL,
1451 &pClient->pHGBuffer);
1452 AssertRC(rc);
1453 if (RT_SUCCESS(rc))
1454 {
1455 pClient->pvHGBuffer = NULL;
1456 pClient->bufpool = crBufferPoolInit(16);
1457 return (HVBOXCRHGSMI_CLIENT) pClient;
1458 }
1459 }
1460 }
1461
1462 return NULL;
1463}
1464
1465static DECLCALLBACK(void) _crVBoxHGSMIClientDestroy(HVBOXCRHGSMI_CLIENT hClient)
1466{
1467 Assert(0);
1468
1469 /* @todo */
1470}
1471
1472
1473bool crVBoxHGSMIInit(CRNetReceiveFuncList *rfl, CRNetCloseFuncList *cfl, unsigned int mtu)
1474{
1475 /* static */ int bHasHGSMI = -1; /* do it for all connections */
1476 (void) mtu;
1477
1478 if (bHasHGSMI < 0)
1479 {
1480 int rc;
1481 VBOXCRHGSMI_CALLBACKS Callbacks;
1482 Callbacks.pfnClientCreate = _crVBoxHGSMIClientCreate;
1483 Callbacks.pfnClientDestroy = _crVBoxHGSMIClientDestroy;
1484 rc = VBoxCrHgsmiInit(&Callbacks);
1485 AssertRC(rc);
1486 if (RT_SUCCESS(rc))
1487 bHasHGSMI = 1;
1488 else
1489 bHasHGSMI = 0;
1490 }
1491
1492 Assert(bHasHGSMI);
1493
1494 if (!bHasHGSMI)
1495 {
1496#ifdef DEBUG_misha
1497 AssertRelease(0);
1498#endif
1499 return false;
1500 }
1501
1502 g_crvboxhgsmi.recv_list = rfl;
1503 g_crvboxhgsmi.close_list = cfl;
1504 if (g_crvboxhgsmi.initialized)
1505 {
1506 return true;
1507 }
1508
1509 g_crvboxhgsmi.initialized = 1;
1510
1511 g_crvboxhgsmi.num_conns = 0;
1512 g_crvboxhgsmi.conns = NULL;
1513 g_crvboxhgsmi.mempool = crBufferPoolInit(16);
1514
1515 /* Can't open VBox guest driver here, because it gets called for host side as well */
1516 /*@todo as we have 2 dll versions, can do it now.*/
1517
1518#ifdef RT_OS_WINDOWS
1519 g_crvboxhgsmi.hGuestDrv = INVALID_HANDLE_VALUE;
1520#else
1521 g_crvboxhgsmi.iGuestDrv = INVALID_HANDLE_VALUE;
1522#endif
1523
1524#ifdef CHROMIUM_THREADSAFE
1525 crInitMutex(&g_crvboxhgsmi.mutex);
1526 crInitMutex(&g_crvboxhgsmi.recvmutex);
1527#endif
1528
1529 return true;
1530}
1531
1532/* Callback function used to free buffer pool entries */
1533void _crVBoxHGSMISysMemFree(void *data)
1534{
1535 Assert(0);
1536
1537 crFree(data);
1538}
1539
1540void crVBoxHGSMITearDown(void)
1541{
1542 int32_t i, cCons;
1543
1544 Assert(0);
1545
1546 if (!g_crvboxhgsmi.initialized) return;
1547
1548 /* Connection count would be changed in calls to crNetDisconnect, so we have to store original value.
1549 * Walking array backwards is not a good idea as it could cause some issues if we'd disconnect clients not in the
1550 * order of their connection.
1551 */
1552 cCons = g_crvboxhgsmi.num_conns;
1553 for (i=0; i<cCons; i++)
1554 {
1555 /* Note that [0] is intended, as the connections array would be shifted in each call to crNetDisconnect */
1556 crNetDisconnect(g_crvboxhgsmi.conns[0]);
1557 }
1558 CRASSERT(0==g_crvboxhgsmi.num_conns);
1559
1560#ifdef CHROMIUM_THREADSAFE
1561 crFreeMutex(&g_crvboxhgsmi.mutex);
1562 crFreeMutex(&g_crvboxhgsmi.recvmutex);
1563#endif
1564
1565 if (g_crvboxhgsmi.mempool)
1566 crBufferPoolCallbackFree(g_crvboxhgsmi.mempool, _crVBoxHGSMISysMemFree);
1567 g_crvboxhgsmi.mempool = NULL;
1568
1569 g_crvboxhgsmi.initialized = 0;
1570
1571 crFree(g_crvboxhgsmi.conns);
1572 g_crvboxhgsmi.conns = NULL;
1573}
1574
1575void crVBoxHGSMIConnection(CRConnection *conn)
1576{
1577 int i, found = 0;
1578 int n_bytes;
1579
1580 CRASSERT(g_crvboxhgsmi.initialized);
1581
1582 conn->type = CR_VBOXHGCM;
1583 conn->Alloc = crVBoxHGSMIAlloc;
1584 conn->Send = crVBoxHGSMISend;
1585 conn->SendExact = crVBoxHGSMIWriteExact;
1586 conn->Recv = crVBoxHGSMISingleRecv;
1587 conn->RecvMsg = crVBoxHGSMIReceiveMessage;
1588 conn->Free = crVBoxHGSMIFree;
1589 conn->Accept = crVBoxHGSMIAccept;
1590 conn->Connect = crVBoxHGSMIDoConnect;
1591 conn->Disconnect = crVBoxHGSMIDoDisconnect;
1592 conn->InstantReclaim = crVBoxHGSMIInstantReclaim;
1593 conn->HandleNewMessage = crVBoxHGSMIHandleNewMessage;
1594 conn->index = g_crvboxhgsmi.num_conns;
1595 conn->sizeof_buffer_header = sizeof(CRVBOXHGSMIBUFFER);
1596 conn->actual_network = 1;
1597
1598 conn->krecv_buf_size = 0;
1599
1600 conn->pBuffer = NULL;
1601 conn->cbBuffer = 0;
1602 conn->allow_redir_ptr = 1;
1603
1604 //@todo remove this crap at all later
1605 conn->cbHostBufferAllocated = 0;//2*1024;
1606 conn->pHostBuffer = NULL;//(uint8_t*) crAlloc(conn->cbHostBufferAllocated);
1607// CRASSERT(conn->pHostBuffer);
1608 conn->cbHostBuffer = 0;
1609
1610 /* Find a free slot */
1611 for (i = 0; i < g_crvboxhgsmi.num_conns; i++) {
1612 if (g_crvboxhgsmi.conns[i] == NULL) {
1613 conn->index = i;
1614 g_crvboxhgsmi.conns[i] = conn;
1615 found = 1;
1616 break;
1617 }
1618 }
1619
1620 /* Realloc connection stack if we couldn't find a free slot */
1621 if (found == 0) {
1622 n_bytes = ( g_crvboxhgsmi.num_conns + 1 ) * sizeof(*g_crvboxhgsmi.conns);
1623 crRealloc( (void **) &g_crvboxhgsmi.conns, n_bytes );
1624 g_crvboxhgsmi.conns[g_crvboxhgsmi.num_conns++] = conn;
1625 }
1626}
1627
1628int crVBoxHGSMIRecv(void)
1629{
1630 int32_t i;
1631 VBOXCRHGSMIPROFILE_FUNC_PROLOGUE();
1632
1633#ifdef IN_GUEST
1634 /* we're on guest side, poll host if it got something for us */
1635 for (i=0; i<g_crvboxhgsmi.num_conns; i++)
1636 {
1637 CRConnection *conn = g_crvboxhgsmi.conns[i];
1638
1639 if ( !conn || conn->type == CR_NO_CONNECTION )
1640 continue;
1641
1642 if (!conn->pBuffer)
1643 {
1644 PCRVBOXHGSMI_CLIENT pClient = _crVBoxHGSMIClientGet();
1645 _crVBoxHGSMIPollHost(conn, pClient);
1646 }
1647 }
1648#endif
1649
1650 for (i=0; i<g_crvboxhgsmi.num_conns; i++)
1651 {
1652 CRConnection *conn = g_crvboxhgsmi.conns[i];
1653
1654 if ( !conn || conn->type == CR_NO_CONNECTION )
1655 continue;
1656
1657 if (conn->cbBuffer>0)
1658 {
1659 PCRVBOXHGSMI_CLIENT pClient = _crVBoxHGSMIClientGet();
1660 _crVBoxHGSMIReceiveMessage(conn, pClient);
1661 }
1662 }
1663
1664 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
1665 return 0;
1666}
1667
1668CRConnection** crVBoxHGSMIDump( int *num )
1669{
1670 VBOXCRHGSMIPROFILE_FUNC_PROLOGUE();
1671 Assert(0);
1672 *num = g_crvboxhgsmi.num_conns;
1673
1674 VBOXCRHGSMIPROFILE_FUNC_EPILOGUE();
1675 return g_crvboxhgsmi.conns;
1676}
1677#endif /* #ifdef IN_GUEST */
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