VirtualBox

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

Last change on this file since 53006 was 52504, checked in by vboxsync, 10 years ago

DevVGA_VDMA: execute vgaUpdateDisplayAll on EMT

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 111.1 KB
Line 
1/** @file
2 * Video DMA (VDMA) support.
3 */
4
5/*
6 * Copyright (C) 2006-2012 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#include <iprt/list.h>
24#include <iprt/param.h>
25
26#include "DevVGA.h"
27#include "HGSMI/SHGSMIHost.h"
28
29#include <VBox/VBoxVideo3D.h>
30#include <VBox/VBoxVideoHost3D.h>
31
32#ifdef DEBUG_misha
33# define VBOXVDBG_MEMCACHE_DISABLE
34#endif
35
36#ifndef VBOXVDBG_MEMCACHE_DISABLE
37# include <iprt/memcache.h>
38#endif
39
40#ifdef DEBUG_misha
41#define WARN_BP() do { AssertFailed(); } while (0)
42#else
43#define WARN_BP() do { } while (0)
44#endif
45#define WARN(_msg) do { \
46 LogRel(_msg); \
47 WARN_BP(); \
48 } while (0)
49
50#define VBOXVDMATHREAD_STATE_TERMINATED 0
51#define VBOXVDMATHREAD_STATE_CREATING 1
52#define VBOXVDMATHREAD_STATE_CREATED 3
53#define VBOXVDMATHREAD_STATE_TERMINATING 4
54
55struct VBOXVDMATHREAD;
56
57typedef DECLCALLBACKPTR(void, PFNVBOXVDMATHREAD_CHANGED)(struct VBOXVDMATHREAD *pThread, int rc, void *pvThreadContext, void *pvChangeContext);
58
59static DECLCALLBACK(int) vboxCmdVBVACmdCallout(struct VBOXVDMAHOST *pVdma, struct VBOXCRCMDCTL* pCmd, VBOXCRCMDCTL_CALLOUT_LISTENTRY *pEntry, PFNVBOXCRCMDCTL_CALLOUT_CB pfnCb);
60
61
62typedef struct VBOXVDMATHREAD
63{
64 RTTHREAD hWorkerThread;
65 RTSEMEVENT hEvent;
66 volatile uint32_t u32State;
67 PFNVBOXVDMATHREAD_CHANGED pfnChanged;
68 void *pvChanged;
69} VBOXVDMATHREAD, *PVBOXVDMATHREAD;
70
71
72/* state transformations:
73 *
74 * submitter | processor
75 *
76 * LISTENING ---> PROCESSING
77 *
78 * */
79#define VBVAEXHOSTCONTEXT_STATE_LISTENING 0
80#define VBVAEXHOSTCONTEXT_STATE_PROCESSING 1
81
82#define VBVAEXHOSTCONTEXT_ESTATE_DISABLED -1
83#define VBVAEXHOSTCONTEXT_ESTATE_PAUSED 0
84#define VBVAEXHOSTCONTEXT_ESTATE_ENABLED 1
85
86typedef struct VBVAEXHOSTCONTEXT
87{
88 VBVABUFFER *pVBVA;
89 volatile int32_t i32State;
90 volatile int32_t i32EnableState;
91 volatile uint32_t u32cCtls;
92 /* critical section for accessing ctl lists */
93 RTCRITSECT CltCritSect;
94 RTLISTANCHOR GuestCtlList;
95 RTLISTANCHOR HostCtlList;
96#ifndef VBOXVDBG_MEMCACHE_DISABLE
97 RTMEMCACHE CtlCache;
98#endif
99} VBVAEXHOSTCONTEXT;
100
101typedef enum
102{
103 VBVAEXHOSTCTL_TYPE_UNDEFINED = 0,
104 VBVAEXHOSTCTL_TYPE_HH_INTERNAL_PAUSE,
105 VBVAEXHOSTCTL_TYPE_HH_INTERNAL_RESUME,
106 VBVAEXHOSTCTL_TYPE_HH_SAVESTATE,
107 VBVAEXHOSTCTL_TYPE_HH_LOADSTATE,
108 VBVAEXHOSTCTL_TYPE_HH_LOADSTATE_DONE,
109 VBVAEXHOSTCTL_TYPE_HH_BE_OPAQUE,
110 VBVAEXHOSTCTL_TYPE_HH_ON_HGCM_UNLOAD,
111 VBVAEXHOSTCTL_TYPE_GHH_BE_OPAQUE,
112 VBVAEXHOSTCTL_TYPE_GHH_ENABLE,
113 VBVAEXHOSTCTL_TYPE_GHH_ENABLE_PAUSED,
114 VBVAEXHOSTCTL_TYPE_GHH_DISABLE,
115 VBVAEXHOSTCTL_TYPE_GHH_RESIZE
116} VBVAEXHOSTCTL_TYPE;
117
118struct VBVAEXHOSTCTL;
119
120typedef DECLCALLBACKPTR(void, PFNVBVAEXHOSTCTL_COMPLETE)(VBVAEXHOSTCONTEXT *pVbva, struct VBVAEXHOSTCTL *pCtl, int rc, void *pvComplete);
121
122typedef struct VBVAEXHOSTCTL
123{
124 RTLISTNODE Node;
125 VBVAEXHOSTCTL_TYPE enmType;
126 union
127 {
128 struct
129 {
130 uint8_t * pu8Cmd;
131 uint32_t cbCmd;
132 } cmd;
133
134 struct
135 {
136 PSSMHANDLE pSSM;
137 uint32_t u32Version;
138 } state;
139 } u;
140 PFNVBVAEXHOSTCTL_COMPLETE pfnComplete;
141 void *pvComplete;
142} VBVAEXHOSTCTL;
143
144/* VBoxVBVAExHP**, i.e. processor functions, can NOT be called concurrently with each other,
145 * but can be called with other VBoxVBVAExS** (submitter) functions except Init/Start/Term aparently.
146 * Can only be called be the processor, i.e. the entity that acquired the processor state by direct or indirect call to the VBoxVBVAExHSCheckCommands
147 * see mor edetailed comments in headers for function definitions */
148typedef enum
149{
150 VBVAEXHOST_DATA_TYPE_NO_DATA = 0,
151 VBVAEXHOST_DATA_TYPE_CMD,
152 VBVAEXHOST_DATA_TYPE_HOSTCTL,
153 VBVAEXHOST_DATA_TYPE_GUESTCTL
154} VBVAEXHOST_DATA_TYPE;
155
156static DECLCALLBACK(int) vdmaVBVANotifyDisable(PVGASTATE pVGAState);
157
158
159static VBVAEXHOST_DATA_TYPE VBoxVBVAExHPDataGet(struct VBVAEXHOSTCONTEXT *pCmdVbva, uint8_t **ppCmd, uint32_t *pcbCmd);
160
161static void VBoxVBVAExHPDataCompleteCmd(struct VBVAEXHOSTCONTEXT *pCmdVbva, uint32_t cbCmd);
162static void VBoxVBVAExHPDataCompleteCtl(struct VBVAEXHOSTCONTEXT *pCmdVbva, VBVAEXHOSTCTL *pCtl, int rc);
163
164/* VBoxVBVAExHP**, i.e. processor functions, can NOT be called concurrently with each other,
165 * can be called concurrently with istelf as well as with other VBoxVBVAEx** functions except Init/Start/Term aparently */
166static int VBoxVBVAExHSCheckCommands(struct VBVAEXHOSTCONTEXT *pCmdVbva);
167
168static int VBoxVBVAExHSInit(struct VBVAEXHOSTCONTEXT *pCmdVbva);
169static int VBoxVBVAExHSEnable(struct VBVAEXHOSTCONTEXT *pCmdVbva, VBVABUFFER *pVBVA);
170static int VBoxVBVAExHSDisable(struct VBVAEXHOSTCONTEXT *pCmdVbva);
171static void VBoxVBVAExHSTerm(struct VBVAEXHOSTCONTEXT *pCmdVbva);
172static int VBoxVBVAExHSSaveState(struct VBVAEXHOSTCONTEXT *pCmdVbva, uint8_t* pu8VramBase, PSSMHANDLE pSSM);
173static int VBoxVBVAExHSLoadState(struct VBVAEXHOSTCONTEXT *pCmdVbva, uint8_t* pu8VramBase, PSSMHANDLE pSSM, uint32_t u32Version);
174
175static VBVAEXHOSTCTL* VBoxVBVAExHCtlAlloc(VBVAEXHOSTCONTEXT *pCmdVbva)
176{
177#ifndef VBOXVDBG_MEMCACHE_DISABLE
178 return (VBVAEXHOSTCTL*)RTMemCacheAlloc(pCmdVbva->CtlCache);
179#else
180 return (VBVAEXHOSTCTL*)RTMemAlloc(sizeof (VBVAEXHOSTCTL));
181#endif
182}
183
184static void VBoxVBVAExHCtlFree(VBVAEXHOSTCONTEXT *pCmdVbva, VBVAEXHOSTCTL *pCtl)
185{
186#ifndef VBOXVDBG_MEMCACHE_DISABLE
187 RTMemCacheFree(pCmdVbva->CtlCache, pCtl);
188#else
189 RTMemFree(pCtl);
190#endif
191}
192
193static VBVAEXHOSTCTL* VBoxVBVAExHCtlCreate(VBVAEXHOSTCONTEXT *pCmdVbva, VBVAEXHOSTCTL_TYPE enmType)
194{
195 VBVAEXHOSTCTL* pCtl = VBoxVBVAExHCtlAlloc(pCmdVbva);
196 if (!pCtl)
197 {
198 WARN(("VBoxVBVAExHCtlAlloc failed\n"));
199 return NULL;
200 }
201
202 pCtl->enmType = enmType;
203 return pCtl;
204}
205
206static int vboxVBVAExHSProcessorAcquire(struct VBVAEXHOSTCONTEXT *pCmdVbva)
207{
208 Assert(pCmdVbva->i32State >= VBVAEXHOSTCONTEXT_STATE_LISTENING);
209
210 if (ASMAtomicCmpXchgS32(&pCmdVbva->i32State, VBVAEXHOSTCONTEXT_STATE_PROCESSING, VBVAEXHOSTCONTEXT_STATE_LISTENING))
211 return VINF_SUCCESS;
212 return VERR_SEM_BUSY;
213}
214
215static VBVAEXHOSTCTL* vboxVBVAExHPCheckCtl(struct VBVAEXHOSTCONTEXT *pCmdVbva, bool *pfHostCtl, bool fHostOnlyMode)
216{
217 Assert(pCmdVbva->i32State == VBVAEXHOSTCONTEXT_STATE_PROCESSING);
218
219 if(!fHostOnlyMode && !ASMAtomicUoReadU32(&pCmdVbva->u32cCtls))
220 return NULL;
221
222 int rc = RTCritSectEnter(&pCmdVbva->CltCritSect);
223 if (RT_SUCCESS(rc))
224 {
225 VBVAEXHOSTCTL* pCtl = RTListGetFirst(&pCmdVbva->HostCtlList, VBVAEXHOSTCTL, Node);
226 if (pCtl)
227 *pfHostCtl = true;
228 else if (!fHostOnlyMode)
229 {
230 if (ASMAtomicUoReadS32(&pCmdVbva->i32EnableState) != VBVAEXHOSTCONTEXT_ESTATE_PAUSED)
231 {
232 pCtl = RTListGetFirst(&pCmdVbva->GuestCtlList, VBVAEXHOSTCTL, Node);
233 /* pCtl can not be null here since pCmdVbva->u32cCtls is not null,
234 * and there are no HostCtl commands*/
235 Assert(pCtl);
236 *pfHostCtl = false;
237 }
238 }
239
240 if (pCtl)
241 {
242 RTListNodeRemove(&pCtl->Node);
243 ASMAtomicDecU32(&pCmdVbva->u32cCtls);
244 }
245
246 RTCritSectLeave(&pCmdVbva->CltCritSect);
247
248 return pCtl;
249 }
250 else
251 WARN(("RTCritSectEnter failed %d\n", rc));
252
253 return NULL;
254}
255
256static VBVAEXHOSTCTL* VBoxVBVAExHPCheckHostCtlOnDisable(struct VBVAEXHOSTCONTEXT *pCmdVbva)
257{
258 bool fHostCtl = false;
259 VBVAEXHOSTCTL* pCtl = vboxVBVAExHPCheckCtl(pCmdVbva, &fHostCtl, true);
260 Assert(!pCtl || fHostCtl);
261 return pCtl;
262}
263
264static int VBoxVBVAExHPPause(struct VBVAEXHOSTCONTEXT *pCmdVbva)
265{
266 if (pCmdVbva->i32EnableState < VBVAEXHOSTCONTEXT_ESTATE_PAUSED)
267 {
268 WARN(("Invalid state\n"));
269 return VERR_INVALID_STATE;
270 }
271
272 ASMAtomicWriteS32(&pCmdVbva->i32EnableState, VBVAEXHOSTCONTEXT_ESTATE_PAUSED);
273 return VINF_SUCCESS;
274}
275
276static int VBoxVBVAExHPResume(struct VBVAEXHOSTCONTEXT *pCmdVbva)
277{
278 if (pCmdVbva->i32EnableState != VBVAEXHOSTCONTEXT_ESTATE_PAUSED)
279 {
280 WARN(("Invalid state\n"));
281 return VERR_INVALID_STATE;
282 }
283
284 ASMAtomicWriteS32(&pCmdVbva->i32EnableState, VBVAEXHOSTCONTEXT_ESTATE_ENABLED);
285 return VINF_SUCCESS;
286}
287
288
289static bool vboxVBVAExHPCheckProcessCtlInternal(struct VBVAEXHOSTCONTEXT *pCmdVbva, VBVAEXHOSTCTL* pCtl)
290{
291 switch (pCtl->enmType)
292 {
293 case VBVAEXHOSTCTL_TYPE_HH_INTERNAL_PAUSE:
294 {
295 int rc = VBoxVBVAExHPPause(pCmdVbva);
296 VBoxVBVAExHPDataCompleteCtl(pCmdVbva, pCtl, VINF_SUCCESS);
297 return true;
298 }
299 case VBVAEXHOSTCTL_TYPE_HH_INTERNAL_RESUME:
300 {
301 int rc = VBoxVBVAExHPResume(pCmdVbva);
302 VBoxVBVAExHPDataCompleteCtl(pCmdVbva, pCtl, VINF_SUCCESS);
303 return true;
304 }
305 default:
306 return false;
307 }
308}
309
310static void vboxVBVAExHPProcessorRelease(struct VBVAEXHOSTCONTEXT *pCmdVbva)
311{
312 Assert(pCmdVbva->i32State == VBVAEXHOSTCONTEXT_STATE_PROCESSING);
313
314 ASMAtomicWriteS32(&pCmdVbva->i32State, VBVAEXHOSTCONTEXT_STATE_LISTENING);
315}
316
317static void vboxVBVAExHPHgEventSet(struct VBVAEXHOSTCONTEXT *pCmdVbva)
318{
319 Assert(pCmdVbva->i32State == VBVAEXHOSTCONTEXT_STATE_PROCESSING);
320 if (pCmdVbva->pVBVA)
321 ASMAtomicOrU32(&pCmdVbva->pVBVA->hostFlags.u32HostEvents, VBVA_F_STATE_PROCESSING);
322}
323
324static void vboxVBVAExHPHgEventClear(struct VBVAEXHOSTCONTEXT *pCmdVbva)
325{
326 Assert(pCmdVbva->i32State == VBVAEXHOSTCONTEXT_STATE_PROCESSING);
327 if (pCmdVbva->pVBVA)
328 ASMAtomicAndU32(&pCmdVbva->pVBVA->hostFlags.u32HostEvents, ~VBVA_F_STATE_PROCESSING);
329}
330
331static int vboxVBVAExHPCmdGet(struct VBVAEXHOSTCONTEXT *pCmdVbva, uint8_t **ppCmd, uint32_t *pcbCmd)
332{
333 Assert(pCmdVbva->i32State == VBVAEXHOSTCONTEXT_STATE_PROCESSING);
334 Assert(pCmdVbva->i32EnableState > VBVAEXHOSTCONTEXT_ESTATE_PAUSED);
335
336 VBVABUFFER *pVBVA = pCmdVbva->pVBVA;
337
338 uint32_t indexRecordFirst = pVBVA->indexRecordFirst;
339 uint32_t indexRecordFree = pVBVA->indexRecordFree;
340
341 Log(("first = %d, free = %d\n",
342 indexRecordFirst, indexRecordFree));
343
344 if (indexRecordFirst == indexRecordFree)
345 {
346 /* No records to process. Return without assigning output variables. */
347 return VINF_EOF;
348 }
349
350 uint32_t cbRecordCurrent = ASMAtomicReadU32(&pVBVA->aRecords[indexRecordFirst].cbRecord);
351
352 /* A new record need to be processed. */
353 if (cbRecordCurrent & VBVA_F_RECORD_PARTIAL)
354 {
355 /* the record is being recorded, try again */
356 return VINF_TRY_AGAIN;
357 }
358
359 uint32_t cbRecord = cbRecordCurrent & ~VBVA_F_RECORD_PARTIAL;
360
361 if (!cbRecord)
362 {
363 /* the record is being recorded, try again */
364 return VINF_TRY_AGAIN;
365 }
366
367 /* we should not get partial commands here actually */
368 Assert(cbRecord);
369
370 /* The size of largest contiguous chunk in the ring biffer. */
371 uint32_t u32BytesTillBoundary = pVBVA->cbData - pVBVA->off32Data;
372
373 /* The pointer to data in the ring buffer. */
374 uint8_t *pSrc = &pVBVA->au8Data[pVBVA->off32Data];
375
376 /* Fetch or point the data. */
377 if (u32BytesTillBoundary >= cbRecord)
378 {
379 /* The command does not cross buffer boundary. Return address in the buffer. */
380 *ppCmd = pSrc;
381 *pcbCmd = cbRecord;
382 return VINF_SUCCESS;
383 }
384
385 LogRel(("CmdVbva: cross-bound writes unsupported\n"));
386 return VERR_INVALID_STATE;
387}
388
389static void VBoxVBVAExHPDataCompleteCmd(struct VBVAEXHOSTCONTEXT *pCmdVbva, uint32_t cbCmd)
390{
391 VBVABUFFER *pVBVA = pCmdVbva->pVBVA;
392 pVBVA->off32Data = (pVBVA->off32Data + cbCmd) % pVBVA->cbData;
393
394 pVBVA->indexRecordFirst = (pVBVA->indexRecordFirst + 1) % RT_ELEMENTS(pVBVA->aRecords);
395}
396
397static void VBoxVBVAExHPDataCompleteCtl(struct VBVAEXHOSTCONTEXT *pCmdVbva, VBVAEXHOSTCTL *pCtl, int rc)
398{
399 if (pCtl->pfnComplete)
400 pCtl->pfnComplete(pCmdVbva, pCtl, rc, pCtl->pvComplete);
401 else
402 VBoxVBVAExHCtlFree(pCmdVbva, pCtl);
403}
404
405static VBVAEXHOST_DATA_TYPE vboxVBVAExHPDataGet(struct VBVAEXHOSTCONTEXT *pCmdVbva, uint8_t **ppCmd, uint32_t *pcbCmd)
406{
407 Assert(pCmdVbva->i32State == VBVAEXHOSTCONTEXT_STATE_PROCESSING);
408 VBVAEXHOSTCTL*pCtl;
409 bool fHostClt;
410
411 for(;;)
412 {
413 pCtl = vboxVBVAExHPCheckCtl(pCmdVbva, &fHostClt, false);
414 if (pCtl)
415 {
416 if (fHostClt)
417 {
418 if (!vboxVBVAExHPCheckProcessCtlInternal(pCmdVbva, pCtl))
419 {
420 *ppCmd = (uint8_t*)pCtl;
421 *pcbCmd = sizeof (*pCtl);
422 return VBVAEXHOST_DATA_TYPE_HOSTCTL;
423 }
424 continue;
425 }
426 else
427 {
428 *ppCmd = (uint8_t*)pCtl;
429 *pcbCmd = sizeof (*pCtl);
430 return VBVAEXHOST_DATA_TYPE_GUESTCTL;
431 }
432 }
433
434 if (ASMAtomicUoReadS32(&pCmdVbva->i32EnableState) <= VBVAEXHOSTCONTEXT_ESTATE_PAUSED)
435 return VBVAEXHOST_DATA_TYPE_NO_DATA;
436
437 int rc = vboxVBVAExHPCmdGet(pCmdVbva, ppCmd, pcbCmd);
438 switch (rc)
439 {
440 case VINF_SUCCESS:
441 return VBVAEXHOST_DATA_TYPE_CMD;
442 case VINF_EOF:
443 return VBVAEXHOST_DATA_TYPE_NO_DATA;
444 case VINF_TRY_AGAIN:
445 RTThreadSleep(1);
446 continue;
447 default:
448 /* this is something really unexpected, i.e. most likely guest has written something incorrect to the VBVA buffer */
449 WARN(("Warning: vboxVBVAExHCmdGet returned unexpected status %d\n", rc));
450 return VBVAEXHOST_DATA_TYPE_NO_DATA;
451 }
452 }
453
454 WARN(("Warning: VBoxVBVAExHCmdGet unexpected state\n"));
455 return VBVAEXHOST_DATA_TYPE_NO_DATA;
456}
457
458static VBVAEXHOST_DATA_TYPE VBoxVBVAExHPDataGet(struct VBVAEXHOSTCONTEXT *pCmdVbva, uint8_t **ppCmd, uint32_t *pcbCmd)
459{
460 VBVAEXHOST_DATA_TYPE enmType = vboxVBVAExHPDataGet(pCmdVbva, ppCmd, pcbCmd);
461 if (enmType == VBVAEXHOST_DATA_TYPE_NO_DATA)
462 {
463 vboxVBVAExHPHgEventClear(pCmdVbva);
464 vboxVBVAExHPProcessorRelease(pCmdVbva);
465 /* we need to prevent racing between us clearing the flag and command check/submission thread, i.e.
466 * 1. we check the queue -> and it is empty
467 * 2. submitter adds command to the queue
468 * 3. submitter checks the "processing" -> and it is true , thus it does not submit a notification
469 * 4. we clear the "processing" state
470 * 5. ->here we need to re-check the queue state to ensure we do not leak the notification of the above command
471 * 6. if the queue appears to be not-empty set the "processing" state back to "true"
472 **/
473 int rc = vboxVBVAExHSProcessorAcquire(pCmdVbva);
474 if (RT_SUCCESS(rc))
475 {
476 /* we are the processor now */
477 enmType = vboxVBVAExHPDataGet(pCmdVbva, ppCmd, pcbCmd);
478 if (enmType == VBVAEXHOST_DATA_TYPE_NO_DATA)
479 {
480 vboxVBVAExHPProcessorRelease(pCmdVbva);
481 return VBVAEXHOST_DATA_TYPE_NO_DATA;
482 }
483
484 vboxVBVAExHPHgEventSet(pCmdVbva);
485 }
486 }
487
488 return enmType;
489}
490
491DECLINLINE(bool) vboxVBVAExHSHasCommands(struct VBVAEXHOSTCONTEXT *pCmdVbva)
492{
493 VBVABUFFER *pVBVA = pCmdVbva->pVBVA;
494
495 if (pVBVA)
496 {
497 uint32_t indexRecordFirst = pVBVA->indexRecordFirst;
498 uint32_t indexRecordFree = pVBVA->indexRecordFree;
499
500 if (indexRecordFirst != indexRecordFree)
501 return true;
502 }
503
504 return !!ASMAtomicReadU32(&pCmdVbva->u32cCtls);
505}
506
507/* Checks whether the new commands are ready for processing
508 * @returns
509 * VINF_SUCCESS - there are commands are in a queue, and the given thread is now the processor (i.e. typically it would delegate processing to a worker thread)
510 * VINF_EOF - no commands in a queue
511 * VINF_ALREADY_INITIALIZED - another thread already processing the commands
512 * VERR_INVALID_STATE - the VBVA is paused or pausing */
513static int VBoxVBVAExHSCheckCommands(struct VBVAEXHOSTCONTEXT *pCmdVbva)
514{
515 int rc = vboxVBVAExHSProcessorAcquire(pCmdVbva);
516 if (RT_SUCCESS(rc))
517 {
518 /* we are the processor now */
519 if (vboxVBVAExHSHasCommands(pCmdVbva))
520 {
521 vboxVBVAExHPHgEventSet(pCmdVbva);
522 return VINF_SUCCESS;
523 }
524
525 vboxVBVAExHPProcessorRelease(pCmdVbva);
526 return VINF_EOF;
527 }
528 if (rc == VERR_SEM_BUSY)
529 return VINF_ALREADY_INITIALIZED;
530 return VERR_INVALID_STATE;
531}
532
533static int VBoxVBVAExHSInit(struct VBVAEXHOSTCONTEXT *pCmdVbva)
534{
535 memset(pCmdVbva, 0, sizeof (*pCmdVbva));
536 int rc = RTCritSectInit(&pCmdVbva->CltCritSect);
537 if (RT_SUCCESS(rc))
538 {
539#ifndef VBOXVDBG_MEMCACHE_DISABLE
540 rc = RTMemCacheCreate(&pCmdVbva->CtlCache, sizeof (VBVAEXHOSTCTL),
541 0, /* size_t cbAlignment */
542 UINT32_MAX, /* uint32_t cMaxObjects */
543 NULL, /* PFNMEMCACHECTOR pfnCtor*/
544 NULL, /* PFNMEMCACHEDTOR pfnDtor*/
545 NULL, /* void *pvUser*/
546 0 /* uint32_t fFlags*/
547 );
548 if (RT_SUCCESS(rc))
549#endif
550 {
551 RTListInit(&pCmdVbva->GuestCtlList);
552 RTListInit(&pCmdVbva->HostCtlList);
553 pCmdVbva->i32State = VBVAEXHOSTCONTEXT_STATE_PROCESSING;
554 pCmdVbva->i32EnableState = VBVAEXHOSTCONTEXT_ESTATE_DISABLED;
555 return VINF_SUCCESS;
556 }
557#ifndef VBOXVDBG_MEMCACHE_DISABLE
558 else
559 WARN(("RTMemCacheCreate failed %d\n", rc));
560#endif
561 }
562 else
563 WARN(("RTCritSectInit failed %d\n", rc));
564
565 return rc;
566}
567
568DECLINLINE(bool) VBoxVBVAExHSIsEnabled(struct VBVAEXHOSTCONTEXT *pCmdVbva)
569{
570 return (ASMAtomicUoReadS32(&pCmdVbva->i32EnableState) >= VBVAEXHOSTCONTEXT_ESTATE_PAUSED);
571}
572
573DECLINLINE(bool) VBoxVBVAExHSIsDisabled(struct VBVAEXHOSTCONTEXT *pCmdVbva)
574{
575 return (ASMAtomicUoReadS32(&pCmdVbva->i32EnableState) == VBVAEXHOSTCONTEXT_ESTATE_DISABLED);
576}
577
578static int VBoxVBVAExHSEnable(struct VBVAEXHOSTCONTEXT *pCmdVbva, VBVABUFFER *pVBVA)
579{
580 if (VBoxVBVAExHSIsEnabled(pCmdVbva))
581 {
582 WARN(("VBVAEx is enabled already\n"));
583 return VERR_INVALID_STATE;
584 }
585
586 pCmdVbva->pVBVA = pVBVA;
587 pCmdVbva->pVBVA->hostFlags.u32HostEvents = 0;
588 ASMAtomicWriteS32(&pCmdVbva->i32EnableState, VBVAEXHOSTCONTEXT_ESTATE_ENABLED);
589 return VINF_SUCCESS;
590}
591
592static int VBoxVBVAExHSDisable(struct VBVAEXHOSTCONTEXT *pCmdVbva)
593{
594 if (VBoxVBVAExHSIsDisabled(pCmdVbva))
595 return VINF_SUCCESS;
596
597 ASMAtomicWriteS32(&pCmdVbva->i32EnableState, VBVAEXHOSTCONTEXT_ESTATE_DISABLED);
598 return VINF_SUCCESS;
599}
600
601static void VBoxVBVAExHSTerm(struct VBVAEXHOSTCONTEXT *pCmdVbva)
602{
603 /* ensure the processor is stopped */
604 Assert(pCmdVbva->i32State >= VBVAEXHOSTCONTEXT_STATE_LISTENING);
605
606 /* ensure no one tries to submit the command */
607 if (pCmdVbva->pVBVA)
608 pCmdVbva->pVBVA->hostFlags.u32HostEvents = 0;
609
610 Assert(RTListIsEmpty(&pCmdVbva->GuestCtlList));
611 Assert(RTListIsEmpty(&pCmdVbva->HostCtlList));
612
613 RTCritSectDelete(&pCmdVbva->CltCritSect);
614
615#ifndef VBOXVDBG_MEMCACHE_DISABLE
616 RTMemCacheDestroy(pCmdVbva->CtlCache);
617#endif
618
619 memset(pCmdVbva, 0, sizeof (*pCmdVbva));
620}
621
622static int vboxVBVAExHSSaveGuestCtl(struct VBVAEXHOSTCONTEXT *pCmdVbva, VBVAEXHOSTCTL* pCtl, uint8_t* pu8VramBase, PSSMHANDLE pSSM)
623{
624 int rc = SSMR3PutU32(pSSM, pCtl->enmType);
625 AssertRCReturn(rc, rc);
626 rc = SSMR3PutU32(pSSM, pCtl->u.cmd.cbCmd);
627 AssertRCReturn(rc, rc);
628 rc = SSMR3PutU32(pSSM, (uint32_t)(pCtl->u.cmd.pu8Cmd - pu8VramBase));
629 AssertRCReturn(rc, rc);
630
631 return VINF_SUCCESS;
632}
633
634static int vboxVBVAExHSSaveStateLocked(struct VBVAEXHOSTCONTEXT *pCmdVbva, uint8_t* pu8VramBase, PSSMHANDLE pSSM)
635{
636 if (ASMAtomicUoReadS32(&pCmdVbva->i32EnableState) != VBVAEXHOSTCONTEXT_ESTATE_PAUSED)
637 {
638 WARN(("vbva not paused\n"));
639 return VERR_INVALID_STATE;
640 }
641
642 VBVAEXHOSTCTL* pCtl;
643 int rc;
644 RTListForEach(&pCmdVbva->GuestCtlList, pCtl, VBVAEXHOSTCTL, Node)
645 {
646 rc = vboxVBVAExHSSaveGuestCtl(pCmdVbva, pCtl, pu8VramBase, pSSM);
647 AssertRCReturn(rc, rc);
648 }
649
650 rc = SSMR3PutU32(pSSM, 0);
651 AssertRCReturn(rc, rc);
652
653 return VINF_SUCCESS;
654}
655/* Saves state
656 * @returns - same as VBoxVBVAExHSCheckCommands, or failure on load state fail
657 */
658static int VBoxVBVAExHSSaveState(struct VBVAEXHOSTCONTEXT *pCmdVbva, uint8_t* pu8VramBase, PSSMHANDLE pSSM)
659{
660 int rc = RTCritSectEnter(&pCmdVbva->CltCritSect);
661 if (RT_FAILURE(rc))
662 {
663 WARN(("RTCritSectEnter failed %d\n", rc));
664 return rc;
665 }
666
667 rc = vboxVBVAExHSSaveStateLocked(pCmdVbva, pu8VramBase, pSSM);
668 if (RT_FAILURE(rc))
669 WARN(("vboxVBVAExHSSaveStateLocked failed %d\n", rc));
670
671 RTCritSectLeave(&pCmdVbva->CltCritSect);
672
673 return rc;
674}
675
676static int vboxVBVAExHSLoadGuestCtl(struct VBVAEXHOSTCONTEXT *pCmdVbva, uint8_t* pu8VramBase, PSSMHANDLE pSSM, uint32_t u32Version)
677{
678 uint32_t u32;
679 int rc = SSMR3GetU32(pSSM, &u32);
680 AssertRCReturn(rc, rc);
681
682 if (!u32)
683 return VINF_EOF;
684
685 VBVAEXHOSTCTL* pHCtl = VBoxVBVAExHCtlCreate(pCmdVbva, (VBVAEXHOSTCTL_TYPE)u32);
686 if (!pHCtl)
687 {
688 WARN(("VBoxVBVAExHCtlCreate failed\n"));
689 return VERR_NO_MEMORY;
690 }
691
692 rc = SSMR3GetU32(pSSM, &u32);
693 AssertRCReturn(rc, rc);
694 pHCtl->u.cmd.cbCmd = u32;
695
696 rc = SSMR3GetU32(pSSM, &u32);
697 AssertRCReturn(rc, rc);
698 pHCtl->u.cmd.pu8Cmd = pu8VramBase + u32;
699
700 RTListAppend(&pCmdVbva->GuestCtlList, &pHCtl->Node);
701 ++pCmdVbva->u32cCtls;
702
703 return VINF_SUCCESS;
704}
705
706
707static int vboxVBVAExHSLoadStateLocked(struct VBVAEXHOSTCONTEXT *pCmdVbva, uint8_t* pu8VramBase, PSSMHANDLE pSSM, uint32_t u32Version)
708{
709 if (ASMAtomicUoReadS32(&pCmdVbva->i32EnableState) != VBVAEXHOSTCONTEXT_ESTATE_PAUSED)
710 {
711 WARN(("vbva not stopped\n"));
712 return VERR_INVALID_STATE;
713 }
714
715 int rc;
716
717 do {
718 rc = vboxVBVAExHSLoadGuestCtl(pCmdVbva, pu8VramBase, pSSM, u32Version);
719 AssertRCReturn(rc, rc);
720 } while (VINF_EOF != rc);
721
722 return VINF_SUCCESS;
723}
724
725/* Loads state
726 * @returns - same as VBoxVBVAExHSCheckCommands, or failure on load state fail
727 */
728static int VBoxVBVAExHSLoadState(struct VBVAEXHOSTCONTEXT *pCmdVbva, uint8_t* pu8VramBase, PSSMHANDLE pSSM, uint32_t u32Version)
729{
730 Assert(VGA_SAVEDSTATE_VERSION_3D <= u32Version);
731 int rc = RTCritSectEnter(&pCmdVbva->CltCritSect);
732 if (RT_FAILURE(rc))
733 {
734 WARN(("RTCritSectEnter failed %d\n", rc));
735 return rc;
736 }
737
738 rc = vboxVBVAExHSLoadStateLocked(pCmdVbva, pu8VramBase, pSSM, u32Version);
739 if (RT_FAILURE(rc))
740 WARN(("vboxVBVAExHSSaveStateLocked failed %d\n", rc));
741
742 RTCritSectLeave(&pCmdVbva->CltCritSect);
743
744 return rc;
745}
746
747typedef enum
748{
749 VBVAEXHOSTCTL_SOURCE_GUEST = 0,
750 VBVAEXHOSTCTL_SOURCE_HOST
751} VBVAEXHOSTCTL_SOURCE;
752
753
754static int VBoxVBVAExHCtlSubmit(VBVAEXHOSTCONTEXT *pCmdVbva, VBVAEXHOSTCTL* pCtl, VBVAEXHOSTCTL_SOURCE enmSource, PFNVBVAEXHOSTCTL_COMPLETE pfnComplete, void *pvComplete)
755{
756 if (!VBoxVBVAExHSIsEnabled(pCmdVbva))
757 {
758 Log(("cmd vbva not enabled\n"));
759 return VERR_INVALID_STATE;
760 }
761
762 pCtl->pfnComplete = pfnComplete;
763 pCtl->pvComplete = pvComplete;
764
765 int rc = RTCritSectEnter(&pCmdVbva->CltCritSect);
766 if (RT_SUCCESS(rc))
767 {
768 if (!VBoxVBVAExHSIsEnabled(pCmdVbva))
769 {
770 Log(("cmd vbva not enabled\n"));
771 RTCritSectLeave(&pCmdVbva->CltCritSect);
772 return VERR_INVALID_STATE;
773 }
774
775 if (enmSource > VBVAEXHOSTCTL_SOURCE_GUEST)
776 {
777 RTListAppend(&pCmdVbva->HostCtlList, &pCtl->Node);
778 }
779 else
780 RTListAppend(&pCmdVbva->GuestCtlList, &pCtl->Node);
781
782 ASMAtomicIncU32(&pCmdVbva->u32cCtls);
783
784 RTCritSectLeave(&pCmdVbva->CltCritSect);
785
786 rc = VBoxVBVAExHSCheckCommands(pCmdVbva);
787 }
788 else
789 WARN(("RTCritSectEnter failed %d\n", rc));
790
791 return rc;
792}
793
794#ifdef VBOX_WITH_CRHGSMI
795typedef struct VBOXVDMA_SOURCE
796{
797 VBVAINFOSCREEN Screen;
798 VBOXCMDVBVA_SCREENMAP_DECL(uint32_t, aTargetMap);
799} VBOXVDMA_SOURCE;
800#endif
801
802typedef struct VBOXVDMAHOST
803{
804 PHGSMIINSTANCE pHgsmi;
805 PVGASTATE pVGAState;
806#ifdef VBOX_WITH_CRHGSMI
807 VBVAEXHOSTCONTEXT CmdVbva;
808 VBOXVDMATHREAD Thread;
809 VBOXCRCMD_SVRINFO CrSrvInfo;
810 VBVAEXHOSTCTL* pCurRemainingHostCtl;
811 RTSEMEVENTMULTI HostCrCtlCompleteEvent;
812 int32_t volatile i32cHostCrCtlCompleted;
813 RTCRITSECT CalloutCritSect;
814// VBOXVDMA_SOURCE aSources[VBOX_VIDEO_MAX_SCREENS];
815#endif
816#ifdef VBOX_VDMA_WITH_WATCHDOG
817 PTMTIMERR3 WatchDogTimer;
818#endif
819} VBOXVDMAHOST, *PVBOXVDMAHOST;
820
821#ifdef VBOX_WITH_CRHGSMI
822
823void VBoxVDMAThreadNotifyConstructSucceeded(PVBOXVDMATHREAD pThread, void *pvThreadContext)
824{
825 Assert(pThread->u32State == VBOXVDMATHREAD_STATE_CREATING);
826 PFNVBOXVDMATHREAD_CHANGED pfnChanged = pThread->pfnChanged;
827 void *pvChanged = pThread->pvChanged;
828
829 pThread->pfnChanged = NULL;
830 pThread->pvChanged = NULL;
831
832 ASMAtomicWriteU32(&pThread->u32State, VBOXVDMATHREAD_STATE_CREATED);
833
834 if (pfnChanged)
835 pfnChanged(pThread, VINF_SUCCESS, pvThreadContext, pvChanged);
836}
837
838void VBoxVDMAThreadNotifyTerminatingSucceeded(PVBOXVDMATHREAD pThread, void *pvThreadContext)
839{
840 Assert(pThread->u32State == VBOXVDMATHREAD_STATE_TERMINATING);
841 PFNVBOXVDMATHREAD_CHANGED pfnChanged = pThread->pfnChanged;
842 void *pvChanged = pThread->pvChanged;
843
844 pThread->pfnChanged = NULL;
845 pThread->pvChanged = NULL;
846
847 if (pfnChanged)
848 pfnChanged(pThread, VINF_SUCCESS, pvThreadContext, pvChanged);
849}
850
851DECLINLINE(bool) VBoxVDMAThreadIsTerminating(PVBOXVDMATHREAD pThread)
852{
853 return ASMAtomicUoReadU32(&pThread->u32State) == VBOXVDMATHREAD_STATE_TERMINATING;
854}
855
856void VBoxVDMAThreadInit(PVBOXVDMATHREAD pThread)
857{
858 memset(pThread, 0, sizeof (*pThread));
859 pThread->u32State = VBOXVDMATHREAD_STATE_TERMINATED;
860}
861
862int VBoxVDMAThreadCleanup(PVBOXVDMATHREAD pThread)
863{
864 uint32_t u32State = ASMAtomicUoReadU32(&pThread->u32State);
865 switch (u32State)
866 {
867 case VBOXVDMATHREAD_STATE_TERMINATED:
868 return VINF_SUCCESS;
869 case VBOXVDMATHREAD_STATE_TERMINATING:
870 {
871 int rc = RTThreadWait(pThread->hWorkerThread, RT_INDEFINITE_WAIT, NULL);
872 if (!RT_SUCCESS(rc))
873 {
874 WARN(("RTThreadWait failed %d\n", rc));
875 return rc;
876 }
877
878 RTSemEventDestroy(pThread->hEvent);
879
880 ASMAtomicWriteU32(&pThread->u32State, VBOXVDMATHREAD_STATE_TERMINATED);
881 return VINF_SUCCESS;
882 }
883 default:
884 WARN(("invalid state"));
885 return VERR_INVALID_STATE;
886 }
887}
888
889int VBoxVDMAThreadCreate(PVBOXVDMATHREAD pThread, PFNRTTHREAD pfnThread, void *pvThread, PFNVBOXVDMATHREAD_CHANGED pfnCreated, void*pvCreated)
890{
891 int rc = VBoxVDMAThreadCleanup(pThread);
892 if (RT_FAILURE(rc))
893 {
894 WARN(("VBoxVDMAThreadCleanup failed %d\n", rc));
895 return rc;
896 }
897
898 rc = RTSemEventCreate(&pThread->hEvent);
899 if (RT_SUCCESS(rc))
900 {
901 pThread->u32State = VBOXVDMATHREAD_STATE_CREATING;
902 pThread->pfnChanged = pfnCreated;
903 pThread->pvChanged = pvCreated;
904 rc = RTThreadCreate(&pThread->hWorkerThread, pfnThread, pvThread, 0, RTTHREADTYPE_IO, RTTHREADFLAGS_WAITABLE, "VDMA");
905 if (RT_SUCCESS(rc))
906 return VINF_SUCCESS;
907 else
908 WARN(("RTThreadCreate failed %d\n", rc));
909
910 RTSemEventDestroy(pThread->hEvent);
911 }
912 else
913 WARN(("RTSemEventCreate failed %d\n", rc));
914
915 pThread->u32State = VBOXVDMATHREAD_STATE_TERMINATED;
916
917 return rc;
918}
919
920DECLINLINE(int) VBoxVDMAThreadEventNotify(PVBOXVDMATHREAD pThread)
921{
922 int rc = RTSemEventSignal(pThread->hEvent);
923 AssertRC(rc);
924 return rc;
925}
926
927DECLINLINE(int) VBoxVDMAThreadEventWait(PVBOXVDMATHREAD pThread, RTMSINTERVAL cMillies)
928{
929 int rc = RTSemEventWait(pThread->hEvent, cMillies);
930 AssertRC(rc);
931 return rc;
932}
933
934int VBoxVDMAThreadTerm(PVBOXVDMATHREAD pThread, PFNVBOXVDMATHREAD_CHANGED pfnTerminated, void*pvTerminated, bool fNotify)
935{
936 int rc;
937 do
938 {
939 uint32_t u32State = ASMAtomicUoReadU32(&pThread->u32State);
940 switch (u32State)
941 {
942 case VBOXVDMATHREAD_STATE_CREATED:
943 pThread->pfnChanged = pfnTerminated;
944 pThread->pvChanged = pvTerminated;
945 ASMAtomicWriteU32(&pThread->u32State, VBOXVDMATHREAD_STATE_TERMINATING);
946 if (fNotify)
947 {
948 rc = VBoxVDMAThreadEventNotify(pThread);
949 AssertRC(rc);
950 }
951 return VINF_SUCCESS;
952 case VBOXVDMATHREAD_STATE_TERMINATING:
953 case VBOXVDMATHREAD_STATE_TERMINATED:
954 {
955 WARN(("thread is marked to termination or terminated\nn"));
956 return VERR_INVALID_STATE;
957 }
958 case VBOXVDMATHREAD_STATE_CREATING:
959 {
960 /* wait till the thread creation is completed */
961 WARN(("concurrent thread create/destron\n"));
962 RTThreadYield();
963 continue;
964 }
965 default:
966 WARN(("invalid state"));
967 return VERR_INVALID_STATE;
968 }
969 } while (1);
970
971 WARN(("should never be here\n"));
972 return VERR_INTERNAL_ERROR;
973}
974
975static int vdmaVBVACtlSubmitSync(PVBOXVDMAHOST pVdma, VBVAEXHOSTCTL* pCtl, VBVAEXHOSTCTL_SOURCE enmSource);
976
977typedef DECLCALLBACK(void) FNVBOXVDMACRCTL_CALLBACK(PVGASTATE pVGAState, PVBOXVDMACMD_CHROMIUM_CTL pCmd, void* pvContext);
978typedef FNVBOXVDMACRCTL_CALLBACK *PFNVBOXVDMACRCTL_CALLBACK;
979
980typedef struct VBOXVDMACMD_CHROMIUM_CTL_PRIVATE
981{
982 uint32_t cRefs;
983 int32_t rc;
984 PFNVBOXVDMACRCTL_CALLBACK pfnCompletion;
985 void *pvCompletion;
986 VBOXVDMACMD_CHROMIUM_CTL Cmd;
987} VBOXVDMACMD_CHROMIUM_CTL_PRIVATE, *PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE;
988
989#define VBOXVDMACMD_CHROMIUM_CTL_PRIVATE_FROM_CTL(_p) ((PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE)(((uint8_t*)(_p)) - RT_OFFSETOF(VBOXVDMACMD_CHROMIUM_CTL_PRIVATE, Cmd)))
990
991static PVBOXVDMACMD_CHROMIUM_CTL vboxVDMACrCtlCreate(VBOXVDMACMD_CHROMIUM_CTL_TYPE enmCmd, uint32_t cbCmd)
992{
993 PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE pHdr = (PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE)RTMemAllocZ(cbCmd + RT_OFFSETOF(VBOXVDMACMD_CHROMIUM_CTL_PRIVATE, Cmd));
994 Assert(pHdr);
995 if (pHdr)
996 {
997 pHdr->cRefs = 1;
998 pHdr->rc = VERR_NOT_IMPLEMENTED;
999 pHdr->Cmd.enmType = enmCmd;
1000 pHdr->Cmd.cbCmd = cbCmd;
1001 return &pHdr->Cmd;
1002 }
1003
1004 return NULL;
1005}
1006
1007DECLINLINE(void) vboxVDMACrCtlRelease (PVBOXVDMACMD_CHROMIUM_CTL pCmd)
1008{
1009 PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE pHdr = VBOXVDMACMD_CHROMIUM_CTL_PRIVATE_FROM_CTL(pCmd);
1010 uint32_t cRefs = ASMAtomicDecU32(&pHdr->cRefs);
1011 if(!cRefs)
1012 {
1013 RTMemFree(pHdr);
1014 }
1015}
1016
1017DECLINLINE(void) vboxVDMACrCtlRetain (PVBOXVDMACMD_CHROMIUM_CTL pCmd)
1018{
1019 PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE pHdr = VBOXVDMACMD_CHROMIUM_CTL_PRIVATE_FROM_CTL(pCmd);
1020 ASMAtomicIncU32(&pHdr->cRefs);
1021}
1022
1023DECLINLINE(int) vboxVDMACrCtlGetRc (PVBOXVDMACMD_CHROMIUM_CTL pCmd)
1024{
1025 PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE pHdr = VBOXVDMACMD_CHROMIUM_CTL_PRIVATE_FROM_CTL(pCmd);
1026 return pHdr->rc;
1027}
1028
1029static DECLCALLBACK(void) vboxVDMACrCtlCbSetEvent(PVGASTATE pVGAState, PVBOXVDMACMD_CHROMIUM_CTL pCmd, void* pvContext)
1030{
1031 RTSemEventSignal((RTSEMEVENT)pvContext);
1032}
1033
1034static DECLCALLBACK(void) vboxVDMACrCtlCbReleaseCmd(PVGASTATE pVGAState, PVBOXVDMACMD_CHROMIUM_CTL pCmd, void* pvContext)
1035{
1036 vboxVDMACrCtlRelease(pCmd);
1037}
1038
1039
1040static int vboxVDMACrCtlPostAsync (PVGASTATE pVGAState, PVBOXVDMACMD_CHROMIUM_CTL pCmd, uint32_t cbCmd, PFNVBOXVDMACRCTL_CALLBACK pfnCompletion, void *pvCompletion)
1041{
1042 if ( pVGAState->pDrv
1043 && pVGAState->pDrv->pfnCrHgsmiControlProcess)
1044 {
1045 PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE pHdr = VBOXVDMACMD_CHROMIUM_CTL_PRIVATE_FROM_CTL(pCmd);
1046 pHdr->pfnCompletion = pfnCompletion;
1047 pHdr->pvCompletion = pvCompletion;
1048 pVGAState->pDrv->pfnCrHgsmiControlProcess(pVGAState->pDrv, pCmd, cbCmd);
1049 return VINF_SUCCESS;
1050 }
1051#ifdef DEBUG_misha
1052 Assert(0);
1053#endif
1054 return VERR_NOT_SUPPORTED;
1055}
1056
1057static int vboxVDMACrCtlPost(PVGASTATE pVGAState, PVBOXVDMACMD_CHROMIUM_CTL pCmd, uint32_t cbCmd)
1058{
1059 RTSEMEVENT hComplEvent;
1060 int rc = RTSemEventCreate(&hComplEvent);
1061 AssertRC(rc);
1062 if(RT_SUCCESS(rc))
1063 {
1064 rc = vboxVDMACrCtlPostAsync(pVGAState, pCmd, cbCmd, vboxVDMACrCtlCbSetEvent, (void*)hComplEvent);
1065#ifdef DEBUG_misha
1066 AssertRC(rc);
1067#endif
1068 if (RT_SUCCESS(rc))
1069 {
1070 rc = RTSemEventWaitNoResume(hComplEvent, RT_INDEFINITE_WAIT);
1071 AssertRC(rc);
1072 if(RT_SUCCESS(rc))
1073 {
1074 RTSemEventDestroy(hComplEvent);
1075 }
1076 }
1077 else
1078 {
1079 /* the command is completed */
1080 RTSemEventDestroy(hComplEvent);
1081 }
1082 }
1083 return rc;
1084}
1085
1086typedef struct VDMA_VBVA_CTL_CYNC_COMPLETION
1087{
1088 int rc;
1089 RTSEMEVENT hEvent;
1090} VDMA_VBVA_CTL_CYNC_COMPLETION;
1091
1092static DECLCALLBACK(void) vboxVDMACrHgcmSubmitSyncCompletion(struct VBOXCRCMDCTL* pCmd, uint32_t cbCmd, int rc, void *pvCompletion)
1093{
1094 VDMA_VBVA_CTL_CYNC_COMPLETION *pData = (VDMA_VBVA_CTL_CYNC_COMPLETION*)pvCompletion;
1095 pData->rc = rc;
1096 rc = RTSemEventSignal(pData->hEvent);
1097 if (!RT_SUCCESS(rc))
1098 WARN(("RTSemEventSignal failed %d\n", rc));
1099}
1100
1101static int vboxVDMACrHgcmSubmitSync(struct VBOXVDMAHOST *pVdma, VBOXCRCMDCTL* pCtl, uint32_t cbCtl)
1102{
1103 VDMA_VBVA_CTL_CYNC_COMPLETION Data;
1104 Data.rc = VERR_NOT_IMPLEMENTED;
1105 int rc = RTSemEventCreate(&Data.hEvent);
1106 if (!RT_SUCCESS(rc))
1107 {
1108 WARN(("RTSemEventCreate failed %d\n", rc));
1109 return rc;
1110 }
1111
1112 pCtl->CalloutList.List.pNext = NULL;
1113
1114 PVGASTATE pVGAState = pVdma->pVGAState;
1115 rc = pVGAState->pDrv->pfnCrHgcmCtlSubmit(pVGAState->pDrv, pCtl, cbCtl, vboxVDMACrHgcmSubmitSyncCompletion, &Data);
1116 if (RT_SUCCESS(rc))
1117 {
1118 rc = RTSemEventWait(Data.hEvent, RT_INDEFINITE_WAIT);
1119 if (RT_SUCCESS(rc))
1120 {
1121 rc = Data.rc;
1122 if (!RT_SUCCESS(rc))
1123 {
1124 WARN(("pfnCrHgcmCtlSubmit command failed %d\n", rc));
1125 }
1126
1127 }
1128 else
1129 WARN(("RTSemEventWait failed %d\n", rc));
1130 }
1131 else
1132 WARN(("pfnCrHgcmCtlSubmit failed %d\n", rc));
1133
1134
1135 RTSemEventDestroy(Data.hEvent);
1136
1137 return rc;
1138}
1139
1140static int vdmaVBVACtlDisableSync(PVBOXVDMAHOST pVdma)
1141{
1142 VBVAEXHOSTCTL HCtl;
1143 HCtl.enmType = VBVAEXHOSTCTL_TYPE_GHH_DISABLE;
1144 int rc = vdmaVBVACtlSubmitSync(pVdma, &HCtl, VBVAEXHOSTCTL_SOURCE_HOST);
1145 if (RT_FAILURE(rc))
1146 {
1147 Log(("vdmaVBVACtlSubmitSync failed %d\n", rc));
1148 return rc;
1149 }
1150
1151 vgaUpdateDisplayAll(pVdma->pVGAState, /* fFailOnResize = */ false);
1152
1153 return VINF_SUCCESS;
1154}
1155
1156static DECLCALLBACK(uint8_t*) vboxVDMACrHgcmHandleEnableRemainingHostCommand(HVBOXCRCMDCTL_REMAINING_HOST_COMMAND hClient, uint32_t *pcbCtl, int prevCmdRc)
1157{
1158 struct VBOXVDMAHOST *pVdma = hClient;
1159 if (!pVdma->pCurRemainingHostCtl)
1160 {
1161 /* disable VBVA, all subsequent host commands will go HGCM way */
1162 VBoxVBVAExHSDisable(&pVdma->CmdVbva);
1163 }
1164 else
1165 {
1166 VBoxVBVAExHPDataCompleteCtl(&pVdma->CmdVbva, pVdma->pCurRemainingHostCtl, prevCmdRc);
1167 }
1168
1169 pVdma->pCurRemainingHostCtl = VBoxVBVAExHPCheckHostCtlOnDisable(&pVdma->CmdVbva);
1170 if (pVdma->pCurRemainingHostCtl)
1171 {
1172 *pcbCtl = pVdma->pCurRemainingHostCtl->u.cmd.cbCmd;
1173 return pVdma->pCurRemainingHostCtl->u.cmd.pu8Cmd;
1174 }
1175
1176 *pcbCtl = 0;
1177 return NULL;
1178}
1179
1180static DECLCALLBACK(void) vboxVDMACrHgcmNotifyTerminatingDoneCb(HVBOXCRCMDCTL_NOTIFY_TERMINATING hClient)
1181{
1182 struct VBOXVDMAHOST *pVdma = hClient;
1183 Assert(pVdma->CmdVbva.i32State == VBVAEXHOSTCONTEXT_STATE_PROCESSING);
1184 Assert(pVdma->Thread.u32State == VBOXVDMATHREAD_STATE_TERMINATING);
1185}
1186
1187static DECLCALLBACK(int) vboxVDMACrHgcmNotifyTerminatingCb(HVBOXCRCMDCTL_NOTIFY_TERMINATING hClient, VBOXCRCMDCTL_HGCMENABLE_DATA *pHgcmEnableData)
1188{
1189 struct VBOXVDMAHOST *pVdma = hClient;
1190 VBVAEXHOSTCTL HCtl;
1191 HCtl.enmType = VBVAEXHOSTCTL_TYPE_HH_ON_HGCM_UNLOAD;
1192 int rc = vdmaVBVACtlSubmitSync(pVdma, &HCtl, VBVAEXHOSTCTL_SOURCE_HOST);
1193
1194 pHgcmEnableData->hRHCmd = pVdma;
1195 pHgcmEnableData->pfnRHCmd = vboxVDMACrHgcmHandleEnableRemainingHostCommand;
1196
1197 if (RT_FAILURE(rc))
1198 {
1199 if (rc == VERR_INVALID_STATE)
1200 rc = VINF_SUCCESS;
1201 else
1202 WARN(("vdmaVBVACtlSubmitSync failed %d\n", rc));
1203 }
1204
1205 return rc;
1206}
1207
1208static int vboxVDMACrHgcmHandleEnable(struct VBOXVDMAHOST *pVdma)
1209{
1210 VBOXCRCMDCTL_ENABLE Enable;
1211 Enable.Hdr.enmType = VBOXCRCMDCTL_TYPE_ENABLE;
1212 Enable.Data.hRHCmd = pVdma;
1213 Enable.Data.pfnRHCmd = vboxVDMACrHgcmHandleEnableRemainingHostCommand;
1214
1215 int rc = vboxVDMACrHgcmSubmitSync(pVdma, &Enable.Hdr, sizeof (Enable));
1216 Assert(!pVdma->pCurRemainingHostCtl);
1217 if (RT_SUCCESS(rc))
1218 {
1219 Assert(!VBoxVBVAExHSIsEnabled(&pVdma->CmdVbva));
1220 return VINF_SUCCESS;
1221 }
1222
1223 Assert(VBoxVBVAExHSIsEnabled(&pVdma->CmdVbva));
1224 WARN(("vboxVDMACrHgcmSubmitSync failed %d\n", rc));
1225
1226 return rc;
1227}
1228
1229static int vdmaVBVAEnableProcess(struct VBOXVDMAHOST *pVdma, uint32_t u32Offset)
1230{
1231 if (VBoxVBVAExHSIsEnabled(&pVdma->CmdVbva))
1232 {
1233 WARN(("vdma VBVA is already enabled\n"));
1234 return VERR_INVALID_STATE;
1235 }
1236
1237 VBVABUFFER *pVBVA = (VBVABUFFER *)HGSMIOffsetToPointerHost(pVdma->pHgsmi, u32Offset);
1238 if (!pVBVA)
1239 {
1240 WARN(("invalid offset %d\n", u32Offset));
1241 return VERR_INVALID_PARAMETER;
1242 }
1243
1244 if (!pVdma->CrSrvInfo.pfnEnable)
1245 {
1246#ifdef DEBUG_misha
1247 WARN(("pfnEnable is NULL\n"));
1248 return VERR_NOT_SUPPORTED;
1249#endif
1250 }
1251
1252 int rc = VBoxVBVAExHSEnable(&pVdma->CmdVbva, pVBVA);
1253 if (RT_SUCCESS(rc))
1254 {
1255 VBOXCRCMDCTL_DISABLE Disable;
1256 Disable.Hdr.enmType = VBOXCRCMDCTL_TYPE_DISABLE;
1257 Disable.Data.hNotifyTerm = pVdma;
1258 Disable.Data.pfnNotifyTerm = vboxVDMACrHgcmNotifyTerminatingCb;
1259 Disable.Data.pfnNotifyTermDone = vboxVDMACrHgcmNotifyTerminatingDoneCb;
1260 rc = vboxVDMACrHgcmSubmitSync(pVdma, &Disable.Hdr, sizeof (Disable));
1261 if (RT_SUCCESS(rc))
1262 {
1263 PVGASTATE pVGAState = pVdma->pVGAState;
1264 VBOXCRCMD_SVRENABLE_INFO Info;
1265 Info.hCltScr = pVGAState->pDrv;
1266 Info.pfnCltScrUpdateBegin = pVGAState->pDrv->pfnVBVAUpdateBegin;
1267 Info.pfnCltScrUpdateProcess = pVGAState->pDrv->pfnVBVAUpdateProcess;
1268 Info.pfnCltScrUpdateEnd = pVGAState->pDrv->pfnVBVAUpdateEnd;
1269 rc = pVdma->CrSrvInfo.pfnEnable(pVdma->CrSrvInfo.hSvr, &Info);
1270 if (RT_SUCCESS(rc))
1271 return VINF_SUCCESS;
1272 else
1273 WARN(("pfnEnable failed %d\n", rc));
1274
1275 vboxVDMACrHgcmHandleEnable(pVdma);
1276 }
1277 else
1278 WARN(("vboxVDMACrHgcmSubmitSync failed %d\n", rc));
1279
1280 VBoxVBVAExHSDisable(&pVdma->CmdVbva);
1281 }
1282 else
1283 WARN(("VBoxVBVAExHSEnable failed %d\n", rc));
1284
1285 return rc;
1286}
1287
1288static int vdmaVBVADisableProcess(struct VBOXVDMAHOST *pVdma, bool fDoHgcmEnable)
1289{
1290 if (!VBoxVBVAExHSIsEnabled(&pVdma->CmdVbva))
1291 {
1292 Log(("vdma VBVA is already disabled\n"));
1293 return VINF_SUCCESS;
1294 }
1295
1296 int rc = pVdma->CrSrvInfo.pfnDisable(pVdma->CrSrvInfo.hSvr);
1297 if (RT_SUCCESS(rc))
1298 {
1299 if (fDoHgcmEnable)
1300 {
1301 PVGASTATE pVGAState = pVdma->pVGAState;
1302
1303 /* disable is a bit tricky
1304 * we need to ensure the host ctl commands do not come out of order
1305 * and do not come over HGCM channel until after it is enabled */
1306 rc = vboxVDMACrHgcmHandleEnable(pVdma);
1307 if (RT_SUCCESS(rc))
1308 {
1309 vdmaVBVANotifyDisable(pVGAState);
1310 return VINF_SUCCESS;
1311 }
1312
1313 VBOXCRCMD_SVRENABLE_INFO Info;
1314 Info.hCltScr = pVGAState->pDrv;
1315 Info.pfnCltScrUpdateBegin = pVGAState->pDrv->pfnVBVAUpdateBegin;
1316 Info.pfnCltScrUpdateProcess = pVGAState->pDrv->pfnVBVAUpdateProcess;
1317 Info.pfnCltScrUpdateEnd = pVGAState->pDrv->pfnVBVAUpdateEnd;
1318 pVdma->CrSrvInfo.pfnEnable(pVdma->CrSrvInfo.hSvr, &Info);
1319 }
1320 }
1321 else
1322 WARN(("pfnDisable failed %d\n", rc));
1323
1324 return rc;
1325}
1326
1327static int vboxVDMACrHostCtlProcess(struct VBOXVDMAHOST *pVdma, VBVAEXHOSTCTL *pCmd, bool *pfContinue)
1328{
1329 *pfContinue = true;
1330
1331 switch (pCmd->enmType)
1332 {
1333 case VBVAEXHOSTCTL_TYPE_GHH_BE_OPAQUE:
1334 {
1335 if (!VBoxVBVAExHSIsEnabled(&pVdma->CmdVbva))
1336 {
1337 WARN(("VBVAEXHOSTCTL_TYPE_GHH_BE_OPAQUE for disabled vdma VBVA\n"));
1338 return VERR_INVALID_STATE;
1339 }
1340 return pVdma->CrSrvInfo.pfnHostCtl(pVdma->CrSrvInfo.hSvr, pCmd->u.cmd.pu8Cmd, pCmd->u.cmd.cbCmd);
1341 }
1342 case VBVAEXHOSTCTL_TYPE_GHH_DISABLE:
1343 {
1344 int rc = vdmaVBVADisableProcess(pVdma, true);
1345 if (RT_FAILURE(rc))
1346 {
1347 WARN(("vdmaVBVADisableProcess failed %d\n", rc));
1348 return rc;
1349 }
1350
1351 return VBoxVDMAThreadTerm(&pVdma->Thread, NULL, NULL, false);
1352 }
1353 case VBVAEXHOSTCTL_TYPE_HH_ON_HGCM_UNLOAD:
1354 {
1355 int rc = vdmaVBVADisableProcess(pVdma, false);
1356 if (RT_FAILURE(rc))
1357 {
1358 WARN(("vdmaVBVADisableProcess failed %d\n", rc));
1359 return rc;
1360 }
1361
1362 rc = VBoxVDMAThreadTerm(&pVdma->Thread, NULL, NULL, true);
1363 if (RT_FAILURE(rc))
1364 {
1365 WARN(("VBoxVDMAThreadTerm failed %d\n", rc));
1366 return rc;
1367 }
1368
1369 *pfContinue = false;
1370 return VINF_SUCCESS;
1371 }
1372 case VBVAEXHOSTCTL_TYPE_HH_SAVESTATE:
1373 {
1374 PVGASTATE pVGAState = pVdma->pVGAState;
1375 uint8_t * pu8VramBase = pVGAState->vram_ptrR3;
1376 int rc = VBoxVBVAExHSSaveState(&pVdma->CmdVbva, pu8VramBase, pCmd->u.state.pSSM);
1377 if (RT_FAILURE(rc))
1378 {
1379 WARN(("VBoxVBVAExHSSaveState failed %d\n", rc));
1380 return rc;
1381 }
1382 return pVdma->CrSrvInfo.pfnSaveState(pVdma->CrSrvInfo.hSvr, pCmd->u.state.pSSM);
1383 }
1384 case VBVAEXHOSTCTL_TYPE_HH_LOADSTATE:
1385 {
1386 PVGASTATE pVGAState = pVdma->pVGAState;
1387 uint8_t * pu8VramBase = pVGAState->vram_ptrR3;
1388
1389 int rc = VBoxVBVAExHSLoadState(&pVdma->CmdVbva, pu8VramBase, pCmd->u.state.pSSM, pCmd->u.state.u32Version);
1390 if (RT_FAILURE(rc))
1391 {
1392 WARN(("VBoxVBVAExHSSaveState failed %d\n", rc));
1393 return rc;
1394 }
1395
1396 rc = pVdma->CrSrvInfo.pfnLoadState(pVdma->CrSrvInfo.hSvr, pCmd->u.state.pSSM, pCmd->u.state.u32Version);
1397 if (RT_FAILURE(rc))
1398 {
1399 WARN(("pfnLoadState failed %d\n", rc));
1400 return rc;
1401 }
1402
1403 return VINF_SUCCESS;
1404 }
1405 case VBVAEXHOSTCTL_TYPE_HH_LOADSTATE_DONE:
1406 {
1407 PVGASTATE pVGAState = pVdma->pVGAState;
1408
1409 for (uint32_t i = 0; i < pVGAState->cMonitors; ++i)
1410 {
1411 VBVAINFOSCREEN CurScreen;
1412 VBVAINFOVIEW CurView;
1413
1414 int rc = VBVAGetInfoViewAndScreen(pVGAState, i, &CurView, &CurScreen);
1415 if (RT_FAILURE(rc))
1416 {
1417 WARN(("VBVAGetInfoViewAndScreen failed %d\n", rc));
1418 return rc;
1419 }
1420
1421 rc = VBVAInfoScreen(pVGAState, &CurScreen);
1422 if (RT_FAILURE(rc))
1423 {
1424 WARN(("VBVAInfoScreen failed %d\n", rc));
1425 return rc;
1426 }
1427 }
1428
1429 return VINF_SUCCESS;
1430 }
1431 default:
1432 WARN(("unexpected host ctl type %d\n", pCmd->enmType));
1433 return VERR_INVALID_PARAMETER;
1434 }
1435}
1436
1437static int vboxVDMACrGuestCtlResizeEntryProcess(struct VBOXVDMAHOST *pVdma, VBOXCMDVBVA_RESIZE_ENTRY *pEntry)
1438{
1439 PVGASTATE pVGAState = pVdma->pVGAState;
1440 VBVAINFOSCREEN Screen = pEntry->Screen;
1441 VBVAINFOVIEW View;
1442 VBOXCMDVBVA_SCREENMAP_DECL(uint32_t, aTargetMap);
1443 uint32_t u32ViewIndex = Screen.u32ViewIndex;
1444 uint16_t u16Flags = Screen.u16Flags;
1445 bool fDisable = false;
1446
1447 memcpy(aTargetMap, pEntry->aTargetMap, sizeof (aTargetMap));
1448
1449 ASMBitClearRange(aTargetMap, pVGAState->cMonitors, VBOX_VIDEO_MAX_SCREENS);
1450
1451 if (u16Flags & VBVA_SCREEN_F_DISABLED)
1452 {
1453 fDisable = true;
1454 memset(&Screen, 0, sizeof (Screen));
1455 Screen.u32ViewIndex = u32ViewIndex;
1456 Screen.u16Flags = VBVA_SCREEN_F_ACTIVE | VBVA_SCREEN_F_DISABLED;
1457 }
1458
1459 if (u32ViewIndex > pVGAState->cMonitors)
1460 {
1461 if (u32ViewIndex != 0xffffffff)
1462 {
1463 WARN(("invalid view index\n"));
1464 return VERR_INVALID_PARAMETER;
1465 }
1466 else if (!fDisable)
1467 {
1468 WARN(("0xffffffff view index only valid for disable requests\n"));
1469 return VERR_INVALID_PARAMETER;
1470 }
1471 }
1472
1473 View.u32ViewOffset = 0;
1474 View.u32ViewSize = Screen.u32LineSize * Screen.u32Height + Screen.u32StartOffset;
1475 View.u32MaxScreenSize = View.u32ViewSize + Screen.u32Width + 1; /* <- make VBVAInfoScreen logic (offEnd < pView->u32MaxScreenSize) happy */
1476
1477 int rc = VINF_SUCCESS;
1478
1479 rc = pVdma->CrSrvInfo.pfnResize(pVdma->CrSrvInfo.hSvr, &Screen, aTargetMap);
1480 if (RT_FAILURE(rc))
1481 {
1482 WARN(("pfnResize failed %d\n", rc));
1483 return rc;
1484 }
1485
1486 for (int i = ASMBitFirstSet(aTargetMap, pVGAState->cMonitors);
1487 i >= 0;
1488 i = ASMBitNextSet(aTargetMap, pVGAState->cMonitors, i))
1489 {
1490 Screen.u32ViewIndex = i;
1491
1492 VBVAINFOSCREEN CurScreen;
1493 VBVAINFOVIEW CurView;
1494
1495 rc = VBVAGetInfoViewAndScreen(pVGAState, i, &CurView, &CurScreen);
1496 AssertRC(rc);
1497
1498 if (!memcmp(&Screen, &CurScreen, sizeof (CurScreen)))
1499 continue;
1500
1501 if (!fDisable || !CurView.u32ViewSize)
1502 {
1503 View.u32ViewIndex = Screen.u32ViewIndex;
1504
1505 rc = VBVAInfoView(pVGAState, &View);
1506 if (RT_FAILURE(rc))
1507 {
1508 WARN(("VBVAInfoView failed %d\n", rc));
1509 break;
1510 }
1511 }
1512
1513 rc = VBVAInfoScreen(pVGAState, &Screen);
1514 if (RT_FAILURE(rc))
1515 {
1516 WARN(("VBVAInfoScreen failed %d\n", rc));
1517 break;
1518 }
1519 }
1520
1521 if (RT_FAILURE(rc))
1522 return rc;
1523
1524 Screen.u32ViewIndex = u32ViewIndex;
1525
1526 return rc;
1527}
1528
1529static int vboxVDMACrGuestCtlProcess(struct VBOXVDMAHOST *pVdma, VBVAEXHOSTCTL *pCmd)
1530{
1531 VBVAEXHOSTCTL_TYPE enmType = pCmd->enmType;
1532 switch (enmType)
1533 {
1534 case VBVAEXHOSTCTL_TYPE_GHH_BE_OPAQUE:
1535 {
1536 if (!VBoxVBVAExHSIsEnabled(&pVdma->CmdVbva))
1537 {
1538 WARN(("VBVAEXHOSTCTL_TYPE_GHH_BE_OPAQUE for disabled vdma VBVA\n"));
1539 return VERR_INVALID_STATE;
1540 }
1541 return pVdma->CrSrvInfo.pfnGuestCtl(pVdma->CrSrvInfo.hSvr, pCmd->u.cmd.pu8Cmd, pCmd->u.cmd.cbCmd);
1542 }
1543 case VBVAEXHOSTCTL_TYPE_GHH_RESIZE:
1544 {
1545 if (!VBoxVBVAExHSIsEnabled(&pVdma->CmdVbva))
1546 {
1547 WARN(("VBVAEXHOSTCTL_TYPE_GHH_BE_OPAQUE for disabled vdma VBVA\n"));
1548 return VERR_INVALID_STATE;
1549 }
1550
1551 uint32_t cbCmd = pCmd->u.cmd.cbCmd;
1552
1553 if (cbCmd % sizeof (VBOXCMDVBVA_RESIZE_ENTRY))
1554 {
1555 WARN(("invalid buffer size\n"));
1556 return VERR_INVALID_PARAMETER;
1557 }
1558
1559 uint32_t cElements = cbCmd / sizeof (VBOXCMDVBVA_RESIZE_ENTRY);
1560 if (!cElements)
1561 {
1562 WARN(("invalid buffer size\n"));
1563 return VERR_INVALID_PARAMETER;
1564 }
1565
1566 VBOXCMDVBVA_RESIZE *pResize = (VBOXCMDVBVA_RESIZE*)pCmd->u.cmd.pu8Cmd;
1567
1568 int rc = VINF_SUCCESS;
1569
1570 for (uint32_t i = 0; i < cElements; ++i)
1571 {
1572 VBOXCMDVBVA_RESIZE_ENTRY *pEntry = &pResize->aEntries[i];
1573 rc = vboxVDMACrGuestCtlResizeEntryProcess(pVdma, pEntry);
1574 if (RT_FAILURE(rc))
1575 {
1576 WARN(("vboxVDMACrGuestCtlResizeEntryProcess failed %d\n", rc));
1577 break;
1578 }
1579 }
1580 return rc;
1581 }
1582 case VBVAEXHOSTCTL_TYPE_GHH_ENABLE:
1583 case VBVAEXHOSTCTL_TYPE_GHH_ENABLE_PAUSED:
1584 {
1585 VBVAENABLE *pEnable = (VBVAENABLE *)pCmd->u.cmd.pu8Cmd;
1586 Assert(pCmd->u.cmd.cbCmd == sizeof (VBVAENABLE));
1587 uint32_t u32Offset = pEnable->u32Offset;
1588 int rc = vdmaVBVAEnableProcess(pVdma, u32Offset);
1589 if (!RT_SUCCESS(rc))
1590 {
1591 WARN(("vdmaVBVAEnableProcess failed %d\n", rc));
1592 return rc;
1593 }
1594
1595 if (enmType == VBVAEXHOSTCTL_TYPE_GHH_ENABLE_PAUSED)
1596 {
1597 rc = VBoxVBVAExHPPause(&pVdma->CmdVbva);
1598 if (!RT_SUCCESS(rc))
1599 {
1600 WARN(("VBoxVBVAExHPPause failed %d\n", rc));
1601 return rc;
1602 }
1603 }
1604
1605 return VINF_SUCCESS;
1606 }
1607 case VBVAEXHOSTCTL_TYPE_GHH_DISABLE:
1608 {
1609 int rc = vdmaVBVADisableProcess(pVdma, true);
1610 if (RT_FAILURE(rc))
1611 {
1612 WARN(("vdmaVBVADisableProcess failed %d\n", rc));
1613 return rc;
1614 }
1615
1616 /* do vgaUpdateDisplayAll right away */
1617 VMR3ReqCallNoWait(PDMDevHlpGetVM(pVdma->pVGAState->pDevInsR3), VMCPUID_ANY,
1618 (PFNRT)vgaUpdateDisplayAll, 2, pVdma->pVGAState, /* fFailOnResize = */ false);
1619
1620 return VBoxVDMAThreadTerm(&pVdma->Thread, NULL, NULL, false);
1621 }
1622 default:
1623 WARN(("unexpected ctl type %d\n", pCmd->enmType));
1624 return VERR_INVALID_PARAMETER;
1625 }
1626}
1627
1628/**
1629 * @param fIn - whether this is a page in or out op.
1630 * the direction is VRA#M - related, so fIn == true - transfer to VRAM); false - transfer from VRAM
1631 */
1632static int vboxVDMACrCmdVbvaProcessPagingEl(PPDMDEVINS pDevIns, VBOXCMDVBVAPAGEIDX iPage, uint8_t *pu8Vram, bool fIn)
1633{
1634 RTGCPHYS phPage = (RTGCPHYS)iPage << PAGE_SHIFT;
1635 PGMPAGEMAPLOCK Lock;
1636 int rc;
1637
1638 if (fIn)
1639 {
1640 const void * pvPage;
1641 rc = PDMDevHlpPhysGCPhys2CCPtrReadOnly(pDevIns, phPage, 0, &pvPage, &Lock);
1642 if (!RT_SUCCESS(rc))
1643 {
1644 WARN(("PDMDevHlpPhysGCPhys2CCPtrReadOnly failed %d", rc));
1645 return rc;
1646 }
1647
1648 memcpy(pu8Vram, pvPage, PAGE_SIZE);
1649
1650 PDMDevHlpPhysReleasePageMappingLock(pDevIns, &Lock);
1651 }
1652 else
1653 {
1654 void * pvPage;
1655 rc = PDMDevHlpPhysGCPhys2CCPtr(pDevIns, phPage, 0, &pvPage, &Lock);
1656 if (!RT_SUCCESS(rc))
1657 {
1658 WARN(("PDMDevHlpPhysGCPhys2CCPtr failed %d", rc));
1659 return rc;
1660 }
1661
1662 memcpy(pvPage, pu8Vram, PAGE_SIZE);
1663
1664 PDMDevHlpPhysReleasePageMappingLock(pDevIns, &Lock);
1665 }
1666
1667 return VINF_SUCCESS;
1668}
1669
1670static int vboxVDMACrCmdVbvaProcessPagingEls(PPDMDEVINS pDevIns, const VBOXCMDVBVAPAGEIDX *piPages, uint32_t cPages, uint8_t *pu8Vram, bool fIn)
1671{
1672 for (uint32_t i = 0; i < cPages; ++i, pu8Vram += PAGE_SIZE)
1673 {
1674 int rc = vboxVDMACrCmdVbvaProcessPagingEl(pDevIns, piPages[i], pu8Vram, fIn);
1675 if (!RT_SUCCESS(rc))
1676 {
1677 WARN(("vboxVDMACrCmdVbvaProcessPagingEl failed %d", rc));
1678 return rc;
1679 }
1680 }
1681
1682 return VINF_SUCCESS;
1683}
1684
1685static int8_t vboxVDMACrCmdVbvaPagingDataInit(PVGASTATE pVGAState, const VBOXCMDVBVA_HDR *pHdr, const VBOXCMDVBVA_PAGING_TRANSFER_DATA *pData, uint32_t cbCmd,
1686 const VBOXCMDVBVAPAGEIDX **ppPages, VBOXCMDVBVAPAGEIDX *pcPages,
1687 uint8_t **ppu8Vram, bool *pfIn)
1688{
1689 if (cbCmd < sizeof (VBOXCMDVBVA_PAGING_TRANSFER))
1690 {
1691 WARN(("cmd too small"));
1692 return -1;
1693 }
1694
1695 VBOXCMDVBVAPAGEIDX cPages = cbCmd - RT_OFFSETOF(VBOXCMDVBVA_PAGING_TRANSFER, Data.aPageNumbers);
1696 if (cPages % sizeof (VBOXCMDVBVAPAGEIDX))
1697 {
1698 WARN(("invalid cmd size"));
1699 return -1;
1700 }
1701 cPages /= sizeof (VBOXCMDVBVAPAGEIDX);
1702
1703 VBOXCMDVBVAOFFSET offVRAM = pData->Alloc.u.offVRAM;
1704 if (offVRAM & PAGE_OFFSET_MASK)
1705 {
1706 WARN(("offVRAM address is not on page boundary\n"));
1707 return -1;
1708 }
1709 const VBOXCMDVBVAPAGEIDX *pPages = pData->aPageNumbers;
1710
1711 uint8_t * pu8VramBase = pVGAState->vram_ptrR3;
1712 uint8_t *pu8VramMax = pu8VramBase + pVGAState->vram_size;
1713 if (offVRAM >= pVGAState->vram_size)
1714 {
1715 WARN(("invalid vram offset"));
1716 return -1;
1717 }
1718
1719 if (~(~(VBOXCMDVBVAPAGEIDX)0 >> PAGE_SHIFT) & cPages)
1720 {
1721 WARN(("invalid cPages %d", cPages));
1722 return -1;
1723 }
1724
1725 if (offVRAM + ((VBOXCMDVBVAOFFSET)cPages << PAGE_SHIFT) >= pVGAState->vram_size)
1726 {
1727 WARN(("invalid cPages %d, exceeding vram size", cPages));
1728 return -1;
1729 }
1730
1731 uint8_t *pu8Vram = pu8VramBase + offVRAM;
1732 bool fIn = !!(pHdr->u8Flags & VBOXCMDVBVA_OPF_PAGING_TRANSFER_IN);
1733
1734 *ppPages = pPages;
1735 *pcPages = cPages;
1736 *ppu8Vram = pu8Vram;
1737 *pfIn = fIn;
1738 return 0;
1739}
1740
1741static int8_t vboxVDMACrCmdVbvaPagingFill(PVGASTATE pVGAState, VBOXCMDVBVA_PAGING_FILL *pFill)
1742{
1743 VBOXCMDVBVAOFFSET offVRAM = pFill->offVRAM;
1744 if (offVRAM & PAGE_OFFSET_MASK)
1745 {
1746 WARN(("offVRAM address is not on page boundary\n"));
1747 return -1;
1748 }
1749
1750 uint8_t * pu8VramBase = pVGAState->vram_ptrR3;
1751 uint8_t *pu8VramMax = pu8VramBase + pVGAState->vram_size;
1752 if (offVRAM >= pVGAState->vram_size)
1753 {
1754 WARN(("invalid vram offset"));
1755 return -1;
1756 }
1757
1758 uint32_t cbFill = pFill->u32CbFill;
1759
1760 if (offVRAM + cbFill >= pVGAState->vram_size)
1761 {
1762 WARN(("invalid cPages"));
1763 return -1;
1764 }
1765
1766 uint32_t *pu32Vram = (uint32_t*)(pu8VramBase + offVRAM);
1767 uint32_t u32Color = pFill->u32Pattern;
1768
1769 Assert(!(cbFill % 4));
1770 for (uint32_t i = 0; i < cbFill / 4; ++i)
1771 {
1772 pu32Vram[i] = u32Color;
1773 }
1774
1775 return 0;
1776}
1777
1778static int8_t vboxVDMACrCmdVbvaProcessCmdData(struct VBOXVDMAHOST *pVdma, const VBOXCMDVBVA_HDR *pCmd, uint32_t cbCmd)
1779{
1780 switch (pCmd->u8OpCode)
1781 {
1782 case VBOXCMDVBVA_OPTYPE_NOPCMD:
1783 return 0;
1784 case VBOXCMDVBVA_OPTYPE_PAGING_TRANSFER:
1785 {
1786 PVGASTATE pVGAState = pVdma->pVGAState;
1787 const VBOXCMDVBVAPAGEIDX *pPages;
1788 uint32_t cPages;
1789 uint8_t *pu8Vram;
1790 bool fIn;
1791 int8_t i8Result = vboxVDMACrCmdVbvaPagingDataInit(pVGAState, pCmd, &((VBOXCMDVBVA_PAGING_TRANSFER*)pCmd)->Data, cbCmd,
1792 &pPages, &cPages,
1793 &pu8Vram, &fIn);
1794 if (i8Result < 0)
1795 {
1796 WARN(("vboxVDMACrCmdVbvaPagingDataInit failed %d", i8Result));
1797 return i8Result;
1798 }
1799
1800 PPDMDEVINS pDevIns = pVGAState->pDevInsR3;
1801 int rc = vboxVDMACrCmdVbvaProcessPagingEls(pDevIns, pPages, cPages, pu8Vram, fIn);
1802 if (!RT_SUCCESS(rc))
1803 {
1804 WARN(("vboxVDMACrCmdVbvaProcessPagingEls failed %d", rc));
1805 return -1;
1806 }
1807
1808 return 0;
1809 }
1810 case VBOXCMDVBVA_OPTYPE_PAGING_FILL:
1811 {
1812 PVGASTATE pVGAState = pVdma->pVGAState;
1813 if (cbCmd != sizeof (VBOXCMDVBVA_PAGING_FILL))
1814 {
1815 WARN(("cmd too small"));
1816 return -1;
1817 }
1818
1819 return vboxVDMACrCmdVbvaPagingFill(pVGAState, (VBOXCMDVBVA_PAGING_FILL*)pCmd);
1820 }
1821 default:
1822 return pVdma->CrSrvInfo.pfnCmd(pVdma->CrSrvInfo.hSvr, pCmd, cbCmd);
1823 }
1824}
1825
1826#if 0
1827typedef struct VBOXCMDVBVA_PAGING_TRANSFER
1828{
1829 VBOXCMDVBVA_HDR Hdr;
1830 /* for now can only contain offVRAM.
1831 * paging transfer can NOT be initiated for allocations having host 3D object (hostID) associated */
1832 VBOXCMDVBVA_ALLOCINFO Alloc;
1833 uint32_t u32Reserved;
1834 VBOXCMDVBVA_SYSMEMEL aSysMem[1];
1835} VBOXCMDVBVA_PAGING_TRANSFER;
1836#endif
1837
1838AssertCompile(sizeof (VBOXCMDVBVA_HDR) == 8);
1839AssertCompile(sizeof (VBOXCMDVBVA_ALLOCINFO) == 4);
1840AssertCompile(sizeof (VBOXCMDVBVAPAGEIDX) == 4);
1841AssertCompile(!(PAGE_SIZE % sizeof (VBOXCMDVBVAPAGEIDX)));
1842
1843#define VBOXCMDVBVA_NUM_SYSMEMEL_PER_PAGE (PAGE_SIZE / sizeof (VBOXCMDVBVA_SYSMEMEL))
1844
1845static int8_t vboxVDMACrCmdVbvaProcess(struct VBOXVDMAHOST *pVdma, const VBOXCMDVBVA_HDR *pCmd, uint32_t cbCmd)
1846{
1847 switch (pCmd->u8OpCode)
1848 {
1849 case VBOXCMDVBVA_OPTYPE_SYSMEMCMD:
1850 {
1851 if (cbCmd < sizeof (VBOXCMDVBVA_SYSMEMCMD))
1852 {
1853 WARN(("invalid command size"));
1854 return -1;
1855 }
1856 VBOXCMDVBVA_SYSMEMCMD *pSysmemCmd = (VBOXCMDVBVA_SYSMEMCMD*)pCmd;
1857 const VBOXCMDVBVA_HDR *pRealCmdHdr;
1858 uint32_t cbRealCmd = pCmd->u8Flags;
1859 cbRealCmd |= (uint32_t)pCmd->u.u8PrimaryID << 8;
1860 if (cbRealCmd < sizeof (VBOXCMDVBVA_HDR))
1861 {
1862 WARN(("invalid sysmem cmd size"));
1863 return -1;
1864 }
1865
1866 RTGCPHYS phCmd = (RTGCPHYS)pSysmemCmd->phCmd;
1867
1868 PGMPAGEMAPLOCK Lock;
1869 PVGASTATE pVGAState = pVdma->pVGAState;
1870 PPDMDEVINS pDevIns = pVGAState->pDevInsR3;
1871 const void * pvCmd;
1872 int rc = PDMDevHlpPhysGCPhys2CCPtrReadOnly(pDevIns, phCmd, 0, &pvCmd, &Lock);
1873 if (!RT_SUCCESS(rc))
1874 {
1875 WARN(("PDMDevHlpPhysGCPhys2CCPtrReadOnly failed %d\n", rc));
1876 return -1;
1877 }
1878
1879 Assert((phCmd & PAGE_OFFSET_MASK) == (((uintptr_t)pvCmd) & PAGE_OFFSET_MASK));
1880
1881 uint32_t cbCmdPart = PAGE_SIZE - (((uintptr_t)pvCmd) & PAGE_OFFSET_MASK);
1882
1883 if (cbRealCmd <= cbCmdPart)
1884 {
1885 pRealCmdHdr = (const VBOXCMDVBVA_HDR *)pvCmd;
1886 uint8_t i8Result = vboxVDMACrCmdVbvaProcessCmdData(pVdma, pRealCmdHdr, cbRealCmd);
1887 PDMDevHlpPhysReleasePageMappingLock(pDevIns, &Lock);
1888 return i8Result;
1889 }
1890
1891 VBOXCMDVBVA_HDR Hdr;
1892 const void *pvCurCmdTail;
1893 uint32_t cbCurCmdTail;
1894 if (cbCmdPart >= sizeof (*pRealCmdHdr))
1895 {
1896 pRealCmdHdr = (const VBOXCMDVBVA_HDR *)pvCmd;
1897 pvCurCmdTail = (const void*)(pRealCmdHdr + 1);
1898 cbCurCmdTail = cbCmdPart - sizeof (*pRealCmdHdr);
1899 }
1900 else
1901 {
1902 memcpy(&Hdr, pvCmd, cbCmdPart);
1903 PDMDevHlpPhysReleasePageMappingLock(pDevIns, &Lock);
1904 phCmd += cbCmdPart;
1905 Assert(!(phCmd & PAGE_OFFSET_MASK));
1906 rc = PDMDevHlpPhysGCPhys2CCPtrReadOnly(pDevIns, phCmd, 0, &pvCmd, &Lock);
1907 if (!RT_SUCCESS(rc))
1908 {
1909 WARN(("PDMDevHlpPhysGCPhys2CCPtrReadOnly failed %d\n", rc));
1910 return -1;
1911 }
1912
1913 cbCmdPart = sizeof (*pRealCmdHdr) - cbCmdPart;
1914 memcpy(((uint8_t*)(&Hdr)) + cbCmdPart, pvCmd, cbCmdPart);
1915 pRealCmdHdr = &Hdr;
1916 pvCurCmdTail = (const void*)(((uint8_t*)pvCmd) + cbCmdPart);
1917 cbCurCmdTail = PAGE_SIZE - cbCmdPart;
1918 }
1919
1920 if (cbCurCmdTail > cbRealCmd - sizeof (*pRealCmdHdr))
1921 cbCurCmdTail = cbRealCmd - sizeof (*pRealCmdHdr);
1922
1923 int8_t i8Result = 0;
1924
1925 switch (pRealCmdHdr->u8OpCode)
1926 {
1927 case VBOXCMDVBVA_OPTYPE_PAGING_TRANSFER:
1928 {
1929 const uint32_t *pPages;
1930 uint32_t cPages;
1931 uint8_t *pu8Vram;
1932 bool fIn;
1933 i8Result = vboxVDMACrCmdVbvaPagingDataInit(pVGAState, pRealCmdHdr, (const VBOXCMDVBVA_PAGING_TRANSFER_DATA*)pvCurCmdTail, cbRealCmd,
1934 &pPages, &cPages,
1935 &pu8Vram, &fIn);
1936 if (i8Result < 0)
1937 {
1938 WARN(("vboxVDMACrCmdVbvaPagingDataInit failed %d", i8Result));
1939 /* we need to break, not return, to ensure currently locked page is released */
1940 break;
1941 }
1942
1943 if (cbCurCmdTail & 3)
1944 {
1945 WARN(("command is not alligned properly %d", cbCurCmdTail));
1946 i8Result = -1;
1947 /* we need to break, not return, to ensure currently locked page is released */
1948 break;
1949 }
1950
1951 uint32_t cCurPages = cbCurCmdTail / sizeof (VBOXCMDVBVAPAGEIDX);
1952 Assert(cCurPages < cPages);
1953
1954 do
1955 {
1956 rc = vboxVDMACrCmdVbvaProcessPagingEls(pDevIns, pPages, cCurPages, pu8Vram, fIn);
1957 if (!RT_SUCCESS(rc))
1958 {
1959 WARN(("vboxVDMACrCmdVbvaProcessPagingEls failed %d", rc));
1960 i8Result = -1;
1961 /* we need to break, not return, to ensure currently locked page is released */
1962 break;
1963 }
1964
1965 Assert(cPages >= cCurPages);
1966 cPages -= cCurPages;
1967
1968 if (!cPages)
1969 break;
1970
1971 PDMDevHlpPhysReleasePageMappingLock(pDevIns, &Lock);
1972
1973 Assert(!(phCmd & PAGE_OFFSET_MASK));
1974
1975 phCmd += PAGE_SIZE;
1976 pu8Vram += (VBOXCMDVBVAOFFSET)cCurPages << PAGE_SHIFT;
1977
1978 rc = PDMDevHlpPhysGCPhys2CCPtrReadOnly(pDevIns, phCmd, 0, &pvCmd, &Lock);
1979 if (!RT_SUCCESS(rc))
1980 {
1981 WARN(("PDMDevHlpPhysGCPhys2CCPtrReadOnly failed %d\n", rc));
1982 /* the page is not locked, return */
1983 return -1;
1984 }
1985
1986 cCurPages = PAGE_SIZE / sizeof (VBOXCMDVBVAPAGEIDX);
1987 if (cCurPages > cPages)
1988 cCurPages = cPages;
1989 } while (1);
1990 break;
1991 }
1992 default:
1993 WARN(("command can not be splitted"));
1994 i8Result = -1;
1995 break;
1996 }
1997
1998 PDMDevHlpPhysReleasePageMappingLock(pDevIns, &Lock);
1999 return i8Result;
2000 }
2001 case VBOXCMDVBVA_OPTYPE_COMPLEXCMD:
2002 {
2003 Assert(cbCmd >= sizeof (VBOXCMDVBVA_HDR));
2004 ++pCmd;
2005 cbCmd -= sizeof (*pCmd);
2006 uint32_t cbCurCmd = 0;
2007 for ( ; cbCmd; cbCmd -= cbCurCmd, pCmd = (VBOXCMDVBVA_HDR*)(((uint8_t*)pCmd) + cbCurCmd))
2008 {
2009 if (cbCmd < sizeof (VBOXCMDVBVA_HDR))
2010 {
2011 WARN(("invalid command size"));
2012 return -1;
2013 }
2014
2015 cbCurCmd = pCmd->u2.complexCmdEl.u16CbCmdHost;
2016 if (cbCmd < cbCurCmd)
2017 {
2018 WARN(("invalid command size"));
2019 return -1;
2020 }
2021
2022 int8_t i8Result = vboxVDMACrCmdVbvaProcess(pVdma, pCmd, cbCurCmd);
2023 if (i8Result < 0)
2024 {
2025 WARN(("vboxVDMACrCmdVbvaProcess failed"));
2026 return i8Result;
2027 }
2028 }
2029 return 0;
2030 }
2031 default:
2032 return vboxVDMACrCmdVbvaProcessCmdData(pVdma, pCmd, cbCmd);
2033 }
2034}
2035
2036static void vboxVDMACrCmdProcess(struct VBOXVDMAHOST *pVdma, uint8_t* pu8Cmd, uint32_t cbCmd)
2037{
2038 if (*pu8Cmd == VBOXCMDVBVA_OPTYPE_NOP)
2039 return;
2040
2041 if (cbCmd < sizeof (VBOXCMDVBVA_HDR))
2042 {
2043 WARN(("invalid command size"));
2044 return;
2045 }
2046
2047 PVBOXCMDVBVA_HDR pCmd = (PVBOXCMDVBVA_HDR)pu8Cmd;
2048
2049 /* check if the command is cancelled */
2050 if (!ASMAtomicCmpXchgU8(&pCmd->u8State, VBOXCMDVBVA_STATE_IN_PROGRESS, VBOXCMDVBVA_STATE_SUBMITTED))
2051 {
2052 Assert(pCmd->u8State == VBOXCMDVBVA_STATE_CANCELLED);
2053 return;
2054 }
2055
2056 pCmd->u.i8Result = vboxVDMACrCmdVbvaProcess(pVdma, pCmd, cbCmd);
2057}
2058
2059static int vboxVDMACrCtlHgsmiSetup(struct VBOXVDMAHOST *pVdma)
2060{
2061 PVBOXVDMACMD_CHROMIUM_CTL_CRHGSMI_SETUP pCmd = (PVBOXVDMACMD_CHROMIUM_CTL_CRHGSMI_SETUP)
2062 vboxVDMACrCtlCreate (VBOXVDMACMD_CHROMIUM_CTL_TYPE_CRHGSMI_SETUP, sizeof (*pCmd));
2063 int rc = VERR_NO_MEMORY;
2064 if (pCmd)
2065 {
2066 PVGASTATE pVGAState = pVdma->pVGAState;
2067 pCmd->pvVRamBase = pVGAState->vram_ptrR3;
2068 pCmd->cbVRam = pVGAState->vram_size;
2069 pCmd->pLed = &pVGAState->Led3D;
2070 pCmd->CrClientInfo.hClient = pVdma;
2071 pCmd->CrClientInfo.pfnCallout = vboxCmdVBVACmdCallout;
2072 rc = vboxVDMACrCtlPost(pVGAState, &pCmd->Hdr, sizeof (*pCmd));
2073 if (RT_SUCCESS(rc))
2074 {
2075 rc = vboxVDMACrCtlGetRc(&pCmd->Hdr);
2076 if (RT_SUCCESS(rc))
2077 pVdma->CrSrvInfo = pCmd->CrCmdServerInfo;
2078 else if (rc != VERR_NOT_SUPPORTED)
2079 WARN(("vboxVDMACrCtlGetRc returned %d\n", rc));
2080 }
2081 else
2082 WARN(("vboxVDMACrCtlPost failed %d\n", rc));
2083
2084 vboxVDMACrCtlRelease(&pCmd->Hdr);
2085 }
2086
2087 if (!RT_SUCCESS(rc))
2088 memset(&pVdma->CrSrvInfo, 0, sizeof (pVdma->CrSrvInfo));
2089
2090 return rc;
2091}
2092
2093static int vboxVDMACmdExecBpbTransfer(PVBOXVDMAHOST pVdma, const PVBOXVDMACMD_DMA_BPB_TRANSFER pTransfer, uint32_t cbBuffer);
2094
2095/* check if this is external cmd to be passed to chromium backend */
2096static int vboxVDMACmdCheckCrCmd(struct VBOXVDMAHOST *pVdma, PVBOXVDMACBUF_DR pCmdDr, uint32_t cbCmdDr)
2097{
2098 PVBOXVDMACMD pDmaCmd = NULL;
2099 uint32_t cbDmaCmd = 0;
2100 uint8_t * pvRam = pVdma->pVGAState->vram_ptrR3;
2101 int rc = VINF_NOT_SUPPORTED;
2102
2103 cbDmaCmd = pCmdDr->cbBuf;
2104
2105 if (pCmdDr->fFlags & VBOXVDMACBUF_FLAG_BUF_FOLLOWS_DR)
2106 {
2107 if (cbCmdDr < sizeof (*pCmdDr) + VBOXVDMACMD_HEADER_SIZE())
2108 {
2109 AssertMsgFailed(("invalid buffer data!"));
2110 return VERR_INVALID_PARAMETER;
2111 }
2112
2113 if (cbDmaCmd < cbCmdDr - sizeof (*pCmdDr) - VBOXVDMACMD_HEADER_SIZE())
2114 {
2115 AssertMsgFailed(("invalid command buffer data!"));
2116 return VERR_INVALID_PARAMETER;
2117 }
2118
2119 pDmaCmd = VBOXVDMACBUF_DR_TAIL(pCmdDr, VBOXVDMACMD);
2120 }
2121 else if (pCmdDr->fFlags & VBOXVDMACBUF_FLAG_BUF_VRAM_OFFSET)
2122 {
2123 VBOXVIDEOOFFSET offBuf = pCmdDr->Location.offVramBuf;
2124 if (offBuf + cbDmaCmd > pVdma->pVGAState->vram_size)
2125 {
2126 AssertMsgFailed(("invalid command buffer data from offset!"));
2127 return VERR_INVALID_PARAMETER;
2128 }
2129 pDmaCmd = (VBOXVDMACMD*)(pvRam + offBuf);
2130 }
2131
2132 if (pDmaCmd)
2133 {
2134 Assert(cbDmaCmd >= VBOXVDMACMD_HEADER_SIZE());
2135 uint32_t cbBody = VBOXVDMACMD_BODY_SIZE(cbDmaCmd);
2136
2137 switch (pDmaCmd->enmType)
2138 {
2139 case VBOXVDMACMD_TYPE_CHROMIUM_CMD:
2140 {
2141 PVBOXVDMACMD_CHROMIUM_CMD pCrCmd = VBOXVDMACMD_BODY(pDmaCmd, VBOXVDMACMD_CHROMIUM_CMD);
2142 if (cbBody < sizeof (*pCrCmd))
2143 {
2144 AssertMsgFailed(("invalid chromium command buffer size!"));
2145 return VERR_INVALID_PARAMETER;
2146 }
2147 PVGASTATE pVGAState = pVdma->pVGAState;
2148 rc = VINF_SUCCESS;
2149 if (pVGAState->pDrv->pfnCrHgsmiCommandProcess)
2150 {
2151 VBoxSHGSMICommandMarkAsynchCompletion(pCmdDr);
2152 pVGAState->pDrv->pfnCrHgsmiCommandProcess(pVGAState->pDrv, pCrCmd, cbBody);
2153 break;
2154 }
2155 else
2156 {
2157 Assert(0);
2158 }
2159
2160 int tmpRc = VBoxSHGSMICommandComplete (pVdma->pHgsmi, pCmdDr);
2161 AssertRC(tmpRc);
2162 break;
2163 }
2164 case VBOXVDMACMD_TYPE_DMA_BPB_TRANSFER:
2165 {
2166 PVBOXVDMACMD_DMA_BPB_TRANSFER pTransfer = VBOXVDMACMD_BODY(pDmaCmd, VBOXVDMACMD_DMA_BPB_TRANSFER);
2167 if (cbBody < sizeof (*pTransfer))
2168 {
2169 AssertMsgFailed(("invalid bpb transfer buffer size!"));
2170 return VERR_INVALID_PARAMETER;
2171 }
2172
2173 rc = vboxVDMACmdExecBpbTransfer(pVdma, pTransfer, sizeof (*pTransfer));
2174 AssertRC(rc);
2175 if (RT_SUCCESS(rc))
2176 {
2177 pCmdDr->rc = VINF_SUCCESS;
2178 rc = VBoxSHGSMICommandComplete (pVdma->pHgsmi, pCmdDr);
2179 AssertRC(rc);
2180 rc = VINF_SUCCESS;
2181 }
2182 break;
2183 }
2184 default:
2185 break;
2186 }
2187 }
2188 return rc;
2189}
2190
2191int vboxVDMACrHgsmiCommandCompleteAsync(PPDMIDISPLAYVBVACALLBACKS pInterface, PVBOXVDMACMD_CHROMIUM_CMD pCmd, int rc)
2192{
2193 PVGASTATE pVGAState = PPDMIDISPLAYVBVACALLBACKS_2_PVGASTATE(pInterface);
2194 PHGSMIINSTANCE pIns = pVGAState->pHGSMI;
2195 VBOXVDMACMD *pDmaHdr = VBOXVDMACMD_FROM_BODY(pCmd);
2196 VBOXVDMACBUF_DR *pDr = VBOXVDMACBUF_DR_FROM_TAIL(pDmaHdr);
2197 AssertRC(rc);
2198 pDr->rc = rc;
2199
2200 Assert(pVGAState->fGuestCaps & VBVACAPS_COMPLETEGCMD_BY_IOREAD);
2201 rc = VBoxSHGSMICommandComplete(pIns, pDr);
2202 AssertRC(rc);
2203 return rc;
2204}
2205
2206int vboxVDMACrHgsmiControlCompleteAsync(PPDMIDISPLAYVBVACALLBACKS pInterface, PVBOXVDMACMD_CHROMIUM_CTL pCmd, int rc)
2207{
2208 PVGASTATE pVGAState = PPDMIDISPLAYVBVACALLBACKS_2_PVGASTATE(pInterface);
2209 PVBOXVDMACMD_CHROMIUM_CTL_PRIVATE pCmdPrivate = VBOXVDMACMD_CHROMIUM_CTL_PRIVATE_FROM_CTL(pCmd);
2210 pCmdPrivate->rc = rc;
2211 if (pCmdPrivate->pfnCompletion)
2212 {
2213 pCmdPrivate->pfnCompletion(pVGAState, pCmd, pCmdPrivate->pvCompletion);
2214 }
2215 return VINF_SUCCESS;
2216}
2217
2218static int vboxVDMACmdExecBltPerform(PVBOXVDMAHOST pVdma,
2219 uint8_t *pvDstSurf, const uint8_t *pvSrcSurf,
2220 const PVBOXVDMA_SURF_DESC pDstDesc, const PVBOXVDMA_SURF_DESC pSrcDesc,
2221 const VBOXVDMA_RECTL * pDstRectl, const VBOXVDMA_RECTL * pSrcRectl)
2222{
2223 /* we do not support color conversion */
2224 Assert(pDstDesc->format == pSrcDesc->format);
2225 /* we do not support stretching */
2226 Assert(pDstRectl->height == pSrcRectl->height);
2227 Assert(pDstRectl->width == pSrcRectl->width);
2228 if (pDstDesc->format != pSrcDesc->format)
2229 return VERR_INVALID_FUNCTION;
2230 if (pDstDesc->width == pDstRectl->width
2231 && pSrcDesc->width == pSrcRectl->width
2232 && pSrcDesc->width == pDstDesc->width)
2233 {
2234 Assert(!pDstRectl->left);
2235 Assert(!pSrcRectl->left);
2236 uint32_t cbOff = pDstDesc->pitch * pDstRectl->top;
2237 uint32_t cbSize = pDstDesc->pitch * pDstRectl->height;
2238 memcpy(pvDstSurf + cbOff, pvSrcSurf + cbOff, cbSize);
2239 }
2240 else
2241 {
2242 uint32_t offDstLineStart = pDstRectl->left * pDstDesc->bpp >> 3;
2243 uint32_t offDstLineEnd = ((pDstRectl->left * pDstDesc->bpp + 7) >> 3) + ((pDstDesc->bpp * pDstRectl->width + 7) >> 3);
2244 uint32_t cbDstLine = offDstLineEnd - offDstLineStart;
2245 uint32_t offDstStart = pDstDesc->pitch * pDstRectl->top + offDstLineStart;
2246 Assert(cbDstLine <= pDstDesc->pitch);
2247 uint32_t cbDstSkip = pDstDesc->pitch;
2248 uint8_t * pvDstStart = pvDstSurf + offDstStart;
2249
2250 uint32_t offSrcLineStart = pSrcRectl->left * pSrcDesc->bpp >> 3;
2251 uint32_t offSrcLineEnd = ((pSrcRectl->left * pSrcDesc->bpp + 7) >> 3) + ((pSrcDesc->bpp * pSrcRectl->width + 7) >> 3);
2252 uint32_t cbSrcLine = offSrcLineEnd - offSrcLineStart;
2253 uint32_t offSrcStart = pSrcDesc->pitch * pSrcRectl->top + offSrcLineStart;
2254 Assert(cbSrcLine <= pSrcDesc->pitch);
2255 uint32_t cbSrcSkip = pSrcDesc->pitch;
2256 const uint8_t * pvSrcStart = pvSrcSurf + offSrcStart;
2257
2258 Assert(cbDstLine == cbSrcLine);
2259
2260 for (uint32_t i = 0; ; ++i)
2261 {
2262 memcpy (pvDstStart, pvSrcStart, cbDstLine);
2263 if (i == pDstRectl->height)
2264 break;
2265 pvDstStart += cbDstSkip;
2266 pvSrcStart += cbSrcSkip;
2267 }
2268 }
2269 return VINF_SUCCESS;
2270}
2271
2272static void vboxVDMARectlUnite(VBOXVDMA_RECTL * pRectl1, const VBOXVDMA_RECTL * pRectl2)
2273{
2274 if (!pRectl1->width)
2275 *pRectl1 = *pRectl2;
2276 else
2277 {
2278 int16_t x21 = pRectl1->left + pRectl1->width;
2279 int16_t x22 = pRectl2->left + pRectl2->width;
2280 if (pRectl1->left > pRectl2->left)
2281 {
2282 pRectl1->left = pRectl2->left;
2283 pRectl1->width = x21 < x22 ? x22 - pRectl1->left : x21 - pRectl1->left;
2284 }
2285 else if (x21 < x22)
2286 pRectl1->width = x22 - pRectl1->left;
2287
2288 x21 = pRectl1->top + pRectl1->height;
2289 x22 = pRectl2->top + pRectl2->height;
2290 if (pRectl1->top > pRectl2->top)
2291 {
2292 pRectl1->top = pRectl2->top;
2293 pRectl1->height = x21 < x22 ? x22 - pRectl1->top : x21 - pRectl1->top;
2294 }
2295 else if (x21 < x22)
2296 pRectl1->height = x22 - pRectl1->top;
2297 }
2298}
2299
2300/*
2301 * @return on success the number of bytes the command contained, otherwise - VERR_xxx error code
2302 */
2303static int vboxVDMACmdExecBlt(PVBOXVDMAHOST pVdma, const PVBOXVDMACMD_DMA_PRESENT_BLT pBlt, uint32_t cbBuffer)
2304{
2305 const uint32_t cbBlt = VBOXVDMACMD_BODY_FIELD_OFFSET(uint32_t, VBOXVDMACMD_DMA_PRESENT_BLT, aDstSubRects[pBlt->cDstSubRects]);
2306 Assert(cbBlt <= cbBuffer);
2307 if (cbBuffer < cbBlt)
2308 return VERR_INVALID_FUNCTION;
2309
2310 /* we do not support stretching for now */
2311 Assert(pBlt->srcRectl.width == pBlt->dstRectl.width);
2312 Assert(pBlt->srcRectl.height == pBlt->dstRectl.height);
2313 if (pBlt->srcRectl.width != pBlt->dstRectl.width)
2314 return VERR_INVALID_FUNCTION;
2315 if (pBlt->srcRectl.height != pBlt->dstRectl.height)
2316 return VERR_INVALID_FUNCTION;
2317 Assert(pBlt->cDstSubRects);
2318
2319 uint8_t * pvRam = pVdma->pVGAState->vram_ptrR3;
2320 VBOXVDMA_RECTL updateRectl = {0, 0, 0, 0};
2321
2322 if (pBlt->cDstSubRects)
2323 {
2324 VBOXVDMA_RECTL dstRectl, srcRectl;
2325 const VBOXVDMA_RECTL *pDstRectl, *pSrcRectl;
2326 for (uint32_t i = 0; i < pBlt->cDstSubRects; ++i)
2327 {
2328 pDstRectl = &pBlt->aDstSubRects[i];
2329 if (pBlt->dstRectl.left || pBlt->dstRectl.top)
2330 {
2331 dstRectl.left = pDstRectl->left + pBlt->dstRectl.left;
2332 dstRectl.top = pDstRectl->top + pBlt->dstRectl.top;
2333 dstRectl.width = pDstRectl->width;
2334 dstRectl.height = pDstRectl->height;
2335 pDstRectl = &dstRectl;
2336 }
2337
2338 pSrcRectl = &pBlt->aDstSubRects[i];
2339 if (pBlt->srcRectl.left || pBlt->srcRectl.top)
2340 {
2341 srcRectl.left = pSrcRectl->left + pBlt->srcRectl.left;
2342 srcRectl.top = pSrcRectl->top + pBlt->srcRectl.top;
2343 srcRectl.width = pSrcRectl->width;
2344 srcRectl.height = pSrcRectl->height;
2345 pSrcRectl = &srcRectl;
2346 }
2347
2348 int rc = vboxVDMACmdExecBltPerform(pVdma, pvRam + pBlt->offDst, pvRam + pBlt->offSrc,
2349 &pBlt->dstDesc, &pBlt->srcDesc,
2350 pDstRectl,
2351 pSrcRectl);
2352 AssertRC(rc);
2353 if (!RT_SUCCESS(rc))
2354 return rc;
2355
2356 vboxVDMARectlUnite(&updateRectl, pDstRectl);
2357 }
2358 }
2359 else
2360 {
2361 int rc = vboxVDMACmdExecBltPerform(pVdma, pvRam + pBlt->offDst, pvRam + pBlt->offSrc,
2362 &pBlt->dstDesc, &pBlt->srcDesc,
2363 &pBlt->dstRectl,
2364 &pBlt->srcRectl);
2365 AssertRC(rc);
2366 if (!RT_SUCCESS(rc))
2367 return rc;
2368
2369 vboxVDMARectlUnite(&updateRectl, &pBlt->dstRectl);
2370 }
2371
2372 return cbBlt;
2373}
2374
2375static int vboxVDMACmdExecBpbTransfer(PVBOXVDMAHOST pVdma, const PVBOXVDMACMD_DMA_BPB_TRANSFER pTransfer, uint32_t cbBuffer)
2376{
2377 if (cbBuffer < sizeof (*pTransfer))
2378 return VERR_INVALID_PARAMETER;
2379
2380 PVGASTATE pVGAState = pVdma->pVGAState;
2381 uint8_t * pvRam = pVGAState->vram_ptrR3;
2382 PGMPAGEMAPLOCK SrcLock;
2383 PGMPAGEMAPLOCK DstLock;
2384 PPDMDEVINS pDevIns = pVdma->pVGAState->pDevInsR3;
2385 const void * pvSrc;
2386 void * pvDst;
2387 int rc = VINF_SUCCESS;
2388 uint32_t cbTransfer = pTransfer->cbTransferSize;
2389 uint32_t cbTransfered = 0;
2390 bool bSrcLocked = false;
2391 bool bDstLocked = false;
2392 do
2393 {
2394 uint32_t cbSubTransfer = cbTransfer;
2395 if (pTransfer->fFlags & VBOXVDMACMD_DMA_BPB_TRANSFER_F_SRC_VRAMOFFSET)
2396 {
2397 pvSrc = pvRam + pTransfer->Src.offVramBuf + cbTransfered;
2398 }
2399 else
2400 {
2401 RTGCPHYS phPage = pTransfer->Src.phBuf;
2402 phPage += cbTransfered;
2403 rc = PDMDevHlpPhysGCPhys2CCPtrReadOnly(pDevIns, phPage, 0, &pvSrc, &SrcLock);
2404 AssertRC(rc);
2405 if (RT_SUCCESS(rc))
2406 {
2407 bSrcLocked = true;
2408 cbSubTransfer = RT_MIN(cbSubTransfer, 0x1000);
2409 }
2410 else
2411 {
2412 break;
2413 }
2414 }
2415
2416 if (pTransfer->fFlags & VBOXVDMACMD_DMA_BPB_TRANSFER_F_DST_VRAMOFFSET)
2417 {
2418 pvDst = pvRam + pTransfer->Dst.offVramBuf + cbTransfered;
2419 }
2420 else
2421 {
2422 RTGCPHYS phPage = pTransfer->Dst.phBuf;
2423 phPage += cbTransfered;
2424 rc = PDMDevHlpPhysGCPhys2CCPtr(pDevIns, phPage, 0, &pvDst, &DstLock);
2425 AssertRC(rc);
2426 if (RT_SUCCESS(rc))
2427 {
2428 bDstLocked = true;
2429 cbSubTransfer = RT_MIN(cbSubTransfer, 0x1000);
2430 }
2431 else
2432 {
2433 break;
2434 }
2435 }
2436
2437 if (RT_SUCCESS(rc))
2438 {
2439 memcpy(pvDst, pvSrc, cbSubTransfer);
2440 cbTransfer -= cbSubTransfer;
2441 cbTransfered += cbSubTransfer;
2442 }
2443 else
2444 {
2445 cbTransfer = 0; /* to break */
2446 }
2447
2448 if (bSrcLocked)
2449 PDMDevHlpPhysReleasePageMappingLock(pDevIns, &SrcLock);
2450 if (bDstLocked)
2451 PDMDevHlpPhysReleasePageMappingLock(pDevIns, &DstLock);
2452 } while (cbTransfer);
2453
2454 if (RT_SUCCESS(rc))
2455 return sizeof (*pTransfer);
2456 return rc;
2457}
2458
2459static int vboxVDMACmdExec(PVBOXVDMAHOST pVdma, const uint8_t *pvBuffer, uint32_t cbBuffer)
2460{
2461 do
2462 {
2463 Assert(pvBuffer);
2464 Assert(cbBuffer >= VBOXVDMACMD_HEADER_SIZE());
2465
2466 if (!pvBuffer)
2467 return VERR_INVALID_PARAMETER;
2468 if (cbBuffer < VBOXVDMACMD_HEADER_SIZE())
2469 return VERR_INVALID_PARAMETER;
2470
2471 PVBOXVDMACMD pCmd = (PVBOXVDMACMD)pvBuffer;
2472 uint32_t cbCmd = 0;
2473 switch (pCmd->enmType)
2474 {
2475 case VBOXVDMACMD_TYPE_CHROMIUM_CMD:
2476 {
2477#ifdef VBOXWDDM_TEST_UHGSMI
2478 static int count = 0;
2479 static uint64_t start, end;
2480 if (count==0)
2481 {
2482 start = RTTimeNanoTS();
2483 }
2484 ++count;
2485 if (count==100000)
2486 {
2487 end = RTTimeNanoTS();
2488 float ems = (end-start)/1000000.f;
2489 LogRel(("100000 calls took %i ms, %i cps\n", (int)ems, (int)(100000.f*1000.f/ems) ));
2490 }
2491#endif
2492 /* todo: post the buffer to chromium */
2493 return VINF_SUCCESS;
2494 }
2495 case VBOXVDMACMD_TYPE_DMA_PRESENT_BLT:
2496 {
2497 const PVBOXVDMACMD_DMA_PRESENT_BLT pBlt = VBOXVDMACMD_BODY(pCmd, VBOXVDMACMD_DMA_PRESENT_BLT);
2498 int cbBlt = vboxVDMACmdExecBlt(pVdma, pBlt, cbBuffer);
2499 Assert(cbBlt >= 0);
2500 Assert((uint32_t)cbBlt <= cbBuffer);
2501 if (cbBlt >= 0)
2502 {
2503 if ((uint32_t)cbBlt == cbBuffer)
2504 return VINF_SUCCESS;
2505 else
2506 {
2507 cbBuffer -= (uint32_t)cbBlt;
2508 pvBuffer -= cbBlt;
2509 }
2510 }
2511 else
2512 return cbBlt; /* error */
2513 break;
2514 }
2515 case VBOXVDMACMD_TYPE_DMA_BPB_TRANSFER:
2516 {
2517 const PVBOXVDMACMD_DMA_BPB_TRANSFER pTransfer = VBOXVDMACMD_BODY(pCmd, VBOXVDMACMD_DMA_BPB_TRANSFER);
2518 int cbTransfer = vboxVDMACmdExecBpbTransfer(pVdma, pTransfer, cbBuffer);
2519 Assert(cbTransfer >= 0);
2520 Assert((uint32_t)cbTransfer <= cbBuffer);
2521 if (cbTransfer >= 0)
2522 {
2523 if ((uint32_t)cbTransfer == cbBuffer)
2524 return VINF_SUCCESS;
2525 else
2526 {
2527 cbBuffer -= (uint32_t)cbTransfer;
2528 pvBuffer -= cbTransfer;
2529 }
2530 }
2531 else
2532 return cbTransfer; /* error */
2533 break;
2534 }
2535 case VBOXVDMACMD_TYPE_DMA_NOP:
2536 return VINF_SUCCESS;
2537 case VBOXVDMACMD_TYPE_CHILD_STATUS_IRQ:
2538 return VINF_SUCCESS;
2539 default:
2540 AssertBreakpoint();
2541 return VERR_INVALID_FUNCTION;
2542 }
2543 } while (1);
2544
2545 /* we should not be here */
2546 AssertBreakpoint();
2547 return VERR_INVALID_STATE;
2548}
2549
2550static DECLCALLBACK(int) vboxVDMAWorkerThread(RTTHREAD ThreadSelf, void *pvUser)
2551{
2552 PVBOXVDMAHOST pVdma = (PVBOXVDMAHOST)pvUser;
2553 PVGASTATE pVGAState = pVdma->pVGAState;
2554 VBVAEXHOSTCONTEXT *pCmdVbva = &pVdma->CmdVbva;
2555 PHGSMIINSTANCE pHgsmi = pVdma->pHgsmi;
2556 uint8_t *pCmd;
2557 uint32_t cbCmd;
2558 int rc;
2559
2560 VBoxVDMAThreadNotifyConstructSucceeded(&pVdma->Thread, pvUser);
2561
2562 while (!VBoxVDMAThreadIsTerminating(&pVdma->Thread))
2563 {
2564 VBVAEXHOST_DATA_TYPE enmType = VBoxVBVAExHPDataGet(pCmdVbva, &pCmd, &cbCmd);
2565 switch (enmType)
2566 {
2567 case VBVAEXHOST_DATA_TYPE_CMD:
2568 vboxVDMACrCmdProcess(pVdma, pCmd, cbCmd);
2569 VBoxVBVAExHPDataCompleteCmd(pCmdVbva, cbCmd);
2570 VBVARaiseIrqNoWait(pVGAState, 0);
2571 break;
2572 case VBVAEXHOST_DATA_TYPE_GUESTCTL:
2573 rc = vboxVDMACrGuestCtlProcess(pVdma, (VBVAEXHOSTCTL*)pCmd);
2574 VBoxVBVAExHPDataCompleteCtl(pCmdVbva, (VBVAEXHOSTCTL*)pCmd, rc);
2575 break;
2576 case VBVAEXHOST_DATA_TYPE_HOSTCTL:
2577 {
2578 bool fContinue = true;
2579 rc = vboxVDMACrHostCtlProcess(pVdma, (VBVAEXHOSTCTL*)pCmd, &fContinue);
2580 VBoxVBVAExHPDataCompleteCtl(pCmdVbva, (VBVAEXHOSTCTL*)pCmd, rc);
2581 if (fContinue)
2582 break;
2583 }
2584 case VBVAEXHOST_DATA_TYPE_NO_DATA:
2585 rc = VBoxVDMAThreadEventWait(&pVdma->Thread, RT_INDEFINITE_WAIT);
2586 AssertRC(rc);
2587 break;
2588 default:
2589 WARN(("unexpected type %d\n", enmType));
2590 break;
2591 }
2592 }
2593
2594 VBoxVDMAThreadNotifyTerminatingSucceeded(&pVdma->Thread, pvUser);
2595
2596 return VINF_SUCCESS;
2597}
2598
2599static void vboxVDMACommandProcess(PVBOXVDMAHOST pVdma, PVBOXVDMACBUF_DR pCmd, uint32_t cbCmd)
2600{
2601 PHGSMIINSTANCE pHgsmi = pVdma->pHgsmi;
2602 const uint8_t * pvBuf;
2603 PGMPAGEMAPLOCK Lock;
2604 int rc;
2605 bool bReleaseLocked = false;
2606
2607 do
2608 {
2609 PPDMDEVINS pDevIns = pVdma->pVGAState->pDevInsR3;
2610
2611 if (pCmd->fFlags & VBOXVDMACBUF_FLAG_BUF_FOLLOWS_DR)
2612 pvBuf = VBOXVDMACBUF_DR_TAIL(pCmd, const uint8_t);
2613 else if (pCmd->fFlags & VBOXVDMACBUF_FLAG_BUF_VRAM_OFFSET)
2614 {
2615 uint8_t * pvRam = pVdma->pVGAState->vram_ptrR3;
2616 pvBuf = pvRam + pCmd->Location.offVramBuf;
2617 }
2618 else
2619 {
2620 RTGCPHYS phPage = pCmd->Location.phBuf & ~0xfffULL;
2621 uint32_t offset = pCmd->Location.phBuf & 0xfff;
2622 Assert(offset + pCmd->cbBuf <= 0x1000);
2623 if (offset + pCmd->cbBuf > 0x1000)
2624 {
2625 /* @todo: more advanced mechanism of command buffer proc is actually needed */
2626 rc = VERR_INVALID_PARAMETER;
2627 break;
2628 }
2629
2630 const void * pvPageBuf;
2631 rc = PDMDevHlpPhysGCPhys2CCPtrReadOnly(pDevIns, phPage, 0, &pvPageBuf, &Lock);
2632 AssertRC(rc);
2633 if (!RT_SUCCESS(rc))
2634 {
2635 /* @todo: if (rc == VERR_PGM_PHYS_PAGE_RESERVED) -> fall back on using PGMPhysRead ?? */
2636 break;
2637 }
2638
2639 pvBuf = (const uint8_t *)pvPageBuf;
2640 pvBuf += offset;
2641
2642 bReleaseLocked = true;
2643 }
2644
2645 rc = vboxVDMACmdExec(pVdma, pvBuf, pCmd->cbBuf);
2646 AssertRC(rc);
2647
2648 if (bReleaseLocked)
2649 PDMDevHlpPhysReleasePageMappingLock(pDevIns, &Lock);
2650 } while (0);
2651
2652 pCmd->rc = rc;
2653
2654 rc = VBoxSHGSMICommandComplete (pHgsmi, pCmd);
2655 AssertRC(rc);
2656}
2657
2658static void vboxVDMAControlProcess(PVBOXVDMAHOST pVdma, PVBOXVDMA_CTL pCmd)
2659{
2660 PHGSMIINSTANCE pHgsmi = pVdma->pHgsmi;
2661 pCmd->i32Result = VINF_SUCCESS;
2662 int rc = VBoxSHGSMICommandComplete (pHgsmi, pCmd);
2663 AssertRC(rc);
2664}
2665
2666#endif /* #ifdef VBOX_WITH_CRHGSMI */
2667
2668#ifdef VBOX_VDMA_WITH_WATCHDOG
2669static DECLCALLBACK(void) vboxVDMAWatchDogTimer(PPDMDEVINS pDevIns, PTMTIMER pTimer, void *pvUser)
2670{
2671 VBOXVDMAHOST *pVdma = (VBOXVDMAHOST *)pvUser;
2672 PVGASTATE pVGAState = pVdma->pVGAState;
2673 VBVARaiseIrq(pVGAState, HGSMIHOSTFLAGS_WATCHDOG);
2674}
2675
2676static int vboxVDMAWatchDogCtl(struct VBOXVDMAHOST *pVdma, uint32_t cMillis)
2677{
2678 PPDMDEVINS pDevIns = pVdma->pVGAState->pDevInsR3;
2679 if (cMillis)
2680 TMTimerSetMillies(pVdma->WatchDogTimer, cMillis);
2681 else
2682 TMTimerStop(pVdma->WatchDogTimer);
2683 return VINF_SUCCESS;
2684}
2685#endif
2686
2687int vboxVDMAConstruct(PVGASTATE pVGAState, uint32_t cPipeElements)
2688{
2689 int rc;
2690 PVBOXVDMAHOST pVdma = (PVBOXVDMAHOST)RTMemAllocZ(sizeof(*pVdma));
2691 Assert(pVdma);
2692 if (pVdma)
2693 {
2694 pVdma->pHgsmi = pVGAState->pHGSMI;
2695 pVdma->pVGAState = pVGAState;
2696
2697#ifdef VBOX_VDMA_WITH_WATCHDOG
2698 rc = PDMDevHlpTMTimerCreate(pVGAState->pDevInsR3, TMCLOCK_REAL, vboxVDMAWatchDogTimer,
2699 pVdma, TMTIMER_FLAGS_NO_CRIT_SECT,
2700 "VDMA WatchDog Timer", &pVdma->WatchDogTimer);
2701 AssertRC(rc);
2702#endif
2703
2704#ifdef VBOX_WITH_CRHGSMI
2705 VBoxVDMAThreadInit(&pVdma->Thread);
2706
2707 rc = RTSemEventMultiCreate(&pVdma->HostCrCtlCompleteEvent);
2708 if (RT_SUCCESS(rc))
2709 {
2710 rc = VBoxVBVAExHSInit(&pVdma->CmdVbva);
2711 if (RT_SUCCESS(rc))
2712 {
2713 rc = RTCritSectInit(&pVdma->CalloutCritSect);
2714 if (RT_SUCCESS(rc))
2715 {
2716 pVGAState->pVdma = pVdma;
2717 int rcIgnored = vboxVDMACrCtlHgsmiSetup(pVdma); NOREF(rcIgnored); /** @todo is this ignoring intentional? */
2718 return VINF_SUCCESS;
2719
2720 RTCritSectDelete(&pVdma->CalloutCritSect);
2721 }
2722 else
2723 WARN(("RTCritSectInit failed %d\n", rc));
2724
2725 VBoxVBVAExHSTerm(&pVdma->CmdVbva);
2726 }
2727 else
2728 WARN(("VBoxVBVAExHSInit failed %d\n", rc));
2729
2730 RTSemEventMultiDestroy(pVdma->HostCrCtlCompleteEvent);
2731 }
2732 else
2733 WARN(("RTSemEventMultiCreate failed %d\n", rc));
2734
2735
2736 RTMemFree(pVdma);
2737#else
2738 pVGAState->pVdma = pVdma;
2739 return VINF_SUCCESS;
2740#endif
2741 }
2742 else
2743 rc = VERR_OUT_OF_RESOURCES;
2744
2745 return rc;
2746}
2747
2748int vboxVDMAReset(struct VBOXVDMAHOST *pVdma)
2749{
2750#ifdef VBOX_WITH_CRHGSMI
2751 vdmaVBVACtlDisableSync(pVdma);
2752#endif
2753 return VINF_SUCCESS;
2754}
2755
2756int vboxVDMADestruct(struct VBOXVDMAHOST *pVdma)
2757{
2758 if (!pVdma)
2759 return VINF_SUCCESS;
2760#ifdef VBOX_WITH_CRHGSMI
2761 vdmaVBVACtlDisableSync(pVdma);
2762 VBoxVDMAThreadCleanup(&pVdma->Thread);
2763 VBoxVBVAExHSTerm(&pVdma->CmdVbva);
2764 RTSemEventMultiDestroy(pVdma->HostCrCtlCompleteEvent);
2765 RTCritSectDelete(&pVdma->CalloutCritSect);
2766#endif
2767 RTMemFree(pVdma);
2768 return VINF_SUCCESS;
2769}
2770
2771void vboxVDMAControl(struct VBOXVDMAHOST *pVdma, PVBOXVDMA_CTL pCmd, uint32_t cbCmd)
2772{
2773 PHGSMIINSTANCE pIns = pVdma->pHgsmi;
2774
2775 switch (pCmd->enmCtl)
2776 {
2777 case VBOXVDMA_CTL_TYPE_ENABLE:
2778 pCmd->i32Result = VINF_SUCCESS;
2779 break;
2780 case VBOXVDMA_CTL_TYPE_DISABLE:
2781 pCmd->i32Result = VINF_SUCCESS;
2782 break;
2783 case VBOXVDMA_CTL_TYPE_FLUSH:
2784 pCmd->i32Result = VINF_SUCCESS;
2785 break;
2786#ifdef VBOX_VDMA_WITH_WATCHDOG
2787 case VBOXVDMA_CTL_TYPE_WATCHDOG:
2788 pCmd->i32Result = vboxVDMAWatchDogCtl(pVdma, pCmd->u32Offset);
2789 break;
2790#endif
2791 default:
2792 WARN(("cmd not supported"));
2793 pCmd->i32Result = VERR_NOT_SUPPORTED;
2794 }
2795
2796 int rc = VBoxSHGSMICommandComplete (pIns, pCmd);
2797 AssertRC(rc);
2798}
2799
2800void vboxVDMACommand(struct VBOXVDMAHOST *pVdma, PVBOXVDMACBUF_DR pCmd, uint32_t cbCmd)
2801{
2802 int rc = VERR_NOT_IMPLEMENTED;
2803
2804#ifdef VBOX_WITH_CRHGSMI
2805 /* chromium commands are processed by crhomium hgcm thread independently from our internal cmd processing pipeline
2806 * this is why we process them specially */
2807 rc = vboxVDMACmdCheckCrCmd(pVdma, pCmd, cbCmd);
2808 if (rc == VINF_SUCCESS)
2809 return;
2810
2811 if (RT_FAILURE(rc))
2812 {
2813 pCmd->rc = rc;
2814 rc = VBoxSHGSMICommandComplete (pVdma->pHgsmi, pCmd);
2815 AssertRC(rc);
2816 return;
2817 }
2818
2819 vboxVDMACommandProcess(pVdma, pCmd, cbCmd);
2820#else
2821 pCmd->rc = rc;
2822 rc = VBoxSHGSMICommandComplete (pVdma->pHgsmi, pCmd);
2823 AssertRC(rc);
2824#endif
2825}
2826
2827/**/
2828#ifdef VBOX_WITH_CRHGSMI
2829
2830static DECLCALLBACK(void) vdmaVBVACtlSubmitSyncCompletion(VBVAEXHOSTCONTEXT *pVbva, struct VBVAEXHOSTCTL *pCtl, int rc, void *pvContext);
2831
2832static int vdmaVBVACtlSubmit(PVBOXVDMAHOST pVdma, VBVAEXHOSTCTL* pCtl, VBVAEXHOSTCTL_SOURCE enmSource, PFNVBVAEXHOSTCTL_COMPLETE pfnComplete, void *pvComplete)
2833{
2834 int rc = VBoxVBVAExHCtlSubmit(&pVdma->CmdVbva, pCtl, enmSource, pfnComplete, pvComplete);
2835 if (RT_SUCCESS(rc))
2836 {
2837 if (rc == VINF_SUCCESS)
2838 return VBoxVDMAThreadEventNotify(&pVdma->Thread);
2839 else
2840 Assert(rc == VINF_ALREADY_INITIALIZED);
2841 }
2842 else
2843 Log(("VBoxVBVAExHCtlSubmit failed %d\n", rc));
2844
2845 return rc;
2846}
2847
2848static DECLCALLBACK(void) vboxCmdVBVACmdCtlGuestCompletion(VBVAEXHOSTCONTEXT *pVbva, struct VBVAEXHOSTCTL *pCtl, int rc, void *pvContext)
2849{
2850 PVBOXVDMAHOST pVdma = (PVBOXVDMAHOST)pvContext;
2851 VBOXCMDVBVA_CTL *pGCtl = (VBOXCMDVBVA_CTL*)(pCtl->u.cmd.pu8Cmd - sizeof (VBOXCMDVBVA_CTL));
2852 AssertRC(rc);
2853 pGCtl->i32Result = rc;
2854
2855 Assert(pVdma->pVGAState->fGuestCaps & VBVACAPS_COMPLETEGCMD_BY_IOREAD);
2856 rc = VBoxSHGSMICommandComplete(pVdma->pHgsmi, pGCtl);
2857 AssertRC(rc);
2858
2859 VBoxVBVAExHCtlFree(pVbva, pCtl);
2860}
2861
2862static int vdmaVBVACtlGenericSubmit(PVBOXVDMAHOST pVdma, VBVAEXHOSTCTL_SOURCE enmSource, VBVAEXHOSTCTL_TYPE enmType, uint8_t* pu8Cmd, uint32_t cbCmd, PFNVBVAEXHOSTCTL_COMPLETE pfnComplete, void *pvComplete)
2863{
2864 VBVAEXHOSTCTL* pHCtl = VBoxVBVAExHCtlCreate(&pVdma->CmdVbva, enmType);
2865 if (!pHCtl)
2866 {
2867 WARN(("VBoxVBVAExHCtlCreate failed\n"));
2868 return VERR_NO_MEMORY;
2869 }
2870
2871 pHCtl->u.cmd.pu8Cmd = pu8Cmd;
2872 pHCtl->u.cmd.cbCmd = cbCmd;
2873 int rc = vdmaVBVACtlSubmit(pVdma, pHCtl, enmSource, pfnComplete, pvComplete);
2874 if (RT_FAILURE(rc))
2875 {
2876 VBoxVBVAExHCtlFree(&pVdma->CmdVbva, pHCtl);
2877 Log(("vdmaVBVACtlSubmit failed rc %d\n", rc));
2878 return rc;;
2879 }
2880 return VINF_SUCCESS;
2881}
2882
2883static int vdmaVBVACtlGenericGuestSubmit(PVBOXVDMAHOST pVdma, VBVAEXHOSTCTL_TYPE enmType, VBOXCMDVBVA_CTL *pCtl, uint32_t cbCtl)
2884{
2885 Assert(cbCtl >= sizeof (VBOXCMDVBVA_CTL));
2886 VBoxSHGSMICommandMarkAsynchCompletion(pCtl);
2887 int rc = vdmaVBVACtlGenericSubmit(pVdma, VBVAEXHOSTCTL_SOURCE_GUEST, enmType, (uint8_t*)(pCtl+1), cbCtl - sizeof (VBOXCMDVBVA_CTL), vboxCmdVBVACmdCtlGuestCompletion, pVdma);
2888 if (RT_SUCCESS(rc))
2889 return VINF_SUCCESS;
2890
2891 WARN(("vdmaVBVACtlGenericSubmit failed %d\n", rc));
2892 pCtl->i32Result = rc;
2893 rc = VBoxSHGSMICommandComplete(pVdma->pHgsmi, pCtl);
2894 AssertRC(rc);
2895 return VINF_SUCCESS;
2896}
2897
2898static DECLCALLBACK(void) vboxCmdVBVACmdCtlHostCompletion(VBVAEXHOSTCONTEXT *pVbva, struct VBVAEXHOSTCTL *pCtl, int rc, void *pvCompletion)
2899{
2900 VBOXCRCMDCTL* pVboxCtl = (VBOXCRCMDCTL*)pCtl->u.cmd.pu8Cmd;
2901 if (pVboxCtl->u.pfnInternal)
2902 ((PFNCRCTLCOMPLETION)pVboxCtl->u.pfnInternal)(pVboxCtl, pCtl->u.cmd.cbCmd, rc, pvCompletion);
2903 VBoxVBVAExHCtlFree(pVbva, pCtl);
2904}
2905
2906static int vdmaVBVACtlOpaqueHostSubmit(PVBOXVDMAHOST pVdma, struct VBOXCRCMDCTL* pCmd, uint32_t cbCmd,
2907 PFNCRCTLCOMPLETION pfnCompletion,
2908 void *pvCompletion)
2909{
2910 pCmd->u.pfnInternal = (void(*)())pfnCompletion;
2911 int rc = vdmaVBVACtlGenericSubmit(pVdma, VBVAEXHOSTCTL_SOURCE_HOST, VBVAEXHOSTCTL_TYPE_GHH_BE_OPAQUE, (uint8_t*)pCmd, cbCmd, vboxCmdVBVACmdCtlHostCompletion, pvCompletion);
2912 if (RT_FAILURE(rc))
2913 {
2914 if (rc == VERR_INVALID_STATE)
2915 {
2916 pCmd->u.pfnInternal = NULL;
2917 PVGASTATE pVGAState = pVdma->pVGAState;
2918 rc = pVGAState->pDrv->pfnCrHgcmCtlSubmit(pVGAState->pDrv, pCmd, cbCmd, pfnCompletion, pvCompletion);
2919 if (!RT_SUCCESS(rc))
2920 WARN(("pfnCrHgsmiControlProcess failed %d\n", rc));
2921
2922 return rc;
2923 }
2924 WARN(("vdmaVBVACtlGenericSubmit failed %d\n", rc));
2925 return rc;
2926 }
2927
2928 return VINF_SUCCESS;
2929}
2930
2931static DECLCALLBACK(int) vdmaVBVANotifyEnable(PVGASTATE pVGAState)
2932{
2933 for (uint32_t i = 0; i < pVGAState->cMonitors; i++)
2934 {
2935 int rc = pVGAState->pDrv->pfnVBVAEnable (pVGAState->pDrv, i, NULL, true);
2936 if (!RT_SUCCESS(rc))
2937 {
2938 WARN(("pfnVBVAEnable failed %d\n", rc));
2939 for (uint32_t j = 0; j < i; j++)
2940 {
2941 pVGAState->pDrv->pfnVBVADisable (pVGAState->pDrv, j);
2942 }
2943
2944 return rc;
2945 }
2946 }
2947 return VINF_SUCCESS;
2948}
2949
2950static DECLCALLBACK(int) vdmaVBVANotifyDisable(PVGASTATE pVGAState)
2951{
2952 for (uint32_t i = 0; i < pVGAState->cMonitors; i++)
2953 {
2954 pVGAState->pDrv->pfnVBVADisable (pVGAState->pDrv, i);
2955 }
2956 return VINF_SUCCESS;
2957}
2958
2959static DECLCALLBACK(void) vdmaVBVACtlThreadCreatedEnable(struct VBOXVDMATHREAD *pThread, int rc, void *pvThreadContext, void *pvContext)
2960{
2961 PVBOXVDMAHOST pVdma = (PVBOXVDMAHOST)pvThreadContext;
2962 VBVAEXHOSTCTL* pHCtl = (VBVAEXHOSTCTL*)pvContext;
2963
2964 if (RT_SUCCESS(rc))
2965 {
2966 rc = vboxVDMACrGuestCtlProcess(pVdma, pHCtl);
2967 /* rc == VINF_SUCCESS would mean the actual state change has occcured */
2968 if (rc == VINF_SUCCESS)
2969 {
2970 /* we need to inform Main about VBVA enable/disable
2971 * main expects notifications to be done from the main thread
2972 * submit it there */
2973 PVGASTATE pVGAState = pVdma->pVGAState;
2974
2975 if (VBoxVBVAExHSIsEnabled(&pVdma->CmdVbva))
2976 vdmaVBVANotifyEnable(pVGAState);
2977 else
2978 vdmaVBVANotifyDisable(pVGAState);
2979 }
2980 else if (RT_FAILURE(rc))
2981 WARN(("vboxVDMACrGuestCtlProcess failed %d\n", rc));
2982 }
2983 else
2984 WARN(("vdmaVBVACtlThreadCreatedEnable is passed %d\n", rc));
2985
2986 VBoxVBVAExHPDataCompleteCtl(&pVdma->CmdVbva, pHCtl, rc);
2987}
2988
2989static int vdmaVBVACtlEnableSubmitInternal(PVBOXVDMAHOST pVdma, VBVAENABLE *pEnable, bool fPaused, PFNVBVAEXHOSTCTL_COMPLETE pfnComplete, void *pvComplete)
2990{
2991 int rc;
2992 VBVAEXHOSTCTL* pHCtl = VBoxVBVAExHCtlCreate(&pVdma->CmdVbva, fPaused ? VBVAEXHOSTCTL_TYPE_GHH_ENABLE_PAUSED : VBVAEXHOSTCTL_TYPE_GHH_ENABLE);
2993 if (pHCtl)
2994 {
2995 pHCtl->u.cmd.pu8Cmd = (uint8_t*)pEnable;
2996 pHCtl->u.cmd.cbCmd = sizeof (*pEnable);
2997 pHCtl->pfnComplete = pfnComplete;
2998 pHCtl->pvComplete = pvComplete;
2999
3000 rc = VBoxVDMAThreadCreate(&pVdma->Thread, vboxVDMAWorkerThread, pVdma, vdmaVBVACtlThreadCreatedEnable, pHCtl);
3001 if (RT_SUCCESS(rc))
3002 return VINF_SUCCESS;
3003 else
3004 WARN(("VBoxVDMAThreadCreate failed %d\n", rc));
3005
3006 VBoxVBVAExHCtlFree(&pVdma->CmdVbva, pHCtl);
3007 }
3008 else
3009 {
3010 WARN(("VBoxVBVAExHCtlCreate failed\n"));
3011 rc = VERR_NO_MEMORY;
3012 }
3013
3014 return rc;
3015}
3016
3017static int vdmaVBVACtlEnableSubmitSync(PVBOXVDMAHOST pVdma, uint32_t offVram, bool fPaused)
3018{
3019 VBVAENABLE Enable = {0};
3020 Enable.u32Flags = VBVA_F_ENABLE;
3021 Enable.u32Offset = offVram;
3022
3023 VDMA_VBVA_CTL_CYNC_COMPLETION Data;
3024 Data.rc = VERR_NOT_IMPLEMENTED;
3025 int rc = RTSemEventCreate(&Data.hEvent);
3026 if (!RT_SUCCESS(rc))
3027 {
3028 WARN(("RTSemEventCreate failed %d\n", rc));
3029 return rc;
3030 }
3031
3032 rc = vdmaVBVACtlEnableSubmitInternal(pVdma, &Enable, fPaused, vdmaVBVACtlSubmitSyncCompletion, &Data);
3033 if (RT_SUCCESS(rc))
3034 {
3035 rc = RTSemEventWait(Data.hEvent, RT_INDEFINITE_WAIT);
3036 if (RT_SUCCESS(rc))
3037 {
3038 rc = Data.rc;
3039 if (!RT_SUCCESS(rc))
3040 WARN(("vdmaVBVACtlSubmitSyncCompletion returned %d\n", rc));
3041 }
3042 else
3043 WARN(("RTSemEventWait failed %d\n", rc));
3044 }
3045 else
3046 WARN(("vdmaVBVACtlSubmit failed %d\n", rc));
3047
3048 RTSemEventDestroy(Data.hEvent);
3049
3050 return rc;
3051}
3052
3053static int vdmaVBVACtlDisableSubmitInternal(PVBOXVDMAHOST pVdma, VBVAENABLE *pEnable, PFNVBVAEXHOSTCTL_COMPLETE pfnComplete, void *pvComplete)
3054{
3055 int rc;
3056 VBVAEXHOSTCTL* pHCtl;
3057 if (VBoxVBVAExHSIsDisabled(&pVdma->CmdVbva))
3058 {
3059 WARN(("VBoxVBVAExHSIsDisabled: disabled"));
3060 return VINF_SUCCESS;
3061 }
3062
3063 pHCtl = VBoxVBVAExHCtlCreate(&pVdma->CmdVbva, VBVAEXHOSTCTL_TYPE_GHH_DISABLE);
3064 if (!pHCtl)
3065 {
3066 WARN(("VBoxVBVAExHCtlCreate failed\n"));
3067 return VERR_NO_MEMORY;
3068 }
3069
3070 pHCtl->u.cmd.pu8Cmd = (uint8_t*)pEnable;
3071 pHCtl->u.cmd.cbCmd = sizeof (*pEnable);
3072 rc = vdmaVBVACtlSubmit(pVdma, pHCtl, VBVAEXHOSTCTL_SOURCE_GUEST, pfnComplete, pvComplete);
3073 if (RT_SUCCESS(rc))
3074 return VINF_SUCCESS;
3075
3076 WARN(("vdmaVBVACtlSubmit failed rc %d\n", rc));
3077 VBoxVBVAExHCtlFree(&pVdma->CmdVbva, pHCtl);
3078 return rc;
3079}
3080
3081static int vdmaVBVACtlEnableDisableSubmitInternal(PVBOXVDMAHOST pVdma, VBVAENABLE *pEnable, PFNVBVAEXHOSTCTL_COMPLETE pfnComplete, void *pvComplete)
3082{
3083 bool fEnable = ((pEnable->u32Flags & (VBVA_F_ENABLE | VBVA_F_DISABLE)) == VBVA_F_ENABLE);
3084 if (fEnable)
3085 return vdmaVBVACtlEnableSubmitInternal(pVdma, pEnable, false, pfnComplete, pvComplete);
3086 return vdmaVBVACtlDisableSubmitInternal(pVdma, pEnable, pfnComplete, pvComplete);
3087}
3088
3089static int vdmaVBVACtlEnableDisableSubmit(PVBOXVDMAHOST pVdma, VBOXCMDVBVA_CTL_ENABLE *pEnable)
3090{
3091 VBoxSHGSMICommandMarkAsynchCompletion(&pEnable->Hdr);
3092 int rc = vdmaVBVACtlEnableDisableSubmitInternal(pVdma, &pEnable->Enable, vboxCmdVBVACmdCtlGuestCompletion, pVdma);
3093 if (RT_SUCCESS(rc))
3094 return VINF_SUCCESS;
3095
3096 WARN(("vdmaVBVACtlEnableDisableSubmitInternal failed %d\n", rc));
3097 pEnable->Hdr.i32Result = rc;
3098 rc = VBoxSHGSMICommandComplete(pVdma->pHgsmi, &pEnable->Hdr);
3099 AssertRC(rc);
3100 return VINF_SUCCESS;
3101}
3102
3103static DECLCALLBACK(void) vdmaVBVACtlSubmitSyncCompletion(VBVAEXHOSTCONTEXT *pVbva, struct VBVAEXHOSTCTL *pCtl, int rc, void *pvContext)
3104{
3105 VDMA_VBVA_CTL_CYNC_COMPLETION *pData = (VDMA_VBVA_CTL_CYNC_COMPLETION*)pvContext;
3106 pData->rc = rc;
3107 rc = RTSemEventSignal(pData->hEvent);
3108 if (!RT_SUCCESS(rc))
3109 WARN(("RTSemEventSignal failed %d\n", rc));
3110}
3111
3112static int vdmaVBVACtlSubmitSync(PVBOXVDMAHOST pVdma, VBVAEXHOSTCTL* pCtl, VBVAEXHOSTCTL_SOURCE enmSource)
3113{
3114 VDMA_VBVA_CTL_CYNC_COMPLETION Data;
3115 Data.rc = VERR_NOT_IMPLEMENTED;
3116 int rc = RTSemEventCreate(&Data.hEvent);
3117 if (!RT_SUCCESS(rc))
3118 {
3119 WARN(("RTSemEventCreate failed %d\n", rc));
3120 return rc;
3121 }
3122
3123 rc = vdmaVBVACtlSubmit(pVdma, pCtl, enmSource, vdmaVBVACtlSubmitSyncCompletion, &Data);
3124 if (RT_SUCCESS(rc))
3125 {
3126 rc = RTSemEventWait(Data.hEvent, RT_INDEFINITE_WAIT);
3127 if (RT_SUCCESS(rc))
3128 {
3129 rc = Data.rc;
3130 if (!RT_SUCCESS(rc))
3131 WARN(("vdmaVBVACtlSubmitSyncCompletion returned %d\n", rc));
3132 }
3133 else
3134 WARN(("RTSemEventWait failed %d\n", rc));
3135 }
3136 else
3137 Log(("vdmaVBVACtlSubmit failed %d\n", rc));
3138
3139 RTSemEventDestroy(Data.hEvent);
3140
3141 return rc;
3142}
3143
3144static int vdmaVBVAPause(PVBOXVDMAHOST pVdma)
3145{
3146 VBVAEXHOSTCTL Ctl;
3147 Ctl.enmType = VBVAEXHOSTCTL_TYPE_HH_INTERNAL_PAUSE;
3148 return vdmaVBVACtlSubmitSync(pVdma, &Ctl, VBVAEXHOSTCTL_SOURCE_HOST);
3149}
3150
3151static int vdmaVBVAResume(PVBOXVDMAHOST pVdma)
3152{
3153 VBVAEXHOSTCTL Ctl;
3154 Ctl.enmType = VBVAEXHOSTCTL_TYPE_HH_INTERNAL_RESUME;
3155 return vdmaVBVACtlSubmitSync(pVdma, &Ctl, VBVAEXHOSTCTL_SOURCE_HOST);
3156}
3157
3158static int vboxVDMACmdSubmitPerform(struct VBOXVDMAHOST *pVdma)
3159{
3160 int rc = VBoxVBVAExHSCheckCommands(&pVdma->CmdVbva);
3161 switch (rc)
3162 {
3163 case VINF_SUCCESS:
3164 return VBoxVDMAThreadEventNotify(&pVdma->Thread);
3165 case VINF_ALREADY_INITIALIZED:
3166 case VINF_EOF:
3167 case VERR_INVALID_STATE:
3168 return VINF_SUCCESS;
3169 default:
3170 Assert(!RT_FAILURE(rc));
3171 return RT_FAILURE(rc) ? rc : VERR_INTERNAL_ERROR;
3172 }
3173}
3174
3175
3176int vboxCmdVBVACmdHostCtl(PPDMIDISPLAYVBVACALLBACKS pInterface,
3177 struct VBOXCRCMDCTL* pCmd, uint32_t cbCmd,
3178 PFNCRCTLCOMPLETION pfnCompletion,
3179 void *pvCompletion)
3180{
3181 PVGASTATE pVGAState = PPDMIDISPLAYVBVACALLBACKS_2_PVGASTATE(pInterface);
3182 struct VBOXVDMAHOST *pVdma = pVGAState->pVdma;
3183 pCmd->CalloutList.List.pNext = NULL;
3184 return vdmaVBVACtlOpaqueHostSubmit(pVdma, pCmd, cbCmd, pfnCompletion, pvCompletion);
3185}
3186
3187typedef struct VBOXCMDVBVA_CMDHOSTCTL_SYNC
3188{
3189 struct VBOXVDMAHOST *pVdma;
3190 uint32_t fProcessing;
3191 int rc;
3192} VBOXCMDVBVA_CMDHOSTCTL_SYNC;
3193
3194static DECLCALLBACK(void) vboxCmdVBVACmdHostCtlSyncCb(struct VBOXCRCMDCTL* pCmd, uint32_t cbCmd, int rc, void *pvCompletion)
3195{
3196 VBOXCMDVBVA_CMDHOSTCTL_SYNC *pData = (VBOXCMDVBVA_CMDHOSTCTL_SYNC*)pvCompletion;
3197
3198 pData->rc = rc;
3199
3200 struct VBOXVDMAHOST *pVdma = pData->pVdma;
3201
3202 ASMAtomicIncS32(&pVdma->i32cHostCrCtlCompleted);
3203
3204 pData->fProcessing = 0;
3205
3206 RTSemEventMultiSignal(pVdma->HostCrCtlCompleteEvent);
3207}
3208
3209static DECLCALLBACK(int) vboxCmdVBVACmdCallout(struct VBOXVDMAHOST *pVdma, struct VBOXCRCMDCTL* pCmd, VBOXCRCMDCTL_CALLOUT_LISTENTRY *pEntry, PFNVBOXCRCMDCTL_CALLOUT_CB pfnCb)
3210{
3211 pEntry->pfnCb = pfnCb;
3212 int rc = RTCritSectEnter(&pVdma->CalloutCritSect);
3213 if (RT_SUCCESS(rc))
3214 {
3215 RTListAppend(&pCmd->CalloutList.List, &pEntry->Node);
3216 RTCritSectLeave(&pVdma->CalloutCritSect);
3217
3218 RTSemEventMultiSignal(pVdma->HostCrCtlCompleteEvent);
3219 }
3220 else
3221 WARN(("RTCritSectEnter failed %d\n", rc));
3222
3223 return rc;
3224}
3225
3226
3227static int vboxCmdVBVACmdCalloutProcess(struct VBOXVDMAHOST *pVdma, struct VBOXCRCMDCTL* pCmd)
3228{
3229 int rc = VINF_SUCCESS;
3230 for(;;)
3231 {
3232 rc = RTCritSectEnter(&pVdma->CalloutCritSect);
3233 if (RT_SUCCESS(rc))
3234 {
3235 VBOXCRCMDCTL_CALLOUT_LISTENTRY* pEntry = RTListGetFirst(&pCmd->CalloutList.List, VBOXCRCMDCTL_CALLOUT_LISTENTRY, Node);
3236 if (pEntry)
3237 RTListNodeRemove(&pEntry->Node);
3238 RTCritSectLeave(&pVdma->CalloutCritSect);
3239
3240 if (!pEntry)
3241 break;
3242
3243 pEntry->pfnCb(pEntry);
3244 }
3245 else
3246 {
3247 WARN(("RTCritSectEnter failed %d\n", rc));
3248 break;
3249 }
3250 }
3251
3252 return rc;
3253}
3254
3255int vboxCmdVBVACmdHostCtlSync(PPDMIDISPLAYVBVACALLBACKS pInterface,
3256 struct VBOXCRCMDCTL* pCmd, uint32_t cbCmd)
3257{
3258 PVGASTATE pVGAState = PPDMIDISPLAYVBVACALLBACKS_2_PVGASTATE(pInterface);
3259 struct VBOXVDMAHOST *pVdma = pVGAState->pVdma;
3260 VBOXCMDVBVA_CMDHOSTCTL_SYNC Data;
3261 Data.pVdma = pVdma;
3262 Data.fProcessing = 1;
3263 Data.rc = VERR_INTERNAL_ERROR;
3264 RTListInit(&pCmd->CalloutList.List);
3265 int rc = vdmaVBVACtlOpaqueHostSubmit(pVdma, pCmd, cbCmd, vboxCmdVBVACmdHostCtlSyncCb, &Data);
3266 if (!RT_SUCCESS(rc))
3267 {
3268 WARN(("vdmaVBVACtlOpaqueHostSubmit failed %d", rc));
3269 return rc;
3270 }
3271
3272 while (Data.fProcessing)
3273 {
3274 /* Poll infrequently to make sure no completed message has been missed. */
3275 RTSemEventMultiWait(pVdma->HostCrCtlCompleteEvent, 500);
3276
3277 vboxCmdVBVACmdCalloutProcess(pVdma, pCmd);
3278
3279 if (Data.fProcessing)
3280 RTThreadYield();
3281 }
3282
3283 /* extra check callouts */
3284 vboxCmdVBVACmdCalloutProcess(pVdma, pCmd);
3285
3286 /* 'Our' message has been processed, so should reset the semaphore.
3287 * There is still possible that another message has been processed
3288 * and the semaphore has been signalled again.
3289 * Reset only if there are no other messages completed.
3290 */
3291 int32_t c = ASMAtomicDecS32(&pVdma->i32cHostCrCtlCompleted);
3292 Assert(c >= 0);
3293 if (!c)
3294 RTSemEventMultiReset(pVdma->HostCrCtlCompleteEvent);
3295
3296 rc = Data.rc;
3297 if (!RT_SUCCESS(rc))
3298 WARN(("host call failed %d", rc));
3299
3300 return rc;
3301}
3302
3303int vboxCmdVBVACmdCtl(PVGASTATE pVGAState, VBOXCMDVBVA_CTL *pCtl, uint32_t cbCtl)
3304{
3305 struct VBOXVDMAHOST *pVdma = pVGAState->pVdma;
3306 int rc = VINF_SUCCESS;
3307 switch (pCtl->u32Type)
3308 {
3309 case VBOXCMDVBVACTL_TYPE_3DCTL:
3310 return vdmaVBVACtlGenericGuestSubmit(pVdma, VBVAEXHOSTCTL_TYPE_GHH_BE_OPAQUE, pCtl, cbCtl);
3311 case VBOXCMDVBVACTL_TYPE_RESIZE:
3312 return vdmaVBVACtlGenericGuestSubmit(pVdma, VBVAEXHOSTCTL_TYPE_GHH_RESIZE, pCtl, cbCtl);
3313 case VBOXCMDVBVACTL_TYPE_ENABLE:
3314 if (cbCtl != sizeof (VBOXCMDVBVA_CTL_ENABLE))
3315 {
3316 WARN(("incorrect enable size\n"));
3317 rc = VERR_INVALID_PARAMETER;
3318 break;
3319 }
3320 return vdmaVBVACtlEnableDisableSubmit(pVdma, (VBOXCMDVBVA_CTL_ENABLE*)pCtl);
3321 default:
3322 WARN(("unsupported type\n"));
3323 rc = VERR_INVALID_PARAMETER;
3324 break;
3325 }
3326
3327 pCtl->i32Result = rc;
3328 rc = VBoxSHGSMICommandComplete(pVdma->pHgsmi, pCtl);
3329 AssertRC(rc);
3330 return VINF_SUCCESS;
3331}
3332
3333int vboxCmdVBVACmdSubmit(PVGASTATE pVGAState)
3334{
3335 if (!VBoxVBVAExHSIsEnabled(&pVGAState->pVdma->CmdVbva))
3336 {
3337 WARN(("vdma VBVA is disabled\n"));
3338 return VERR_INVALID_STATE;
3339 }
3340
3341 return vboxVDMACmdSubmitPerform(pVGAState->pVdma);
3342}
3343
3344int vboxCmdVBVACmdFlush(PVGASTATE pVGAState)
3345{
3346 WARN(("flush\n"));
3347 if (!VBoxVBVAExHSIsEnabled(&pVGAState->pVdma->CmdVbva))
3348 {
3349 WARN(("vdma VBVA is disabled\n"));
3350 return VERR_INVALID_STATE;
3351 }
3352 return vboxVDMACmdSubmitPerform(pVGAState->pVdma);
3353}
3354
3355void vboxCmdVBVACmdTimer(PVGASTATE pVGAState)
3356{
3357 if (!VBoxVBVAExHSIsEnabled(&pVGAState->pVdma->CmdVbva))
3358 return;
3359 vboxVDMACmdSubmitPerform(pVGAState->pVdma);
3360}
3361
3362bool vboxCmdVBVAIsEnabled(PVGASTATE pVGAState)
3363{
3364 return VBoxVBVAExHSIsEnabled(&pVGAState->pVdma->CmdVbva);
3365}
3366#endif
3367
3368int vboxVDMASaveStateExecPrep(struct VBOXVDMAHOST *pVdma, PSSMHANDLE pSSM)
3369{
3370#ifdef VBOX_WITH_CRHGSMI
3371 int rc = vdmaVBVAPause(pVdma);
3372 if (RT_SUCCESS(rc))
3373 return VINF_SUCCESS;
3374
3375 if (rc != VERR_INVALID_STATE)
3376 {
3377 WARN(("vdmaVBVAPause failed %d\n", rc));
3378 return rc;
3379 }
3380
3381#ifdef DEBUG_misha
3382 WARN(("debug prep"));
3383#endif
3384
3385 PVGASTATE pVGAState = pVdma->pVGAState;
3386 PVBOXVDMACMD_CHROMIUM_CTL pCmd = (PVBOXVDMACMD_CHROMIUM_CTL)vboxVDMACrCtlCreate(
3387 VBOXVDMACMD_CHROMIUM_CTL_TYPE_SAVESTATE_BEGIN, sizeof (*pCmd));
3388 Assert(pCmd);
3389 if (pCmd)
3390 {
3391 rc = vboxVDMACrCtlPost(pVGAState, pCmd, sizeof (*pCmd));
3392 AssertRC(rc);
3393 if (RT_SUCCESS(rc))
3394 {
3395 rc = vboxVDMACrCtlGetRc(pCmd);
3396 }
3397 vboxVDMACrCtlRelease(pCmd);
3398 return rc;
3399 }
3400 return VERR_NO_MEMORY;
3401#else
3402 return VINF_SUCCESS;
3403#endif
3404}
3405
3406int vboxVDMASaveStateExecDone(struct VBOXVDMAHOST *pVdma, PSSMHANDLE pSSM)
3407{
3408#ifdef VBOX_WITH_CRHGSMI
3409 int rc = vdmaVBVAResume(pVdma);
3410 if (RT_SUCCESS(rc))
3411 return VINF_SUCCESS;
3412
3413 if (rc != VERR_INVALID_STATE)
3414 {
3415 WARN(("vdmaVBVAResume failed %d\n", rc));
3416 return rc;
3417 }
3418
3419#ifdef DEBUG_misha
3420 WARN(("debug done"));
3421#endif
3422
3423 PVGASTATE pVGAState = pVdma->pVGAState;
3424 PVBOXVDMACMD_CHROMIUM_CTL pCmd = (PVBOXVDMACMD_CHROMIUM_CTL)vboxVDMACrCtlCreate(
3425 VBOXVDMACMD_CHROMIUM_CTL_TYPE_SAVESTATE_END, sizeof (*pCmd));
3426 Assert(pCmd);
3427 if (pCmd)
3428 {
3429 rc = vboxVDMACrCtlPost(pVGAState, pCmd, sizeof (*pCmd));
3430 AssertRC(rc);
3431 if (RT_SUCCESS(rc))
3432 {
3433 rc = vboxVDMACrCtlGetRc(pCmd);
3434 }
3435 vboxVDMACrCtlRelease(pCmd);
3436 return rc;
3437 }
3438 return VERR_NO_MEMORY;
3439#else
3440 return VINF_SUCCESS;
3441#endif
3442}
3443
3444int vboxVDMASaveStateExecPerform(struct VBOXVDMAHOST *pVdma, PSSMHANDLE pSSM)
3445{
3446 int rc;
3447
3448#ifdef VBOX_WITH_CRHGSMI
3449 if (!VBoxVBVAExHSIsEnabled(&pVdma->CmdVbva))
3450#endif
3451 {
3452 rc = SSMR3PutU32(pSSM, 0xffffffff);
3453 AssertRCReturn(rc, rc);
3454 return VINF_SUCCESS;
3455 }
3456
3457#ifdef VBOX_WITH_CRHGSMI
3458 PVGASTATE pVGAState = pVdma->pVGAState;
3459 uint8_t * pu8VramBase = pVGAState->vram_ptrR3;
3460
3461 rc = SSMR3PutU32(pSSM, (uint32_t)(((uint8_t*)pVdma->CmdVbva.pVBVA) - pu8VramBase));
3462 AssertRCReturn(rc, rc);
3463
3464 VBVAEXHOSTCTL HCtl;
3465 HCtl.enmType = VBVAEXHOSTCTL_TYPE_HH_SAVESTATE;
3466 HCtl.u.state.pSSM = pSSM;
3467 HCtl.u.state.u32Version = 0;
3468 return vdmaVBVACtlSubmitSync(pVdma, &HCtl, VBVAEXHOSTCTL_SOURCE_HOST);
3469#endif
3470}
3471
3472int vboxVDMASaveLoadExecPerform(struct VBOXVDMAHOST *pVdma, PSSMHANDLE pSSM, uint32_t u32Version)
3473{
3474 uint32_t u32;
3475 int rc = SSMR3GetU32(pSSM, &u32);
3476 AssertRCReturn(rc, rc);
3477
3478 if (u32 != 0xffffffff)
3479 {
3480#ifdef VBOX_WITH_CRHGSMI
3481 rc = vdmaVBVACtlEnableSubmitSync(pVdma, u32, true);
3482 AssertRCReturn(rc, rc);
3483
3484 Assert(pVdma->CmdVbva.i32State == VBVAEXHOSTCONTEXT_ESTATE_PAUSED);
3485
3486 VBVAEXHOSTCTL HCtl;
3487 HCtl.enmType = VBVAEXHOSTCTL_TYPE_HH_LOADSTATE;
3488 HCtl.u.state.pSSM = pSSM;
3489 HCtl.u.state.u32Version = u32Version;
3490 rc = vdmaVBVACtlSubmitSync(pVdma, &HCtl, VBVAEXHOSTCTL_SOURCE_HOST);
3491 AssertRCReturn(rc, rc);
3492
3493 rc = vdmaVBVAResume(pVdma);
3494 AssertRCReturn(rc, rc);
3495
3496 return VINF_SUCCESS;
3497#else
3498 WARN(("Unsupported VBVACtl info!\n"));
3499 return VERR_VERSION_MISMATCH;
3500#endif
3501 }
3502
3503 return VINF_SUCCESS;
3504}
3505
3506int vboxVDMASaveLoadDone(struct VBOXVDMAHOST *pVdma)
3507{
3508#ifdef VBOX_WITH_CRHGSMI
3509 if (!VBoxVBVAExHSIsEnabled(&pVdma->CmdVbva))
3510 return VINF_SUCCESS;
3511
3512/** @todo r=bird: BTW. would be great if you put in a couple of comments here and there explaining what
3513 * the purpose of this code is. */
3514 VBVAEXHOSTCTL* pHCtl = VBoxVBVAExHCtlCreate(&pVdma->CmdVbva, VBVAEXHOSTCTL_TYPE_HH_LOADSTATE_DONE);
3515 if (!pHCtl)
3516 {
3517 WARN(("VBoxVBVAExHCtlCreate failed\n"));
3518 return VERR_NO_MEMORY;
3519 }
3520
3521 /* sanity */
3522 pHCtl->u.cmd.pu8Cmd = NULL;
3523 pHCtl->u.cmd.cbCmd = 0;
3524
3525 /* NULL completion will just free the ctl up */
3526 int rc = vdmaVBVACtlSubmit(pVdma, pHCtl, VBVAEXHOSTCTL_SOURCE_HOST, NULL, NULL);
3527 if (RT_FAILURE(rc))
3528 {
3529 Log(("vdmaVBVACtlSubmit failed rc %d\n", rc));
3530 VBoxVBVAExHCtlFree(&pVdma->CmdVbva, pHCtl);
3531 return rc;
3532 }
3533#endif
3534 return VINF_SUCCESS;
3535}
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