VirtualBox

source: vbox/trunk/src/VBox/Devices/Graphics/DevVGA_VDMA.cpp@ 42392

Last change on this file since 42392 was 41974, checked in by vboxsync, 13 years ago

DevVGA: Don't assert if the HGSMI driver is not attached.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 42.3 KB
Line 
1/** @file
2 * Video DMA (VDMA) support.
3 */
4
5/*
6 * Copyright (C) 2006-2009 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//#include <VBox/VMMDev.h>
17#include <VBox/vmm/pdmdev.h>
18#include <VBox/VBoxVideo.h>
19#include <iprt/semaphore.h>
20#include <iprt/thread.h>
21#include <iprt/mem.h>
22#include <iprt/asm.h>
23
24#include "DevVGA.h"
25#include "HGSMI/SHGSMIHost.h"
26#include "HGSMI/HGSMIHostHlp.h"
27
28#ifdef VBOX_VDMA_WITH_WORKERTHREAD
29typedef enum
30{
31 VBOXVDMAPIPE_STATE_CLOSED = 0,
32 VBOXVDMAPIPE_STATE_CREATED = 1,
33 VBOXVDMAPIPE_STATE_OPENNED = 2,
34 VBOXVDMAPIPE_STATE_CLOSING = 3
35} VBOXVDMAPIPE_STATE;
36
37typedef struct VBOXVDMAPIPE
38{
39 RTSEMEVENT hEvent;
40 /* critical section for accessing pipe properties */
41 RTCRITSECT hCritSect;
42 VBOXVDMAPIPE_STATE enmState;
43 /* true iff the other end needs Event notification */
44 bool bNeedNotify;
45} VBOXVDMAPIPE, *PVBOXVDMAPIPE;
46
47typedef enum
48{
49 VBOXVDMAPIPE_CMD_TYPE_UNDEFINED = 0,
50 VBOXVDMAPIPE_CMD_TYPE_DMACMD = 1,
51 VBOXVDMAPIPE_CMD_TYPE_DMACTL = 2
52} VBOXVDMAPIPE_CMD_TYPE;
53
54typedef struct VBOXVDMAPIPE_CMD_BODY
55{
56 VBOXVDMAPIPE_CMD_TYPE enmType;
57 union
58 {
59 PVBOXVDMACBUF_DR pDr;
60 PVBOXVDMA_CTL pCtl;
61 void *pvCmd;
62 } u;
63}VBOXVDMAPIPE_CMD_BODY, *PVBOXVDMAPIPE_CMD_BODY;
64
65typedef struct VBOXVDMAPIPE_CMD
66{
67 HGSMILISTENTRY Entry;
68 VBOXVDMAPIPE_CMD_BODY Cmd;
69} VBOXVDMAPIPE_CMD, *PVBOXVDMAPIPE_CMD;
70
71#define VBOXVDMAPIPE_CMD_FROM_ENTRY(_pE) ( (PVBOXVDMAPIPE_CMD)((uint8_t *)(_pE) - RT_OFFSETOF(VBOXVDMAPIPE_CMD, Entry)) )
72
73typedef struct VBOXVDMAPIPE_CMD_POOL
74{
75 HGSMILIST List;
76 uint32_t cCmds;
77 VBOXVDMAPIPE_CMD aCmds[1];
78} VBOXVDMAPIPE_CMD_POOL, *PVBOXVDMAPIPE_CMD_POOL;
79#endif
80
81typedef struct VBOXVDMAHOST
82{
83 PHGSMIINSTANCE pHgsmi;
84 PVGASTATE pVGAState;
85#ifdef VBOX_VDMA_WITH_WATCHDOG
86 PTMTIMERR3 WatchDogTimer;
87#endif
88#ifdef VBOX_VDMA_WITH_WORKERTHREAD
89 VBOXVDMAPIPE Pipe;
90 HGSMILIST PendingList;
91 RTTHREAD hWorkerThread;
92 VBOXVDMAPIPE_CMD_POOL CmdPool;
93#endif
94} VBOXVDMAHOST, *PVBOXVDMAHOST;
95
96
97#ifdef VBOX_WITH_CRHGSMI
98
99typedef DECLCALLBACK(void) FNVBOXVDMACRCTL_CALLBACK(PVGASTATE pVGAState, PVBOXVDMACMD_CHROMIUM_CTL pCmd, void* pvContext);
100typedef FNVBOXVDMACRCTL_CALLBACK *PFNVBOXVDMACRCTL_CALLBACK;
101
102typedef struct VBOXVDMACMD_CHROMIUM_CTL_PRIVATE
103{
104 uint32_t cRefs;
105 int32_t rc;
106 PFNVBOXVDMACRCTL_CALLBACK pfnCompletion;
107 void *pvCompletion;
108 VBOXVDMACMD_CHROMIUM_CTL Cmd;
109} VBOXVDMACMD_CHROMIUM_CTL_PRIVATE, *PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE;
110
111#define VBOXVDMACMD_CHROMIUM_CTL_PRIVATE_FROM_CTL(_p) ((PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE)(((uint8_t*)(_p)) - RT_OFFSETOF(VBOXVDMACMD_CHROMIUM_CTL_PRIVATE, Cmd)))
112
113static PVBOXVDMACMD_CHROMIUM_CTL vboxVDMACrCtlCreate(VBOXVDMACMD_CHROMIUM_CTL_TYPE enmCmd, uint32_t cbCmd)
114{
115 PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE pHdr = (PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE)RTMemAllocZ(cbCmd + RT_OFFSETOF(VBOXVDMACMD_CHROMIUM_CTL_PRIVATE, Cmd));
116 Assert(pHdr);
117 if (pHdr)
118 {
119 pHdr->cRefs = 1;
120 pHdr->rc = VERR_NOT_IMPLEMENTED;
121 pHdr->Cmd.enmType = enmCmd;
122 pHdr->Cmd.cbCmd = cbCmd;
123 return &pHdr->Cmd;
124 }
125
126 return NULL;
127}
128
129DECLINLINE(void) vboxVDMACrCtlRelease (PVBOXVDMACMD_CHROMIUM_CTL pCmd)
130{
131 PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE pHdr = VBOXVDMACMD_CHROMIUM_CTL_PRIVATE_FROM_CTL(pCmd);
132 uint32_t cRefs = ASMAtomicDecU32(&pHdr->cRefs);
133 if(!cRefs)
134 {
135 RTMemFree(pHdr);
136 }
137}
138
139DECLINLINE(void) vboxVDMACrCtlRetain (PVBOXVDMACMD_CHROMIUM_CTL pCmd)
140{
141 PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE pHdr = VBOXVDMACMD_CHROMIUM_CTL_PRIVATE_FROM_CTL(pCmd);
142 ASMAtomicIncU32(&pHdr->cRefs);
143}
144
145DECLINLINE(int) vboxVDMACrCtlGetRc (PVBOXVDMACMD_CHROMIUM_CTL pCmd)
146{
147 PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE pHdr = VBOXVDMACMD_CHROMIUM_CTL_PRIVATE_FROM_CTL(pCmd);
148 return pHdr->rc;
149}
150
151static DECLCALLBACK(void) vboxVDMACrCtlCbSetEvent(PVGASTATE pVGAState, PVBOXVDMACMD_CHROMIUM_CTL pCmd, void* pvContext)
152{
153 RTSemEventSignal((RTSEMEVENT)pvContext);
154}
155
156static DECLCALLBACK(void) vboxVDMACrCtlCbReleaseCmd(PVGASTATE pVGAState, PVBOXVDMACMD_CHROMIUM_CTL pCmd, void* pvContext)
157{
158 vboxVDMACrCtlRelease(pCmd);
159}
160
161
162static int vboxVDMACrCtlPostAsync (PVGASTATE pVGAState, PVBOXVDMACMD_CHROMIUM_CTL pCmd, uint32_t cbCmd, PFNVBOXVDMACRCTL_CALLBACK pfnCompletion, void *pvCompletion)
163{
164 if ( pVGAState->pDrv
165 && pVGAState->pDrv->pfnCrHgsmiControlProcess)
166 {
167 PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE pHdr = VBOXVDMACMD_CHROMIUM_CTL_PRIVATE_FROM_CTL(pCmd);
168 pHdr->pfnCompletion = pfnCompletion;
169 pHdr->pvCompletion = pvCompletion;
170 pVGAState->pDrv->pfnCrHgsmiControlProcess(pVGAState->pDrv, pCmd, cbCmd);
171 return VINF_SUCCESS;
172 }
173#ifdef DEBUG_misha
174 Assert(0);
175#endif
176 return VERR_NOT_SUPPORTED;
177}
178
179static int vboxVDMACrCtlPost(PVGASTATE pVGAState, PVBOXVDMACMD_CHROMIUM_CTL pCmd, uint32_t cbCmd)
180{
181 RTSEMEVENT hComplEvent;
182 int rc = RTSemEventCreate(&hComplEvent);
183 AssertRC(rc);
184 if(RT_SUCCESS(rc))
185 {
186 rc = vboxVDMACrCtlPostAsync (pVGAState, pCmd, cbCmd, vboxVDMACrCtlCbSetEvent, (void*)hComplEvent);
187#ifdef DEBUG_misha
188 AssertRC(rc);
189#endif
190 if (RT_SUCCESS(rc))
191 {
192 rc = RTSemEventWaitNoResume(hComplEvent, RT_INDEFINITE_WAIT);
193 AssertRC(rc);
194 if(RT_SUCCESS(rc))
195 {
196 RTSemEventDestroy(hComplEvent);
197 }
198 }
199 else
200 {
201 /* the command is completed */
202 RTSemEventDestroy(hComplEvent);
203 }
204 }
205 return rc;
206}
207
208static int vboxVDMACrCtlHgsmiSetup(struct VBOXVDMAHOST *pVdma)
209{
210 PVBOXVDMACMD_CHROMIUM_CTL_CRHGSMI_SETUP pCmd;
211 pCmd = (PVBOXVDMACMD_CHROMIUM_CTL_CRHGSMI_SETUP) vboxVDMACrCtlCreate (VBOXVDMACMD_CHROMIUM_CTL_TYPE_CRHGSMI_SETUP,
212 sizeof (*pCmd));
213 if (pCmd)
214 {
215 PVGASTATE pVGAState = pVdma->pVGAState;
216 pCmd->pvVRamBase = pVGAState->vram_ptrR3;
217 pCmd->cbVRam = pVGAState->vram_size;
218 int rc = vboxVDMACrCtlPost(pVGAState, &pCmd->Hdr, sizeof (*pCmd));
219 Assert(RT_SUCCESS(rc) || rc == VERR_NOT_SUPPORTED);
220 if (RT_SUCCESS(rc))
221 {
222 rc = vboxVDMACrCtlGetRc(&pCmd->Hdr);
223 }
224 vboxVDMACrCtlRelease(&pCmd->Hdr);
225 return rc;
226 }
227 return VERR_NO_MEMORY;
228}
229
230static int vboxVDMACmdExecBpbTransfer(PVBOXVDMAHOST pVdma, const PVBOXVDMACMD_DMA_BPB_TRANSFER pTransfer, uint32_t cbBuffer);
231
232/* check if this is external cmd to be passed to chromium backend */
233static int vboxVDMACmdCheckCrCmd(struct VBOXVDMAHOST *pVdma, PVBOXVDMACBUF_DR pCmdDr, uint32_t cbCmdDr)
234{
235 PVBOXVDMACMD pDmaCmd = NULL;
236 uint32_t cbDmaCmd = 0;
237 uint8_t * pvRam = pVdma->pVGAState->vram_ptrR3;
238 int rc = VINF_NOT_SUPPORTED;
239
240 if (pCmdDr->fFlags & VBOXVDMACBUF_FLAG_BUF_FOLLOWS_DR)
241 {
242 if (cbCmdDr < sizeof (*pCmdDr) + VBOXVDMACMD_HEADER_SIZE())
243 {
244 AssertMsgFailed(("invalid buffer data!"));
245 return VERR_INVALID_PARAMETER;
246 }
247
248 cbDmaCmd = pCmdDr->cbBuf;
249 if (cbDmaCmd < cbCmdDr - sizeof (*pCmdDr) - VBOXVDMACMD_HEADER_SIZE())
250 {
251 AssertMsgFailed(("invalid command buffer data!"));
252 return VERR_INVALID_PARAMETER;
253 }
254
255 pDmaCmd = VBOXVDMACBUF_DR_TAIL(pCmdDr, VBOXVDMACMD);
256 }
257
258 if (pDmaCmd)
259 {
260 Assert(cbDmaCmd >= VBOXVDMACMD_HEADER_SIZE());
261 uint32_t cbBody = VBOXVDMACMD_BODY_SIZE(cbDmaCmd);
262
263 switch (pDmaCmd->enmType)
264 {
265 case VBOXVDMACMD_TYPE_CHROMIUM_CMD:
266 {
267 PVBOXVDMACMD_CHROMIUM_CMD pCrCmd = VBOXVDMACMD_BODY(pDmaCmd, VBOXVDMACMD_CHROMIUM_CMD);
268 if (cbBody < sizeof (*pCrCmd))
269 {
270 AssertMsgFailed(("invalid chromium command buffer size!"));
271 return VERR_INVALID_PARAMETER;
272 }
273 PVGASTATE pVGAState = pVdma->pVGAState;
274 rc = VINF_SUCCESS;
275 if (pVGAState->pDrv->pfnCrHgsmiCommandProcess)
276 {
277 VBoxSHGSMICommandMarkAsynchCompletion(pCmdDr);
278 pVGAState->pDrv->pfnCrHgsmiCommandProcess(pVGAState->pDrv, pCrCmd, cbBody);
279 break;
280 }
281 else
282 {
283 Assert(0);
284 }
285
286 int tmpRc = VBoxSHGSMICommandComplete (pVdma->pHgsmi, pCmdDr);
287 AssertRC(tmpRc);
288 break;
289 }
290 case VBOXVDMACMD_TYPE_DMA_BPB_TRANSFER:
291 {
292 PVBOXVDMACMD_DMA_BPB_TRANSFER pTransfer = VBOXVDMACMD_BODY(pDmaCmd, VBOXVDMACMD_DMA_BPB_TRANSFER);
293 if (cbBody < sizeof (*pTransfer))
294 {
295 AssertMsgFailed(("invalid bpb transfer buffer size!"));
296 return VERR_INVALID_PARAMETER;
297 }
298
299 rc = vboxVDMACmdExecBpbTransfer(pVdma, pTransfer, sizeof (*pTransfer));
300 AssertRC(rc);
301 if (RT_SUCCESS(rc))
302 {
303 pCmdDr->rc = VINF_SUCCESS;
304 rc = VBoxSHGSMICommandComplete (pVdma->pHgsmi, pCmdDr);
305 AssertRC(rc);
306 rc = VINF_SUCCESS;
307 }
308 break;
309 }
310 default:
311 break;
312 }
313 }
314 return rc;
315}
316
317int vboxVDMACrHgsmiCommandCompleteAsync(PPDMIDISPLAYVBVACALLBACKS pInterface, PVBOXVDMACMD_CHROMIUM_CMD pCmd, int rc)
318{
319 PVGASTATE pVGAState = PPDMIDISPLAYVBVACALLBACKS_2_PVGASTATE(pInterface);
320 PHGSMIINSTANCE pIns = pVGAState->pHGSMI;
321 VBOXVDMACMD *pDmaHdr = VBOXVDMACMD_FROM_BODY(pCmd);
322 VBOXVDMACBUF_DR *pDr = VBOXVDMACBUF_DR_FROM_TAIL(pDmaHdr);
323 AssertRC(rc);
324 pDr->rc = rc;
325
326 Assert(pVGAState->fGuestCaps & VBVACAPS_COMPLETEGCMD_BY_IOREAD);
327 rc = VBoxSHGSMICommandComplete(pIns, pDr);
328 AssertRC(rc);
329 return rc;
330}
331
332int vboxVDMACrHgsmiControlCompleteAsync(PPDMIDISPLAYVBVACALLBACKS pInterface, PVBOXVDMACMD_CHROMIUM_CTL pCmd, int rc)
333{
334 PVGASTATE pVGAState = PPDMIDISPLAYVBVACALLBACKS_2_PVGASTATE(pInterface);
335 PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE pCmdPrivate = VBOXVDMACMD_CHROMIUM_CTL_PRIVATE_FROM_CTL(pCmd);
336 pCmdPrivate->rc = rc;
337 if (pCmdPrivate->pfnCompletion)
338 {
339 pCmdPrivate->pfnCompletion(pVGAState, pCmd, pCmdPrivate->pvCompletion);
340 }
341 return VINF_SUCCESS;
342}
343
344#endif
345
346#ifdef VBOX_VDMA_WITH_WORKERTHREAD
347/* to simplify things and to avoid extra backend if modifications we assume the VBOXVDMA_RECTL is the same as VBVACMDHDR */
348AssertCompile(sizeof(VBOXVDMA_RECTL) == sizeof(VBVACMDHDR));
349AssertCompile(RT_SIZEOFMEMB(VBOXVDMA_RECTL, left) == RT_SIZEOFMEMB(VBVACMDHDR, x));
350AssertCompile(RT_SIZEOFMEMB(VBOXVDMA_RECTL, top) == RT_SIZEOFMEMB(VBVACMDHDR, y));
351AssertCompile(RT_SIZEOFMEMB(VBOXVDMA_RECTL, width) == RT_SIZEOFMEMB(VBVACMDHDR, w));
352AssertCompile(RT_SIZEOFMEMB(VBOXVDMA_RECTL, height) == RT_SIZEOFMEMB(VBVACMDHDR, h));
353AssertCompile(RT_OFFSETOF(VBOXVDMA_RECTL, left) == RT_OFFSETOF(VBVACMDHDR, x));
354AssertCompile(RT_OFFSETOF(VBOXVDMA_RECTL, top) == RT_OFFSETOF(VBVACMDHDR, y));
355AssertCompile(RT_OFFSETOF(VBOXVDMA_RECTL, width) == RT_OFFSETOF(VBVACMDHDR, w));
356AssertCompile(RT_OFFSETOF(VBOXVDMA_RECTL, height) == RT_OFFSETOF(VBVACMDHDR, h));
357
358static int vboxVDMANotifyPrimaryUpdate (PVGASTATE pVGAState, unsigned uScreenId, const VBOXVDMA_RECTL * pRectl)
359{
360 pVGAState->pDrv->pfnVBVAUpdateBegin (pVGAState->pDrv, uScreenId);
361
362 /* Updates the rectangle and sends the command to the VRDP server. */
363 pVGAState->pDrv->pfnVBVAUpdateProcess (pVGAState->pDrv, uScreenId,
364 (const PVBVACMDHDR)pRectl /* <- see above AssertCompile's and comments */,
365 sizeof (VBOXVDMA_RECTL));
366
367 pVGAState->pDrv->pfnVBVAUpdateEnd (pVGAState->pDrv, uScreenId, pRectl->left, pRectl->top,
368 pRectl->width, pRectl->height);
369
370 return VINF_SUCCESS;
371}
372#endif
373
374static int vboxVDMACmdExecBltPerform(PVBOXVDMAHOST pVdma,
375 uint8_t *pvDstSurf, const uint8_t *pvSrcSurf,
376 const PVBOXVDMA_SURF_DESC pDstDesc, const PVBOXVDMA_SURF_DESC pSrcDesc,
377 const VBOXVDMA_RECTL * pDstRectl, const VBOXVDMA_RECTL * pSrcRectl)
378{
379 /* we do not support color conversion */
380 Assert(pDstDesc->format == pSrcDesc->format);
381 /* we do not support stretching */
382 Assert(pDstRectl->height == pSrcRectl->height);
383 Assert(pDstRectl->width == pSrcRectl->width);
384 if (pDstDesc->format != pSrcDesc->format)
385 return VERR_INVALID_FUNCTION;
386 if (pDstDesc->width == pDstRectl->width
387 && pSrcDesc->width == pSrcRectl->width
388 && pSrcDesc->width == pDstDesc->width)
389 {
390 Assert(!pDstRectl->left);
391 Assert(!pSrcRectl->left);
392 uint32_t cbOff = pDstDesc->pitch * pDstRectl->top;
393 uint32_t cbSize = pDstDesc->pitch * pDstRectl->height;
394 memcpy(pvDstSurf + cbOff, pvSrcSurf + cbOff, cbSize);
395 }
396 else
397 {
398 uint32_t offDstLineStart = pDstRectl->left * pDstDesc->bpp >> 3;
399 uint32_t offDstLineEnd = ((pDstRectl->left * pDstDesc->bpp + 7) >> 3) + ((pDstDesc->bpp * pDstRectl->width + 7) >> 3);
400 uint32_t cbDstLine = offDstLineEnd - offDstLineStart;
401 uint32_t offDstStart = pDstDesc->pitch * pDstRectl->top + offDstLineStart;
402 Assert(cbDstLine <= pDstDesc->pitch);
403 uint32_t cbDstSkip = pDstDesc->pitch;
404 uint8_t * pvDstStart = pvDstSurf + offDstStart;
405
406 uint32_t offSrcLineStart = pSrcRectl->left * pSrcDesc->bpp >> 3;
407 uint32_t offSrcLineEnd = ((pSrcRectl->left * pSrcDesc->bpp + 7) >> 3) + ((pSrcDesc->bpp * pSrcRectl->width + 7) >> 3);
408 uint32_t cbSrcLine = offSrcLineEnd - offSrcLineStart;
409 uint32_t offSrcStart = pSrcDesc->pitch * pSrcRectl->top + offSrcLineStart;
410 Assert(cbSrcLine <= pSrcDesc->pitch);
411 uint32_t cbSrcSkip = pSrcDesc->pitch;
412 const uint8_t * pvSrcStart = pvSrcSurf + offSrcStart;
413
414 Assert(cbDstLine == cbSrcLine);
415
416 for (uint32_t i = 0; ; ++i)
417 {
418 memcpy (pvDstStart, pvSrcStart, cbDstLine);
419 if (i == pDstRectl->height)
420 break;
421 pvDstStart += cbDstSkip;
422 pvSrcStart += cbSrcSkip;
423 }
424 }
425 return VINF_SUCCESS;
426}
427
428static void vboxVDMARectlUnite(VBOXVDMA_RECTL * pRectl1, const VBOXVDMA_RECTL * pRectl2)
429{
430 if (!pRectl1->width)
431 *pRectl1 = *pRectl2;
432 else
433 {
434 int16_t x21 = pRectl1->left + pRectl1->width;
435 int16_t x22 = pRectl2->left + pRectl2->width;
436 if (pRectl1->left > pRectl2->left)
437 {
438 pRectl1->left = pRectl2->left;
439 pRectl1->width = x21 < x22 ? x22 - pRectl1->left : x21 - pRectl1->left;
440 }
441 else if (x21 < x22)
442 pRectl1->width = x22 - pRectl1->left;
443
444 x21 = pRectl1->top + pRectl1->height;
445 x22 = pRectl2->top + pRectl2->height;
446 if (pRectl1->top > pRectl2->top)
447 {
448 pRectl1->top = pRectl2->top;
449 pRectl1->height = x21 < x22 ? x22 - pRectl1->top : x21 - pRectl1->top;
450 }
451 else if (x21 < x22)
452 pRectl1->height = x22 - pRectl1->top;
453 }
454}
455
456/*
457 * @return on success the number of bytes the command contained, otherwise - VERR_xxx error code
458 */
459static int vboxVDMACmdExecBlt(PVBOXVDMAHOST pVdma, const PVBOXVDMACMD_DMA_PRESENT_BLT pBlt, uint32_t cbBuffer)
460{
461 const uint32_t cbBlt = VBOXVDMACMD_BODY_FIELD_OFFSET(uint32_t, VBOXVDMACMD_DMA_PRESENT_BLT, aDstSubRects[pBlt->cDstSubRects]);
462 Assert(cbBlt <= cbBuffer);
463 if (cbBuffer < cbBlt)
464 return VERR_INVALID_FUNCTION;
465
466 /* we do not support stretching for now */
467 Assert(pBlt->srcRectl.width == pBlt->dstRectl.width);
468 Assert(pBlt->srcRectl.height == pBlt->dstRectl.height);
469 if (pBlt->srcRectl.width != pBlt->dstRectl.width)
470 return VERR_INVALID_FUNCTION;
471 if (pBlt->srcRectl.height != pBlt->dstRectl.height)
472 return VERR_INVALID_FUNCTION;
473 Assert(pBlt->cDstSubRects);
474
475 uint8_t * pvRam = pVdma->pVGAState->vram_ptrR3;
476 VBOXVDMA_RECTL updateRectl = {0, 0, 0, 0};
477
478 if (pBlt->cDstSubRects)
479 {
480 VBOXVDMA_RECTL dstRectl, srcRectl;
481 const VBOXVDMA_RECTL *pDstRectl, *pSrcRectl;
482 for (uint32_t i = 0; i < pBlt->cDstSubRects; ++i)
483 {
484 pDstRectl = &pBlt->aDstSubRects[i];
485 if (pBlt->dstRectl.left || pBlt->dstRectl.top)
486 {
487 dstRectl.left = pDstRectl->left + pBlt->dstRectl.left;
488 dstRectl.top = pDstRectl->top + pBlt->dstRectl.top;
489 dstRectl.width = pDstRectl->width;
490 dstRectl.height = pDstRectl->height;
491 pDstRectl = &dstRectl;
492 }
493
494 pSrcRectl = &pBlt->aDstSubRects[i];
495 if (pBlt->srcRectl.left || pBlt->srcRectl.top)
496 {
497 srcRectl.left = pSrcRectl->left + pBlt->srcRectl.left;
498 srcRectl.top = pSrcRectl->top + pBlt->srcRectl.top;
499 srcRectl.width = pSrcRectl->width;
500 srcRectl.height = pSrcRectl->height;
501 pSrcRectl = &srcRectl;
502 }
503
504 int rc = vboxVDMACmdExecBltPerform(pVdma, pvRam + pBlt->offDst, pvRam + pBlt->offSrc,
505 &pBlt->dstDesc, &pBlt->srcDesc,
506 pDstRectl,
507 pSrcRectl);
508 AssertRC(rc);
509 if (!RT_SUCCESS(rc))
510 return rc;
511
512 vboxVDMARectlUnite(&updateRectl, pDstRectl);
513 }
514 }
515 else
516 {
517 int rc = vboxVDMACmdExecBltPerform(pVdma, pvRam + pBlt->offDst, pvRam + pBlt->offSrc,
518 &pBlt->dstDesc, &pBlt->srcDesc,
519 &pBlt->dstRectl,
520 &pBlt->srcRectl);
521 AssertRC(rc);
522 if (!RT_SUCCESS(rc))
523 return rc;
524
525 vboxVDMARectlUnite(&updateRectl, &pBlt->dstRectl);
526 }
527
528#ifdef VBOX_VDMA_WITH_WORKERTHREAD
529 int iView = 0;
530 /* @todo: fixme: check if update is needed and get iView */
531 vboxVDMANotifyPrimaryUpdate (pVdma->pVGAState, iView, &updateRectl);
532#endif
533
534 return cbBlt;
535}
536
537static int vboxVDMACmdExecBpbTransfer(PVBOXVDMAHOST pVdma, const PVBOXVDMACMD_DMA_BPB_TRANSFER pTransfer, uint32_t cbBuffer)
538{
539 if (cbBuffer < sizeof (*pTransfer))
540 return VERR_INVALID_PARAMETER;
541
542 PVGASTATE pVGAState = pVdma->pVGAState;
543 uint8_t * pvRam = pVGAState->vram_ptrR3;
544 PGMPAGEMAPLOCK SrcLock;
545 PGMPAGEMAPLOCK DstLock;
546 PPDMDEVINS pDevIns = pVdma->pVGAState->pDevInsR3;
547 const void * pvSrc;
548 void * pvDst;
549 int rc = VINF_SUCCESS;
550 uint32_t cbTransfer = pTransfer->cbTransferSize;
551 uint32_t cbTransfered = 0;
552 bool bSrcLocked = false;
553 bool bDstLocked = false;
554 do
555 {
556 uint32_t cbSubTransfer = cbTransfer;
557 if (pTransfer->fFlags & VBOXVDMACMD_DMA_BPB_TRANSFER_F_SRC_VRAMOFFSET)
558 {
559 pvSrc = pvRam + pTransfer->Src.offVramBuf + cbTransfered;
560 }
561 else
562 {
563 RTGCPHYS phPage = pTransfer->Src.phBuf;
564 phPage += cbTransfered;
565 rc = PDMDevHlpPhysGCPhys2CCPtrReadOnly(pDevIns, phPage, 0, &pvSrc, &SrcLock);
566 AssertRC(rc);
567 if (RT_SUCCESS(rc))
568 {
569 bSrcLocked = true;
570 cbSubTransfer = RT_MIN(cbSubTransfer, 0x1000);
571 }
572 else
573 {
574 break;
575 }
576 }
577
578 if (pTransfer->fFlags & VBOXVDMACMD_DMA_BPB_TRANSFER_F_DST_VRAMOFFSET)
579 {
580 pvDst = pvRam + pTransfer->Dst.offVramBuf + cbTransfered;
581 }
582 else
583 {
584 RTGCPHYS phPage = pTransfer->Dst.phBuf;
585 phPage += cbTransfered;
586 rc = PDMDevHlpPhysGCPhys2CCPtr(pDevIns, phPage, 0, &pvDst, &DstLock);
587 AssertRC(rc);
588 if (RT_SUCCESS(rc))
589 {
590 bDstLocked = true;
591 cbSubTransfer = RT_MIN(cbSubTransfer, 0x1000);
592 }
593 else
594 {
595 break;
596 }
597 }
598
599 if (RT_SUCCESS(rc))
600 {
601 memcpy(pvDst, pvSrc, cbSubTransfer);
602 cbTransfer -= cbSubTransfer;
603 cbTransfered += cbSubTransfer;
604 }
605 else
606 {
607 cbTransfer = 0; /* to break */
608 }
609
610 if (bSrcLocked)
611 PDMDevHlpPhysReleasePageMappingLock(pDevIns, &SrcLock);
612 if (bDstLocked)
613 PDMDevHlpPhysReleasePageMappingLock(pDevIns, &DstLock);
614 } while (cbTransfer);
615
616 if (RT_SUCCESS(rc))
617 return sizeof (*pTransfer);
618 return rc;
619}
620
621static int vboxVDMACmdExec(PVBOXVDMAHOST pVdma, const uint8_t *pvBuffer, uint32_t cbBuffer)
622{
623 do
624 {
625 Assert(pvBuffer);
626 Assert(cbBuffer >= VBOXVDMACMD_HEADER_SIZE());
627
628 if (!pvBuffer)
629 return VERR_INVALID_PARAMETER;
630 if (cbBuffer < VBOXVDMACMD_HEADER_SIZE())
631 return VERR_INVALID_PARAMETER;
632
633 PVBOXVDMACMD pCmd = (PVBOXVDMACMD)pvBuffer;
634 uint32_t cbCmd = 0;
635 switch (pCmd->enmType)
636 {
637 case VBOXVDMACMD_TYPE_CHROMIUM_CMD:
638 {
639#ifdef VBOXWDDM_TEST_UHGSMI
640 static int count = 0;
641 static uint64_t start, end;
642 if (count==0)
643 {
644 start = RTTimeNanoTS();
645 }
646 ++count;
647 if (count==100000)
648 {
649 end = RTTimeNanoTS();
650 float ems = (end-start)/1000000.f;
651 LogRel(("100000 calls took %i ms, %i cps\n", (int)ems, (int)(100000.f*1000.f/ems) ));
652 }
653#endif
654 /* todo: post the buffer to chromium */
655 return VINF_SUCCESS;
656 }
657 case VBOXVDMACMD_TYPE_DMA_PRESENT_BLT:
658 {
659 const PVBOXVDMACMD_DMA_PRESENT_BLT pBlt = VBOXVDMACMD_BODY(pCmd, VBOXVDMACMD_DMA_PRESENT_BLT);
660 int cbBlt = vboxVDMACmdExecBlt(pVdma, pBlt, cbBuffer);
661 Assert(cbBlt >= 0);
662 Assert((uint32_t)cbBlt <= cbBuffer);
663 if (cbBlt >= 0)
664 {
665 if ((uint32_t)cbBlt == cbBuffer)
666 return VINF_SUCCESS;
667 else
668 {
669 cbBuffer -= (uint32_t)cbBlt;
670 pvBuffer -= cbBlt;
671 }
672 }
673 else
674 return cbBlt; /* error */
675 break;
676 }
677 case VBOXVDMACMD_TYPE_DMA_BPB_TRANSFER:
678 {
679 const PVBOXVDMACMD_DMA_BPB_TRANSFER pTransfer = VBOXVDMACMD_BODY(pCmd, VBOXVDMACMD_DMA_BPB_TRANSFER);
680 int cbTransfer = vboxVDMACmdExecBpbTransfer(pVdma, pTransfer, cbBuffer);
681 Assert(cbTransfer >= 0);
682 Assert((uint32_t)cbTransfer <= cbBuffer);
683 if (cbTransfer >= 0)
684 {
685 if ((uint32_t)cbTransfer == cbBuffer)
686 return VINF_SUCCESS;
687 else
688 {
689 cbBuffer -= (uint32_t)cbTransfer;
690 pvBuffer -= cbTransfer;
691 }
692 }
693 else
694 return cbTransfer; /* error */
695 break;
696 }
697 case VBOXVDMACMD_TYPE_DMA_NOP:
698 return VINF_SUCCESS;
699 case VBOXVDMACMD_TYPE_CHILD_STATUS_IRQ:
700 return VINF_SUCCESS;
701 default:
702 AssertBreakpoint();
703 return VERR_INVALID_FUNCTION;
704 }
705 } while (1);
706
707 /* we should not be here */
708 AssertBreakpoint();
709 return VERR_INVALID_STATE;
710}
711
712#ifdef VBOX_VDMA_WITH_WORKERTHREAD
713
714int vboxVDMAPipeConstruct(PVBOXVDMAPIPE pPipe)
715{
716 int rc = RTSemEventCreate(&pPipe->hEvent);
717 AssertRC(rc);
718 if (RT_SUCCESS(rc))
719 {
720 rc = RTCritSectInit(&pPipe->hCritSect);
721 AssertRC(rc);
722 if (RT_SUCCESS(rc))
723 {
724 pPipe->enmState = VBOXVDMAPIPE_STATE_CREATED;
725 pPipe->bNeedNotify = true;
726 return VINF_SUCCESS;
727// RTCritSectDelete(pPipe->hCritSect);
728 }
729 RTSemEventDestroy(pPipe->hEvent);
730 }
731 return rc;
732}
733
734int vboxVDMAPipeOpenServer(PVBOXVDMAPIPE pPipe)
735{
736 int rc = RTCritSectEnter(&pPipe->hCritSect);
737 AssertRC(rc);
738 if (RT_SUCCESS(rc))
739 {
740 Assert(pPipe->enmState == VBOXVDMAPIPE_STATE_CREATED);
741 switch (pPipe->enmState)
742 {
743 case VBOXVDMAPIPE_STATE_CREATED:
744 pPipe->enmState = VBOXVDMAPIPE_STATE_OPENNED;
745 pPipe->bNeedNotify = false;
746 rc = VINF_SUCCESS;
747 break;
748 case VBOXVDMAPIPE_STATE_OPENNED:
749 pPipe->bNeedNotify = false;
750 rc = VINF_ALREADY_INITIALIZED;
751 break;
752 default:
753 AssertBreakpoint();
754 rc = VERR_INVALID_STATE;
755 break;
756 }
757
758 RTCritSectLeave(&pPipe->hCritSect);
759 }
760 return rc;
761}
762
763int vboxVDMAPipeCloseServer(PVBOXVDMAPIPE pPipe)
764{
765 int rc = RTCritSectEnter(&pPipe->hCritSect);
766 AssertRC(rc);
767 if (RT_SUCCESS(rc))
768 {
769 Assert(pPipe->enmState == VBOXVDMAPIPE_STATE_CLOSED
770 || pPipe->enmState == VBOXVDMAPIPE_STATE_CLOSING);
771 switch (pPipe->enmState)
772 {
773 case VBOXVDMAPIPE_STATE_CLOSING:
774 pPipe->enmState = VBOXVDMAPIPE_STATE_CLOSED;
775 rc = VINF_SUCCESS;
776 break;
777 case VBOXVDMAPIPE_STATE_CLOSED:
778 rc = VINF_ALREADY_INITIALIZED;
779 break;
780 default:
781 AssertBreakpoint();
782 rc = VERR_INVALID_STATE;
783 break;
784 }
785
786 RTCritSectLeave(&pPipe->hCritSect);
787 }
788 return rc;
789}
790
791int vboxVDMAPipeCloseClient(PVBOXVDMAPIPE pPipe)
792{
793 int rc = RTCritSectEnter(&pPipe->hCritSect);
794 AssertRC(rc);
795 if (RT_SUCCESS(rc))
796 {
797 bool bNeedNotify = false;
798 Assert(pPipe->enmState == VBOXVDMAPIPE_STATE_OPENNED
799 || pPipe->enmState == VBOXVDMAPIPE_STATE_CREATED
800 || pPipe->enmState == VBOXVDMAPIPE_STATE_CLOSED);
801 switch (pPipe->enmState)
802 {
803 case VBOXVDMAPIPE_STATE_OPENNED:
804 pPipe->enmState = VBOXVDMAPIPE_STATE_CLOSING;
805 bNeedNotify = pPipe->bNeedNotify;
806 pPipe->bNeedNotify = false;
807 break;
808 case VBOXVDMAPIPE_STATE_CREATED:
809 pPipe->enmState = VBOXVDMAPIPE_STATE_CLOSED;
810 pPipe->bNeedNotify = false;
811 break;
812 case VBOXVDMAPIPE_STATE_CLOSED:
813 rc = VINF_ALREADY_INITIALIZED;
814 break;
815 default:
816 AssertBreakpoint();
817 rc = VERR_INVALID_STATE;
818 break;
819 }
820
821 RTCritSectLeave(&pPipe->hCritSect);
822
823 if (bNeedNotify)
824 {
825 rc = RTSemEventSignal(pPipe->hEvent);
826 AssertRC(rc);
827 }
828 }
829 return rc;
830}
831
832
833typedef DECLCALLBACK(bool) FNHVBOXVDMARWCB(PVBOXVDMAPIPE pPipe, void *pvCallback);
834typedef FNHVBOXVDMARWCB *PFNHVBOXVDMARWCB;
835
836int vboxVDMAPipeModifyServer(PVBOXVDMAPIPE pPipe, PFNHVBOXVDMARWCB pfnCallback, void * pvCallback)
837{
838 int rc = RTCritSectEnter(&pPipe->hCritSect);
839 AssertRC(rc);
840 if (RT_SUCCESS(rc))
841 {
842 do
843 {
844 Assert(pPipe->enmState == VBOXVDMAPIPE_STATE_OPENNED
845 || pPipe->enmState == VBOXVDMAPIPE_STATE_CLOSING);
846
847 if (pPipe->enmState >= VBOXVDMAPIPE_STATE_OPENNED)
848 {
849 bool bProcessing = pfnCallback(pPipe, pvCallback);
850 pPipe->bNeedNotify = !bProcessing;
851 if (bProcessing)
852 {
853 RTCritSectLeave(&pPipe->hCritSect);
854 rc = VINF_SUCCESS;
855 break;
856 }
857 else if (pPipe->enmState == VBOXVDMAPIPE_STATE_CLOSING)
858 {
859 pPipe->enmState = VBOXVDMAPIPE_STATE_CLOSED;
860 RTCritSectLeave(&pPipe->hCritSect);
861 rc = VINF_EOF;
862 break;
863 }
864 }
865 else
866 {
867 AssertBreakpoint();
868 rc = VERR_INVALID_STATE;
869 RTCritSectLeave(&pPipe->hCritSect);
870 break;
871 }
872
873 RTCritSectLeave(&pPipe->hCritSect);
874
875 rc = RTSemEventWait(pPipe->hEvent, RT_INDEFINITE_WAIT);
876 AssertRC(rc);
877 if (!RT_SUCCESS(rc))
878 break;
879
880 rc = RTCritSectEnter(&pPipe->hCritSect);
881 AssertRC(rc);
882 if (!RT_SUCCESS(rc))
883 break;
884 } while (1);
885 }
886
887 return rc;
888}
889
890int vboxVDMAPipeModifyClient(PVBOXVDMAPIPE pPipe, PFNHVBOXVDMARWCB pfnCallback, void * pvCallback)
891{
892 int rc = RTCritSectEnter(&pPipe->hCritSect);
893 AssertRC(rc);
894 if (RT_SUCCESS(rc))
895 {
896 bool bNeedNotify = false;
897 Assert(pPipe->enmState == VBOXVDMAPIPE_STATE_OPENNED);
898 if (pPipe->enmState == VBOXVDMAPIPE_STATE_OPENNED)
899 {
900 bool bModified = pfnCallback(pPipe, pvCallback);
901 if (bModified)
902 {
903 bNeedNotify = pPipe->bNeedNotify;
904 pPipe->bNeedNotify = false;
905 }
906 }
907 else
908 rc = VERR_INVALID_STATE;
909
910 RTCritSectLeave(&pPipe->hCritSect);
911
912 if (bNeedNotify)
913 {
914 rc = RTSemEventSignal(pPipe->hEvent);
915 AssertRC(rc);
916 }
917 }
918 return rc;
919}
920
921int vboxVDMAPipeDestruct(PVBOXVDMAPIPE pPipe)
922{
923 Assert(pPipe->enmState == VBOXVDMAPIPE_STATE_CLOSED
924 || pPipe->enmState == VBOXVDMAPIPE_STATE_CREATED);
925 /* ensure the pipe is closed */
926 vboxVDMAPipeCloseClient(pPipe);
927
928 Assert(pPipe->enmState == VBOXVDMAPIPE_STATE_CLOSED);
929
930 if (pPipe->enmState != VBOXVDMAPIPE_STATE_CLOSED)
931 return VERR_INVALID_STATE;
932
933 int rc = RTCritSectDelete(&pPipe->hCritSect);
934 AssertRC(rc);
935
936 rc = RTSemEventDestroy(pPipe->hEvent);
937 AssertRC(rc);
938
939 return VINF_SUCCESS;
940}
941#endif
942
943static void vboxVDMACommandProcess(PVBOXVDMAHOST pVdma, PVBOXVDMACBUF_DR pCmd, uint32_t cbCmd)
944{
945 PHGSMIINSTANCE pHgsmi = pVdma->pHgsmi;
946 const uint8_t * pvBuf;
947 PGMPAGEMAPLOCK Lock;
948 int rc;
949 bool bReleaseLocked = false;
950
951 do
952 {
953 PPDMDEVINS pDevIns = pVdma->pVGAState->pDevInsR3;
954
955 if (pCmd->fFlags & VBOXVDMACBUF_FLAG_BUF_FOLLOWS_DR)
956 pvBuf = VBOXVDMACBUF_DR_TAIL(pCmd, const uint8_t);
957 else if (pCmd->fFlags & VBOXVDMACBUF_FLAG_BUF_VRAM_OFFSET)
958 {
959 uint8_t * pvRam = pVdma->pVGAState->vram_ptrR3;
960 pvBuf = pvRam + pCmd->Location.offVramBuf;
961 }
962 else
963 {
964 RTGCPHYS phPage = pCmd->Location.phBuf & ~0xfffULL;
965 uint32_t offset = pCmd->Location.phBuf & 0xfff;
966 Assert(offset + pCmd->cbBuf <= 0x1000);
967 if (offset + pCmd->cbBuf > 0x1000)
968 {
969 /* @todo: more advanced mechanism of command buffer proc is actually needed */
970 rc = VERR_INVALID_PARAMETER;
971 break;
972 }
973
974 const void * pvPageBuf;
975 rc = PDMDevHlpPhysGCPhys2CCPtrReadOnly(pDevIns, phPage, 0, &pvPageBuf, &Lock);
976 AssertRC(rc);
977 if (!RT_SUCCESS(rc))
978 {
979 /* @todo: if (rc == VERR_PGM_PHYS_PAGE_RESERVED) -> fall back on using PGMPhysRead ?? */
980 break;
981 }
982
983 pvBuf = (const uint8_t *)pvPageBuf;
984 pvBuf += offset;
985
986 bReleaseLocked = true;
987 }
988
989 rc = vboxVDMACmdExec(pVdma, pvBuf, pCmd->cbBuf);
990 AssertRC(rc);
991
992 if (bReleaseLocked)
993 PDMDevHlpPhysReleasePageMappingLock(pDevIns, &Lock);
994 } while (0);
995
996 pCmd->rc = rc;
997
998 rc = VBoxSHGSMICommandComplete (pHgsmi, pCmd);
999 AssertRC(rc);
1000}
1001
1002static void vboxVDMAControlProcess(PVBOXVDMAHOST pVdma, PVBOXVDMA_CTL pCmd)
1003{
1004 PHGSMIINSTANCE pHgsmi = pVdma->pHgsmi;
1005 pCmd->i32Result = VINF_SUCCESS;
1006 int rc = VBoxSHGSMICommandComplete (pHgsmi, pCmd);
1007 AssertRC(rc);
1008}
1009
1010#ifdef VBOX_VDMA_WITH_WORKERTHREAD
1011typedef struct
1012{
1013 struct VBOXVDMAHOST *pVdma;
1014 VBOXVDMAPIPE_CMD_BODY Cmd;
1015 bool bHasCmd;
1016} VBOXVDMACMD_PROCESS_CONTEXT, *PVBOXVDMACMD_PROCESS_CONTEXT;
1017
1018static DECLCALLBACK(bool) vboxVDMACommandProcessCb(PVBOXVDMAPIPE pPipe, void *pvCallback)
1019{
1020 PVBOXVDMACMD_PROCESS_CONTEXT pContext = (PVBOXVDMACMD_PROCESS_CONTEXT)pvCallback;
1021 struct VBOXVDMAHOST *pVdma = pContext->pVdma;
1022 HGSMILISTENTRY *pEntry = hgsmiListRemoveHead(&pVdma->PendingList);
1023 if (pEntry)
1024 {
1025 PVBOXVDMAPIPE_CMD pPipeCmd = VBOXVDMAPIPE_CMD_FROM_ENTRY(pEntry);
1026 Assert(pPipeCmd);
1027 pContext->Cmd = pPipeCmd->Cmd;
1028 hgsmiListPrepend(&pVdma->CmdPool.List, pEntry);
1029 pContext->bHasCmd = true;
1030 return true;
1031 }
1032
1033 pContext->bHasCmd = false;
1034 return false;
1035}
1036
1037static DECLCALLBACK(int) vboxVDMAWorkerThread(RTTHREAD ThreadSelf, void *pvUser)
1038{
1039 PVBOXVDMAHOST pVdma = (PVBOXVDMAHOST)pvUser;
1040 PHGSMIINSTANCE pHgsmi = pVdma->pHgsmi;
1041 VBOXVDMACMD_PROCESS_CONTEXT Context;
1042 Context.pVdma = pVdma;
1043
1044 int rc = vboxVDMAPipeOpenServer(&pVdma->Pipe);
1045 AssertRC(rc);
1046 if (RT_SUCCESS(rc))
1047 {
1048 do
1049 {
1050 rc = vboxVDMAPipeModifyServer(&pVdma->Pipe, vboxVDMACommandProcessCb, &Context);
1051 AssertRC(rc);
1052 if (RT_SUCCESS(rc))
1053 {
1054 switch (Context.Cmd.enmType)
1055 {
1056 case VBOXVDMAPIPE_CMD_TYPE_DMACMD:
1057 {
1058 PVBOXVDMACBUF_DR pDr = Context.Cmd.u.pDr;
1059 vboxVDMACommandProcess(pVdma, pDr);
1060 break;
1061 }
1062 case VBOXVDMAPIPE_CMD_TYPE_DMACTL:
1063 {
1064 PVBOXVDMA_CTL pCtl = Context.Cmd.u.pCtl;
1065 vboxVDMAControlProcess(pVdma, pCtl);
1066 break;
1067 }
1068 default:
1069 AssertBreakpoint();
1070 break;
1071 }
1072
1073 if (rc == VINF_EOF)
1074 {
1075 rc = VINF_SUCCESS;
1076 break;
1077 }
1078 }
1079 else
1080 break;
1081 } while (1);
1082 }
1083
1084 /* always try to close the pipe to make sure the client side is notified */
1085 int tmpRc = vboxVDMAPipeCloseServer(&pVdma->Pipe);
1086 AssertRC(tmpRc);
1087 return rc;
1088}
1089#endif
1090
1091#ifdef VBOX_VDMA_WITH_WATCHDOG
1092static DECLCALLBACK(void) vboxVDMAWatchDogTimer(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
1093{
1094 VBOXVDMAHOST *pVdma = (VBOXVDMAHOST *)pvUser;
1095 PVGASTATE pVGAState = pVdma->pVGAState;
1096 VBVARaiseIrq(pVGAState, HGSMIHOSTFLAGS_WATCHDOG);
1097}
1098
1099static int vboxVDMAWatchDogCtl(struct VBOXVDMAHOST *pVdma, uint32_t cMillis)
1100{
1101 PPDMDEVINS pDevIns = pVdma->pVGAState->pDevInsR3;
1102 if (cMillis)
1103 TMTimerSetMillies(pVdma->WatchDogTimer, cMillis);
1104 else
1105 TMTimerStop(pVdma->WatchDogTimer);
1106 return VINF_SUCCESS;
1107}
1108#endif
1109
1110int vboxVDMAConstruct(PVGASTATE pVGAState, uint32_t cPipeElements)
1111{
1112 int rc;
1113#ifdef VBOX_VDMA_WITH_WORKERTHREAD
1114 PVBOXVDMAHOST pVdma = (PVBOXVDMAHOST)RTMemAllocZ(RT_OFFSETOF(VBOXVDMAHOST, CmdPool.aCmds[cPipeElements]));
1115#else
1116 PVBOXVDMAHOST pVdma = (PVBOXVDMAHOST)RTMemAllocZ(sizeof(*pVdma));
1117#endif
1118 Assert(pVdma);
1119 if (pVdma)
1120 {
1121 pVdma->pHgsmi = pVGAState->pHGSMI;
1122 pVdma->pVGAState = pVGAState;
1123
1124#ifdef VBOX_VDMA_WITH_WATCHDOG
1125 rc = PDMDevHlpTMTimerCreate(pVGAState->pDevInsR3, TMCLOCK_REAL, vboxVDMAWatchDogTimer,
1126 pVdma, TMTIMER_FLAGS_NO_CRIT_SECT,
1127 "VDMA WatchDog Timer", &pVdma->WatchDogTimer);
1128 AssertRC(rc);
1129#endif
1130#ifdef VBOX_VDMA_WITH_WORKERTHREAD
1131 hgsmiListInit(&pVdma->PendingList);
1132 rc = vboxVDMAPipeConstruct(&pVdma->Pipe);
1133 AssertRC(rc);
1134 if (RT_SUCCESS(rc))
1135 {
1136 rc = RTThreadCreate(&pVdma->hWorkerThread, vboxVDMAWorkerThread, pVdma, 0, RTTHREADTYPE_IO, RTTHREADFLAGS_WAITABLE, "VDMA");
1137 AssertRC(rc);
1138 if (RT_SUCCESS(rc))
1139 {
1140 hgsmiListInit(&pVdma->CmdPool.List);
1141 pVdma->CmdPool.cCmds = cPipeElements;
1142 for (uint32_t i = 0; i < cPipeElements; ++i)
1143 {
1144 hgsmiListAppend(&pVdma->CmdPool.List, &pVdma->CmdPool.aCmds[i].Entry);
1145 }
1146# if 0 //def VBOX_WITH_CRHGSMI
1147 int tmpRc = vboxVDMACrCtlHgsmiSetup(pVdma);
1148# endif
1149#endif
1150 pVGAState->pVdma = pVdma;
1151#ifdef VBOX_WITH_CRHGSMI
1152 int rcIgnored = vboxVDMACrCtlHgsmiSetup(pVdma); NOREF(rcIgnored); /** @todo is this ignoring intentional? */
1153#endif
1154 return VINF_SUCCESS;
1155#ifdef VBOX_VDMA_WITH_WORKERTHREAD
1156 }
1157
1158 int tmpRc = vboxVDMAPipeDestruct(&pVdma->Pipe);
1159 AssertRC(tmpRc);
1160 }
1161
1162 RTMemFree(pVdma);
1163#endif
1164 }
1165 else
1166 rc = VERR_OUT_OF_RESOURCES;
1167
1168 return rc;
1169}
1170
1171int vboxVDMADestruct(struct VBOXVDMAHOST *pVdma)
1172{
1173#ifdef VBOX_VDMA_WITH_WORKERTHREAD
1174 /* @todo: implement*/
1175 AssertBreakpoint();
1176#endif
1177 RTMemFree(pVdma);
1178 return VINF_SUCCESS;
1179}
1180
1181#ifdef VBOX_VDMA_WITH_WORKERTHREAD
1182typedef struct
1183{
1184 struct VBOXVDMAHOST *pVdma;
1185 VBOXVDMAPIPE_CMD_BODY Cmd;
1186 bool bQueued;
1187} VBOXVDMACMD_SUBMIT_CONTEXT, *PVBOXVDMACMD_SUBMIT_CONTEXT;
1188
1189DECLCALLBACK(bool) vboxVDMACommandSubmitCb(PVBOXVDMAPIPE pPipe, void *pvCallback)
1190{
1191 PVBOXVDMACMD_SUBMIT_CONTEXT pContext = (PVBOXVDMACMD_SUBMIT_CONTEXT)pvCallback;
1192 struct VBOXVDMAHOST *pVdma = pContext->pVdma;
1193 HGSMILISTENTRY *pEntry = hgsmiListRemoveHead(&pVdma->CmdPool.List);
1194 Assert(pEntry);
1195 if (pEntry)
1196 {
1197 PVBOXVDMAPIPE_CMD pPipeCmd = VBOXVDMAPIPE_CMD_FROM_ENTRY(pEntry);
1198 pPipeCmd->Cmd = pContext->Cmd;
1199 VBoxSHGSMICommandMarkAsynchCompletion(pContext->Cmd.u.pvCmd);
1200 pContext->bQueued = true;
1201 hgsmiListAppend(&pVdma->PendingList, pEntry);
1202 return true;
1203 }
1204
1205 /* @todo: should we try to flush some commands here? */
1206 pContext->bQueued = false;
1207 return false;
1208}
1209#endif
1210
1211int vboxVDMASaveStateExecPrep(struct VBOXVDMAHOST *pVdma, PSSMHANDLE pSSM)
1212{
1213#ifdef VBOX_WITH_CRHGSMI
1214 PVGASTATE pVGAState = pVdma->pVGAState;
1215 PVBOXVDMACMD_CHROMIUM_CTL pCmd = (PVBOXVDMACMD_CHROMIUM_CTL)vboxVDMACrCtlCreate(
1216 VBOXVDMACMD_CHROMIUM_CTL_TYPE_SAVESTATE_BEGIN, sizeof (*pCmd));
1217 Assert(pCmd);
1218 if (pCmd)
1219 {
1220 int rc = vboxVDMACrCtlPost(pVGAState, pCmd, sizeof (*pCmd));
1221 AssertRC(rc);
1222 if (RT_SUCCESS(rc))
1223 {
1224 rc = vboxVDMACrCtlGetRc(pCmd);
1225 }
1226 vboxVDMACrCtlRelease(pCmd);
1227 return rc;
1228 }
1229 return VERR_NO_MEMORY;
1230#else
1231 return VINF_SUCCESS;
1232#endif
1233}
1234
1235int vboxVDMASaveStateExecDone(struct VBOXVDMAHOST *pVdma, PSSMHANDLE pSSM)
1236{
1237#ifdef VBOX_WITH_CRHGSMI
1238 PVGASTATE pVGAState = pVdma->pVGAState;
1239 PVBOXVDMACMD_CHROMIUM_CTL pCmd = (PVBOXVDMACMD_CHROMIUM_CTL)vboxVDMACrCtlCreate(
1240 VBOXVDMACMD_CHROMIUM_CTL_TYPE_SAVESTATE_END, sizeof (*pCmd));
1241 Assert(pCmd);
1242 if (pCmd)
1243 {
1244 int rc = vboxVDMACrCtlPost(pVGAState, pCmd, sizeof (*pCmd));
1245 AssertRC(rc);
1246 if (RT_SUCCESS(rc))
1247 {
1248 rc = vboxVDMACrCtlGetRc(pCmd);
1249 }
1250 vboxVDMACrCtlRelease(pCmd);
1251 return rc;
1252 }
1253 return VERR_NO_MEMORY;
1254#else
1255 return VINF_SUCCESS;
1256#endif
1257}
1258
1259void vboxVDMAControl(struct VBOXVDMAHOST *pVdma, PVBOXVDMA_CTL pCmd, uint32_t cbCmd)
1260{
1261#if 1
1262 PHGSMIINSTANCE pIns = pVdma->pHgsmi;
1263
1264 switch (pCmd->enmCtl)
1265 {
1266 case VBOXVDMA_CTL_TYPE_ENABLE:
1267 pCmd->i32Result = VINF_SUCCESS;
1268 break;
1269 case VBOXVDMA_CTL_TYPE_DISABLE:
1270 pCmd->i32Result = VINF_SUCCESS;
1271 break;
1272 case VBOXVDMA_CTL_TYPE_FLUSH:
1273 pCmd->i32Result = VINF_SUCCESS;
1274 break;
1275#ifdef VBOX_VDMA_WITH_WATCHDOG
1276 case VBOXVDMA_CTL_TYPE_WATCHDOG:
1277 pCmd->i32Result = vboxVDMAWatchDogCtl(pVdma, pCmd->u32Offset);
1278 break;
1279#endif
1280 default:
1281 AssertBreakpoint();
1282 pCmd->i32Result = VERR_NOT_SUPPORTED;
1283 }
1284
1285 int rc = VBoxSHGSMICommandComplete (pIns, pCmd);
1286 AssertRC(rc);
1287#else
1288 /* test asinch completion */
1289 VBOXVDMACMD_SUBMIT_CONTEXT Context;
1290 Context.pVdma = pVdma;
1291 Context.Cmd.enmType = VBOXVDMAPIPE_CMD_TYPE_DMACTL;
1292 Context.Cmd.u.pCtl = pCmd;
1293
1294 int rc = vboxVDMAPipeModifyClient(&pVdma->Pipe, vboxVDMACommandSubmitCb, &Context);
1295 AssertRC(rc);
1296 if (RT_SUCCESS(rc))
1297 {
1298 Assert(Context.bQueued);
1299 if (Context.bQueued)
1300 {
1301 /* success */
1302 return;
1303 }
1304 rc = VERR_OUT_OF_RESOURCES;
1305 }
1306
1307 /* failure */
1308 Assert(RT_FAILURE(rc));
1309 PHGSMIINSTANCE pIns = pVdma->pHgsmi;
1310 pCmd->i32Result = rc;
1311 int tmpRc = VBoxSHGSMICommandComplete (pIns, pCmd);
1312 AssertRC(tmpRc);
1313
1314#endif
1315}
1316
1317void vboxVDMACommand(struct VBOXVDMAHOST *pVdma, PVBOXVDMACBUF_DR pCmd, uint32_t cbCmd)
1318{
1319 int rc = VERR_NOT_IMPLEMENTED;
1320
1321#ifdef VBOX_WITH_CRHGSMI
1322 /* chromium commands are processed by crhomium hgcm thread independently from our internal cmd processing pipeline
1323 * this is why we process them specially */
1324 rc = vboxVDMACmdCheckCrCmd(pVdma, pCmd, cbCmd);
1325 if (rc == VINF_SUCCESS)
1326 return;
1327
1328 if (RT_FAILURE(rc))
1329 {
1330 pCmd->rc = rc;
1331 rc = VBoxSHGSMICommandComplete (pVdma->pHgsmi, pCmd);
1332 AssertRC(rc);
1333 return;
1334 }
1335#endif
1336
1337#ifndef VBOX_VDMA_WITH_WORKERTHREAD
1338 vboxVDMACommandProcess(pVdma, pCmd, cbCmd);
1339#else
1340
1341# ifdef DEBUG_misha
1342 Assert(0);
1343# endif
1344
1345 VBOXVDMACMD_SUBMIT_CONTEXT Context;
1346 Context.pVdma = pVdma;
1347 Context.Cmd.enmType = VBOXVDMAPIPE_CMD_TYPE_DMACMD;
1348 Context.Cmd.u.pDr = pCmd;
1349
1350 rc = vboxVDMAPipeModifyClient(&pVdma->Pipe, vboxVDMACommandSubmitCb, &Context);
1351 AssertRC(rc);
1352 if (RT_SUCCESS(rc))
1353 {
1354 Assert(Context.bQueued);
1355 if (Context.bQueued)
1356 {
1357 /* success */
1358 return;
1359 }
1360 rc = VERR_OUT_OF_RESOURCES;
1361 }
1362 /* failure */
1363 Assert(RT_FAILURE(rc));
1364 PHGSMIINSTANCE pIns = pVdma->pHgsmi;
1365 pCmd->rc = rc;
1366 int tmpRc = VBoxSHGSMICommandComplete (pIns, pCmd);
1367 AssertRC(tmpRc);
1368#endif
1369}
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