VirtualBox

source: vbox/trunk/src/VBox/Devices/Graphics/DevVGA_VBVA.cpp@ 56814

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

Devices/Graphics: re-set VBVA capabilities when VBVA is disabled (for screen 0).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 95.6 KB
Line 
1/* $Id: DevVGA_VBVA.cpp 56814 2015-07-06 11:10:52Z vboxsync $ */
2/** @file
3 * VirtualBox Video Acceleration (VBVA).
4 */
5
6/*
7 * Copyright (C) 2006-2015 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18/*******************************************************************************
19* Header Files *
20*******************************************************************************/
21#define LOG_GROUP LOG_GROUP_DEV_VGA
22#include <VBox/vmm/pdmifs.h>
23#include <VBox/vmm/pdmdev.h>
24#include <VBox/vmm/pgm.h>
25#include <VBox/vmm/ssm.h>
26#include <VBox/VMMDev.h>
27#include <VBox/VBoxVideo.h>
28#include <iprt/alloc.h>
29#include <iprt/assert.h>
30#include <iprt/asm.h>
31#include <iprt/string.h>
32#include <iprt/param.h>
33#ifdef VBOX_WITH_VIDEOHWACCEL
34#include <iprt/semaphore.h>
35#endif
36
37#include "DevVGA.h"
38
39/* A very detailed logging. */
40#if 0 // def DEBUG_sunlover
41#define LOGVBVABUFFER(a) LogFlow(a)
42#else
43#define LOGVBVABUFFER(a) do {} while (0)
44#endif
45
46/*******************************************************************************
47* Structures and Typedefs *
48*******************************************************************************/
49typedef struct VBVAPARTIALRECORD
50{
51 uint8_t *pu8;
52 uint32_t cb;
53} VBVAPARTIALRECORD;
54
55typedef struct VBVADATA
56{
57 struct
58 {
59 VBVABUFFER *pVBVA; /* Pointer to the guest memory with the VBVABUFFER. */
60 uint8_t *pu8Data; /* For convenience, pointer to the guest ring buffer (VBVABUFFER::au8Data). */
61 } guest;
62 uint32_t u32VBVAOffset; /* VBVABUFFER offset in the guest VRAM. */
63 VBVAPARTIALRECORD partialRecord; /* Partial record temporary storage. */
64 uint32_t off32Data; /* The offset where the data starts in the VBVABUFFER.
65 * The host code uses it instead of VBVABUFFER::off32Data.
66 */
67 uint32_t indexRecordFirst; /* Index of the first filled record in VBVABUFFER::aRecords. */
68 uint32_t cbPartialWriteThreshold; /* Copy of VBVABUFFER::cbPartialWriteThreshold used by host code. */
69 uint32_t cbData; /* Copy of VBVABUFFER::cbData used by host code. */
70} VBVADATA;
71
72typedef struct VBVAVIEW
73{
74 VBVAINFOVIEW view;
75 VBVAINFOSCREEN screen;
76 VBVADATA vbva;
77} VBVAVIEW;
78
79typedef struct VBVAMOUSESHAPEINFO
80{
81 bool fSet;
82 bool fVisible;
83 bool fAlpha;
84 uint32_t u32HotX;
85 uint32_t u32HotY;
86 uint32_t u32Width;
87 uint32_t u32Height;
88 uint32_t cbShape;
89 uint32_t cbAllocated;
90 uint8_t *pu8Shape;
91} VBVAMOUSESHAPEINFO;
92
93/** @todo saved state: save and restore VBVACONTEXT */
94typedef struct VBVACONTEXT
95{
96 uint32_t cViews;
97 VBVAVIEW aViews[VBOX_VIDEO_MAX_SCREENS];
98 VBVAMOUSESHAPEINFO mouseShapeInfo;
99 bool fPaused;
100 uint32_t xCursor;
101 uint32_t yCursor;
102 VBVAMODEHINT aModeHints[VBOX_VIDEO_MAX_SCREENS];
103} VBVACONTEXT;
104
105
106static void vbvaDataCleanup(VBVADATA *pVBVAData)
107{
108 if (pVBVAData->guest.pVBVA)
109 {
110 RT_ZERO(pVBVAData->guest.pVBVA->hostFlags);
111 }
112
113 RTMemFree(pVBVAData->partialRecord.pu8);
114
115 RT_ZERO(*pVBVAData);
116 pVBVAData->u32VBVAOffset = HGSMIOFFSET_VOID;
117}
118
119/** Copies @a cb bytes from the VBVA ring buffer to the @a pu8Dst.
120 * Used for partial records or for records which cross the ring boundary.
121 */
122static bool vbvaFetchBytes(VBVADATA *pVBVAData, uint8_t *pu8Dst, uint32_t cb)
123{
124 if (cb >= pVBVAData->cbData)
125 {
126 AssertMsgFailed(("cb = 0x%08X, ring buffer size 0x%08X", cb, pVBVAData->cbData));
127 return false;
128 }
129
130 const uint32_t u32BytesTillBoundary = pVBVAData->cbData - pVBVAData->off32Data;
131 const uint8_t *pu8Src = &pVBVAData->guest.pu8Data[pVBVAData->off32Data];
132 const int32_t i32Diff = cb - u32BytesTillBoundary;
133
134 if (i32Diff <= 0)
135 {
136 /* Chunk will not cross buffer boundary. */
137 memcpy(pu8Dst, pu8Src, cb);
138 }
139 else
140 {
141 /* Chunk crosses buffer boundary. */
142 memcpy(pu8Dst, pu8Src, u32BytesTillBoundary);
143 memcpy(pu8Dst + u32BytesTillBoundary, &pVBVAData->guest.pu8Data[0], i32Diff);
144 }
145
146 /* Advance data offset and sync with guest. */
147 pVBVAData->off32Data = (pVBVAData->off32Data + cb) % pVBVAData->cbData;
148 pVBVAData->guest.pVBVA->off32Data = pVBVAData->off32Data;
149 return true;
150}
151
152
153static bool vbvaPartialRead(uint32_t cbRecord, VBVADATA *pVBVAData)
154{
155 VBVAPARTIALRECORD *pPartialRecord = &pVBVAData->partialRecord;
156 uint8_t *pu8New;
157
158 LOGVBVABUFFER(("vbvaPartialRead: p = %p, cb = %d, cbRecord 0x%08X\n",
159 pPartialRecord->pu8, pPartialRecord->cb, cbRecord));
160
161 Assert(cbRecord > pPartialRecord->cb); /* Caller ensures this. */
162
163 const uint32_t cbChunk = cbRecord - pPartialRecord->cb;
164 if (cbChunk >= pVBVAData->cbData)
165 {
166 return false;
167 }
168
169 if (pPartialRecord->pu8)
170 {
171 Assert(pPartialRecord->cb);
172 pu8New = (uint8_t *)RTMemRealloc(pPartialRecord->pu8, cbRecord);
173 }
174 else
175 {
176 Assert(!pPartialRecord->cb);
177 pu8New = (uint8_t *)RTMemAlloc(cbRecord);
178 }
179
180 if (!pu8New)
181 {
182 /* Memory allocation failed, fail the function. */
183 Log(("vbvaPartialRead: failed to (re)alocate memory for partial record!!! cbRecord 0x%08X\n",
184 cbRecord));
185
186 return false;
187 }
188
189 /* Fetch data from the ring buffer. */
190 if (!vbvaFetchBytes(pVBVAData, pu8New + pPartialRecord->cb, cbChunk))
191 {
192 return false;
193 }
194
195 pPartialRecord->pu8 = pu8New;
196 pPartialRecord->cb = cbRecord;
197
198 return true;
199}
200
201/* For contiguous chunks just return the address in the buffer.
202 * For crossing boundary - allocate a buffer from heap.
203 */
204static bool vbvaFetchCmd(VBVADATA *pVBVAData, VBVACMDHDR **ppHdr, uint32_t *pcbCmd)
205{
206 VBVAPARTIALRECORD *pPartialRecord = &pVBVAData->partialRecord;
207 uint32_t indexRecordFirst = pVBVAData->indexRecordFirst;
208 const uint32_t indexRecordFree = ASMAtomicReadU32(&pVBVAData->guest.pVBVA->indexRecordFree);
209
210 LOGVBVABUFFER(("first = %d, free = %d\n",
211 indexRecordFirst, indexRecordFree));
212
213 if (indexRecordFree >= RT_ELEMENTS(pVBVAData->guest.pVBVA->aRecords))
214 {
215 return false;
216 }
217
218 if (indexRecordFirst == indexRecordFree)
219 {
220 /* No records to process. Return without assigning output variables. */
221 return true;
222 }
223
224 uint32_t cbRecordCurrent = ASMAtomicReadU32(&pVBVAData->guest.pVBVA->aRecords[indexRecordFirst].cbRecord);
225
226 LOGVBVABUFFER(("cbRecord = 0x%08X, pPartialRecord->cb = 0x%08X\n", cbRecordCurrent, pPartialRecord->cb));
227
228 uint32_t cbRecord = cbRecordCurrent & ~VBVA_F_RECORD_PARTIAL;
229
230 if (cbRecord > VBVA_MAX_RECORD_SIZE)
231 {
232 return false;
233 }
234
235 if (pPartialRecord->cb)
236 {
237 /* There is a partial read in process. Continue with it. */
238 Assert (pPartialRecord->pu8);
239
240 LOGVBVABUFFER(("continue partial record cb = %d cbRecord 0x%08X, first = %d, free = %d\n",
241 pPartialRecord->cb, cbRecordCurrent, indexRecordFirst, indexRecordFree));
242
243 if (cbRecord > pPartialRecord->cb)
244 {
245 /* New data has been added to the record. */
246 if (!vbvaPartialRead(cbRecord, pVBVAData))
247 {
248 return false;
249 }
250 }
251
252 if (!(cbRecordCurrent & VBVA_F_RECORD_PARTIAL))
253 {
254 /* The record is completed by guest. Return it to the caller. */
255 *ppHdr = (VBVACMDHDR *)pPartialRecord->pu8;
256 *pcbCmd = pPartialRecord->cb;
257
258 pPartialRecord->pu8 = NULL;
259 pPartialRecord->cb = 0;
260
261 /* Advance the record index and sync with guest. */
262 pVBVAData->indexRecordFirst = (indexRecordFirst + 1) % RT_ELEMENTS(pVBVAData->guest.pVBVA->aRecords);
263 pVBVAData->guest.pVBVA->indexRecordFirst = pVBVAData->indexRecordFirst;
264
265 LOGVBVABUFFER(("partial done ok, data = %d, free = %d\n",
266 pVBVAData->off32Data, pVBVAData->guest.pVBVA->off32Free));
267 }
268
269 return true;
270 }
271
272 /* A new record need to be processed. */
273 if (cbRecordCurrent & VBVA_F_RECORD_PARTIAL)
274 {
275 /* Current record is being written by guest. '=' is important here,
276 * because the guest will do a FLUSH at this condition.
277 * This partial record is too large for the ring buffer and must
278 * be accumulated in an allocated buffer.
279 */
280 if (cbRecord >= pVBVAData->cbData - pVBVAData->cbPartialWriteThreshold)
281 {
282 /* Partial read must be started. */
283 if (!vbvaPartialRead(cbRecord, pVBVAData))
284 {
285 return false;
286 }
287
288 LOGVBVABUFFER(("started partial record cb = 0x%08X cbRecord 0x%08X, first = %d, free = %d\n",
289 pPartialRecord->cb, cbRecordCurrent, indexRecordFirst, indexRecordFree));
290 }
291
292 return true;
293 }
294
295 /* Current record is complete. If it is not empty, process it. */
296 if (cbRecord >= pVBVAData->cbData)
297 {
298 return false;
299 }
300
301 if (cbRecord)
302 {
303 /* The size of largest contiguous chunk in the ring buffer. */
304 uint32_t u32BytesTillBoundary = pVBVAData->cbData - pVBVAData->off32Data;
305
306 /* The pointer to data in the ring buffer. */
307 uint8_t *pu8Src = &pVBVAData->guest.pu8Data[pVBVAData->off32Data];
308
309 /* Fetch or point the data. */
310 if (u32BytesTillBoundary >= cbRecord)
311 {
312 /* The command does not cross buffer boundary. Return address in the buffer. */
313 *ppHdr = (VBVACMDHDR *)pu8Src;
314
315 /* Advance data offset and sync with guest. */
316 pVBVAData->off32Data = (pVBVAData->off32Data + cbRecord) % pVBVAData->cbData;
317 pVBVAData->guest.pVBVA->off32Data = pVBVAData->off32Data;
318 }
319 else
320 {
321 /* The command crosses buffer boundary. Rare case, so not optimized. */
322 uint8_t *pu8Dst = (uint8_t *)RTMemAlloc(cbRecord);
323
324 if (!pu8Dst)
325 {
326 LogFlowFunc (("could not allocate %d bytes from heap!!!\n", cbRecord));
327 return false;
328 }
329
330 vbvaFetchBytes(pVBVAData, pu8Dst, cbRecord);
331
332 *ppHdr = (VBVACMDHDR *)pu8Dst;
333
334 LOGVBVABUFFER(("Allocated from heap %p\n", pu8Dst));
335 }
336 }
337
338 *pcbCmd = cbRecord;
339
340 /* Advance the record index and sync with guest. */
341 pVBVAData->indexRecordFirst = (indexRecordFirst + 1) % RT_ELEMENTS(pVBVAData->guest.pVBVA->aRecords);
342 pVBVAData->guest.pVBVA->indexRecordFirst = pVBVAData->indexRecordFirst;
343
344 LOGVBVABUFFER(("done ok, data = %d, free = %d\n",
345 pVBVAData->off32Data, pVBVAData->guest.pVBVA->off32Free));
346
347 return true;
348}
349
350static void vbvaReleaseCmd(VBVADATA *pVBVAData, VBVACMDHDR *pHdr, uint32_t cbCmd)
351{
352 VBVAPARTIALRECORD *pPartialRecord = &pVBVAData->partialRecord;
353 uint8_t *au8RingBuffer = pVBVAData->guest.pu8Data;
354
355 if ( (uint8_t *)pHdr >= au8RingBuffer
356 && (uint8_t *)pHdr < &au8RingBuffer[pVBVAData->cbData])
357 {
358 /* The pointer is inside ring buffer. Must be continuous chunk. */
359 Assert(pVBVAData->cbData - ((uint8_t *)pHdr - au8RingBuffer) >= cbCmd);
360
361 /* Do nothing. */
362
363 Assert (!pPartialRecord->pu8 && pPartialRecord->cb == 0);
364 }
365 else
366 {
367 /* The pointer is outside. It is then an allocated copy. */
368 LOGVBVABUFFER(("Free heap %p\n", pHdr));
369
370 if ((uint8_t *)pHdr == pPartialRecord->pu8)
371 {
372 pPartialRecord->pu8 = NULL;
373 pPartialRecord->cb = 0;
374 }
375 else
376 {
377 Assert(!pPartialRecord->pu8 && pPartialRecord->cb == 0);
378 }
379
380 RTMemFree(pHdr);
381 }
382}
383
384static int vbvaFlushProcess(unsigned uScreenId, PVGASTATE pVGAState, VBVADATA *pVBVAData)
385{
386 LOGVBVABUFFER(("uScreenId %d, indexRecordFirst = %d, indexRecordFree = %d, off32Data = %d, off32Free = %d\n",
387 uScreenId, pVBVAData->indexRecordFirst, pVBVAData->guest.pVBVA->indexRecordFree,
388 pVBVAData->off32Data, pVBVAData->guest.pVBVA->off32Free));
389 struct {
390 /* The rectangle that includes all dirty rectangles. */
391 int32_t xLeft;
392 int32_t xRight;
393 int32_t yTop;
394 int32_t yBottom;
395 } dirtyRect;
396 RT_ZERO(dirtyRect);
397
398 bool fUpdate = false; /* Whether there were any updates. */
399 bool fDirtyEmpty = true;
400
401 for (;;)
402 {
403 VBVACMDHDR *phdr = NULL;
404 uint32_t cbCmd = ~0;
405
406 /* Fetch the command data. */
407 if (!vbvaFetchCmd(pVBVAData, &phdr, &cbCmd))
408 {
409 LogFunc(("unable to fetch command. off32Data = %d, off32Free = %d!!!\n",
410 pVBVAData->off32Data, pVBVAData->guest.pVBVA->off32Free));
411
412 return VERR_NOT_SUPPORTED;
413 }
414
415 if (cbCmd == uint32_t(~0))
416 {
417 /* No more commands yet in the queue. */
418 break;
419 }
420
421 if (cbCmd < sizeof(VBVACMDHDR))
422 {
423 LogFunc(("short command. off32Data = %d, off32Free = %d, cbCmd %d!!!\n",
424 pVBVAData->off32Data, pVBVAData->guest.pVBVA->off32Free, cbCmd));
425
426 return VERR_NOT_SUPPORTED;
427 }
428
429 if (cbCmd != 0)
430 {
431 if (!fUpdate)
432 {
433 pVGAState->pDrv->pfnVBVAUpdateBegin(pVGAState->pDrv, uScreenId);
434 fUpdate = true;
435 }
436
437 /* Updates the rectangle and sends the command to the VRDP server. */
438 pVGAState->pDrv->pfnVBVAUpdateProcess(pVGAState->pDrv, uScreenId, phdr, cbCmd);
439
440 int32_t xRight = phdr->x + phdr->w;
441 int32_t yBottom = phdr->y + phdr->h;
442
443 /* These are global coords, relative to the primary screen. */
444
445 LOGVBVABUFFER(("cbCmd = %d, x=%d, y=%d, w=%d, h=%d\n",
446 cbCmd, phdr->x, phdr->y, phdr->w, phdr->h));
447 LogRel3(("%s: update command cbCmd = %d, x=%d, y=%d, w=%d, h=%d\n",
448 __FUNCTION__, cbCmd, phdr->x, phdr->y, phdr->w, phdr->h));
449
450 /* Collect all rects into one. */
451 if (fDirtyEmpty)
452 {
453 /* This is the first rectangle to be added. */
454 dirtyRect.xLeft = phdr->x;
455 dirtyRect.yTop = phdr->y;
456 dirtyRect.xRight = xRight;
457 dirtyRect.yBottom = yBottom;
458 fDirtyEmpty = false;
459 }
460 else
461 {
462 /* Adjust region coordinates. */
463 if (dirtyRect.xLeft > phdr->x)
464 {
465 dirtyRect.xLeft = phdr->x;
466 }
467
468 if (dirtyRect.yTop > phdr->y)
469 {
470 dirtyRect.yTop = phdr->y;
471 }
472
473 if (dirtyRect.xRight < xRight)
474 {
475 dirtyRect.xRight = xRight;
476 }
477
478 if (dirtyRect.yBottom < yBottom)
479 {
480 dirtyRect.yBottom = yBottom;
481 }
482 }
483 }
484
485 vbvaReleaseCmd(pVBVAData, phdr, cbCmd);
486 }
487
488 if (fUpdate)
489 {
490 if (dirtyRect.xRight - dirtyRect.xLeft)
491 {
492 LogRel3(("%s: sending update screen=%d, x=%d, y=%d, w=%d, h=%d\n",
493 __FUNCTION__, uScreenId, dirtyRect.xLeft,
494 dirtyRect.yTop, dirtyRect.xRight - dirtyRect.xLeft,
495 dirtyRect.yBottom - dirtyRect.yTop));
496 pVGAState->pDrv->pfnVBVAUpdateEnd(pVGAState->pDrv, uScreenId, dirtyRect.xLeft, dirtyRect.yTop,
497 dirtyRect.xRight - dirtyRect.xLeft, dirtyRect.yBottom - dirtyRect.yTop);
498 }
499 else
500 {
501 pVGAState->pDrv->pfnVBVAUpdateEnd(pVGAState->pDrv, uScreenId, 0, 0, 0, 0);
502 }
503 }
504
505 return VINF_SUCCESS;
506}
507
508static int vbvaFlush(PVGASTATE pVGAState, VBVACONTEXT *pCtx)
509{
510 int rc = VINF_SUCCESS;
511
512 unsigned uScreenId;
513 for (uScreenId = 0; uScreenId < pCtx->cViews; uScreenId++)
514 {
515 VBVADATA *pVBVAData = &pCtx->aViews[uScreenId].vbva;
516
517 if (pVBVAData->guest.pVBVA)
518 {
519 rc = vbvaFlushProcess(uScreenId, pVGAState, pVBVAData);
520 if (RT_FAILURE(rc))
521 {
522 break;
523 }
524 }
525 }
526
527 if (RT_FAILURE(rc))
528 {
529 /* Turn off VBVA processing. */
530 LogRel(("VBVA: Disabling\n", rc));
531 pVGAState->fGuestCaps = 0;
532 pVGAState->pDrv->pfnVBVAGuestCapabilityUpdate(pVGAState->pDrv, pVGAState->fGuestCaps);
533 for (uScreenId = 0; uScreenId < pCtx->cViews; uScreenId++)
534 {
535 VBVADATA *pVBVAData = &pCtx->aViews[uScreenId].vbva;
536 if (pVBVAData->guest.pVBVA)
537 {
538 vbvaDataCleanup(pVBVAData);
539 pVGAState->pDrv->pfnVBVADisable(pVGAState->pDrv, uScreenId);
540 }
541 }
542 }
543
544 return rc;
545}
546
547static int vbvaResize(PVGASTATE pVGAState, VBVAVIEW *pView, const VBVAINFOSCREEN *pNewScreen)
548{
549 /* Callers ensure that pNewScreen contains valid data. */
550
551 /* Apply these changes. */
552 pView->screen = *pNewScreen;
553
554 uint8_t *pu8VRAM = pVGAState->vram_ptrR3 + pView->view.u32ViewOffset;
555 return pVGAState->pDrv->pfnVBVAResize (pVGAState->pDrv, &pView->view, &pView->screen, pu8VRAM);
556}
557
558static int vbvaEnable(unsigned uScreenId, PVGASTATE pVGAState, VBVACONTEXT *pCtx, VBVABUFFER *pVBVA, uint32_t u32Offset, bool fRestored)
559{
560 int rc;
561
562 /* Check if VBVABUFFER content makes sense. */
563 const VBVABUFFER parms = *pVBVA;
564
565 uint32_t cbVBVABuffer = RT_UOFFSETOF(VBVABUFFER, au8Data) + parms.cbData;
566 if ( parms.cbData > UINT32_MAX - RT_UOFFSETOF(VBVABUFFER, au8Data)
567 || cbVBVABuffer > pVGAState->vram_size
568 || u32Offset > pVGAState->vram_size - cbVBVABuffer)
569 {
570 return VERR_INVALID_PARAMETER;
571 }
572
573 if (!fRestored)
574 {
575 if ( parms.off32Data != 0
576 || parms.off32Free != 0
577 || parms.indexRecordFirst != 0
578 || parms.indexRecordFree != 0)
579 {
580 return VERR_INVALID_PARAMETER;
581 }
582 }
583
584 if ( parms.cbPartialWriteThreshold >= parms.cbData
585 || parms.cbPartialWriteThreshold == 0)
586 {
587 return VERR_INVALID_PARAMETER;
588 }
589
590 if (pVGAState->pDrv->pfnVBVAEnable)
591 {
592 RT_ZERO(pVBVA->hostFlags);
593 rc = pVGAState->pDrv->pfnVBVAEnable(pVGAState->pDrv, uScreenId, &pVBVA->hostFlags, false);
594 }
595 else
596 {
597 rc = VERR_NOT_SUPPORTED;
598 }
599
600 if (RT_SUCCESS(rc))
601 {
602 /* pVBVA->hostFlags has been set up by pfnVBVAEnable. */
603 LogFlowFunc(("u32HostEvents 0x%08X, u32SupportedOrders 0x%08X\n",
604 pVBVA->hostFlags.u32HostEvents, pVBVA->hostFlags.u32SupportedOrders));
605
606 VBVADATA *pVBVAData = &pCtx->aViews[uScreenId].vbva;
607 pVBVAData->guest.pVBVA = pVBVA;
608 pVBVAData->guest.pu8Data = &pVBVA->au8Data[0];
609 pVBVAData->u32VBVAOffset = u32Offset;
610 pVBVAData->off32Data = parms.off32Data;
611 pVBVAData->indexRecordFirst = parms.indexRecordFirst;
612 pVBVAData->cbPartialWriteThreshold = parms.cbPartialWriteThreshold;
613 pVBVAData->cbData = parms.cbData;
614
615 if (!fRestored)
616 {
617 /* @todo Actually this function must not touch the partialRecord structure at all,
618 * because initially it is a zero and when VBVA is disabled this should be set to zero.
619 * But I'm not sure that no code depends on zeroing partialRecord here.
620 * So for now (a quick fix for 4.1) just do not do this if the VM was restored,
621 * when partialRecord might be loaded already from the saved state.
622 */
623 pVBVAData->partialRecord.pu8 = NULL;
624 pVBVAData->partialRecord.cb = 0;
625 }
626
627 /* VBVA is working so disable the pause. */
628 pCtx->fPaused = false;
629 }
630
631 return rc;
632}
633
634static int vbvaDisable (unsigned uScreenId, PVGASTATE pVGAState, VBVACONTEXT *pCtx)
635{
636 /* Process any pending orders and empty the VBVA ring buffer. */
637 vbvaFlush (pVGAState, pCtx);
638
639 VBVADATA *pVBVAData = &pCtx->aViews[uScreenId].vbva;
640 vbvaDataCleanup(pVBVAData);
641
642 if (uScreenId == 0)
643 {
644 pVGAState->fGuestCaps = 0;
645 pVGAState->pDrv->pfnVBVAGuestCapabilityUpdate(pVGAState->pDrv, pVGAState->fGuestCaps);
646 }
647 pVGAState->pDrv->pfnVBVADisable(pVGAState->pDrv, uScreenId);
648 return VINF_SUCCESS;
649}
650
651bool VBVAIsEnabled(PVGASTATE pVGAState)
652{
653 PHGSMIINSTANCE pHGSMI = pVGAState->pHGSMI;
654 if (pHGSMI)
655 {
656 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext(pHGSMI);
657 if (pCtx)
658 {
659 if (pCtx->cViews)
660 {
661 VBVAVIEW * pView = &pCtx->aViews[0];
662 if (pView->vbva.guest.pVBVA)
663 return true;
664 }
665 }
666 }
667 return false;
668}
669
670#ifdef DEBUG_sunlover
671void dumpMouseShapeInfo(const VBVAMOUSESHAPEINFO *pMouseShapeInfo)
672{
673 LogFlow(("fSet = %d, fVisible %d, fAlpha %d, @%d,%d %dx%d (%p, %d/%d)\n",
674 pMouseShapeInfo->fSet,
675 pMouseShapeInfo->fVisible,
676 pMouseShapeInfo->fAlpha,
677 pMouseShapeInfo->u32HotX,
678 pMouseShapeInfo->u32HotY,
679 pMouseShapeInfo->u32Width,
680 pMouseShapeInfo->u32Height,
681 pMouseShapeInfo->pu8Shape,
682 pMouseShapeInfo->cbShape,
683 pMouseShapeInfo->cbAllocated
684 ));
685}
686#endif
687
688static int vbvaUpdateMousePointerShape(PVGASTATE pVGAState, VBVAMOUSESHAPEINFO *pMouseShapeInfo, bool fShape)
689{
690 LogFlowFunc(("pVGAState %p, pMouseShapeInfo %p, fShape %d\n",
691 pVGAState, pMouseShapeInfo, fShape));
692#ifdef DEBUG_sunlover
693 dumpMouseShapeInfo(pMouseShapeInfo);
694#endif
695
696 if (pVGAState->pDrv->pfnVBVAMousePointerShape == NULL)
697 {
698 return VERR_NOT_SUPPORTED;
699 }
700
701 int rc;
702 if (fShape && pMouseShapeInfo->pu8Shape != NULL)
703 {
704 rc = pVGAState->pDrv->pfnVBVAMousePointerShape (pVGAState->pDrv,
705 pMouseShapeInfo->fVisible,
706 pMouseShapeInfo->fAlpha,
707 pMouseShapeInfo->u32HotX,
708 pMouseShapeInfo->u32HotY,
709 pMouseShapeInfo->u32Width,
710 pMouseShapeInfo->u32Height,
711 pMouseShapeInfo->pu8Shape);
712 }
713 else
714 {
715 rc = pVGAState->pDrv->pfnVBVAMousePointerShape (pVGAState->pDrv,
716 pMouseShapeInfo->fVisible,
717 false,
718 0, 0,
719 0, 0,
720 NULL);
721 }
722
723 return rc;
724}
725
726static int vbvaMousePointerShape(PVGASTATE pVGAState, VBVACONTEXT *pCtx, const VBVAMOUSEPOINTERSHAPE *pShape, HGSMISIZE cbShape)
727{
728 const VBVAMOUSEPOINTERSHAPE parms = *pShape;
729
730 LogFlowFunc(("VBVA_MOUSE_POINTER_SHAPE: i32Result 0x%x, fu32Flags 0x%x, hot spot %d,%d, size %dx%d\n",
731 parms.i32Result,
732 parms.fu32Flags,
733 parms.u32HotX,
734 parms.u32HotY,
735 parms.u32Width,
736 parms.u32Height));
737
738 const bool fVisible = RT_BOOL(parms.fu32Flags & VBOX_MOUSE_POINTER_VISIBLE);
739 const bool fAlpha = RT_BOOL(parms.fu32Flags & VBOX_MOUSE_POINTER_ALPHA);
740 const bool fShape = RT_BOOL(parms.fu32Flags & VBOX_MOUSE_POINTER_SHAPE);
741
742 HGSMISIZE cbPointerData = 0;
743
744 if (fShape)
745 {
746 if (parms.u32Width > 8192 || parms.u32Height > 8192)
747 {
748 Log(("vbvaMousePointerShape: unsupported size %ux%u\n",
749 parms.u32Width, parms.u32Height));
750 return VERR_INVALID_PARAMETER;
751 }
752
753 cbPointerData = ((((parms.u32Width + 7) / 8) * parms.u32Height + 3) & ~3)
754 + parms.u32Width * 4 * parms.u32Height;
755 }
756
757 if (cbPointerData > cbShape - RT_UOFFSETOF(VBVAMOUSEPOINTERSHAPE, au8Data))
758 {
759 Log(("vbvaMousePointerShape: calculated pointer data size is too big (%d bytes, limit %d)\n",
760 cbPointerData, cbShape - RT_UOFFSETOF(VBVAMOUSEPOINTERSHAPE, au8Data)));
761 return VERR_INVALID_PARAMETER;
762 }
763
764 /* Save mouse info it will be used to restore mouse pointer after restoring saved state. */
765 pCtx->mouseShapeInfo.fSet = true;
766 pCtx->mouseShapeInfo.fVisible = fVisible;
767 pCtx->mouseShapeInfo.fAlpha = fAlpha;
768 if (fShape)
769 {
770 /* Data related to shape. */
771 pCtx->mouseShapeInfo.u32HotX = parms.u32HotX;
772 pCtx->mouseShapeInfo.u32HotY = parms.u32HotY;
773 pCtx->mouseShapeInfo.u32Width = parms.u32Width;
774 pCtx->mouseShapeInfo.u32Height = parms.u32Height;
775
776 /* Reallocate memory buffer if necessary. */
777 if (cbPointerData > pCtx->mouseShapeInfo.cbAllocated)
778 {
779 RTMemFree (pCtx->mouseShapeInfo.pu8Shape);
780 pCtx->mouseShapeInfo.pu8Shape = NULL;
781 pCtx->mouseShapeInfo.cbShape = 0;
782
783 uint8_t *pu8Shape = (uint8_t *)RTMemAlloc (cbPointerData);
784 if (pu8Shape)
785 {
786 pCtx->mouseShapeInfo.pu8Shape = pu8Shape;
787 pCtx->mouseShapeInfo.cbAllocated = cbPointerData;
788 }
789 }
790
791 /* Copy shape bitmaps. */
792 if (pCtx->mouseShapeInfo.pu8Shape)
793 {
794 memcpy(pCtx->mouseShapeInfo.pu8Shape, &pShape->au8Data[0], cbPointerData);
795 pCtx->mouseShapeInfo.cbShape = cbPointerData;
796 }
797 }
798
799 int rc = vbvaUpdateMousePointerShape(pVGAState, &pCtx->mouseShapeInfo, fShape);
800
801 return rc;
802}
803
804static uint32_t vbvaViewFromBufferPtr(PHGSMIINSTANCE pIns, const VBVACONTEXT *pCtx, const void *pvBuffer)
805{
806 /* Check which view contains the buffer. */
807 HGSMIOFFSET offBuffer = HGSMIPointerToOffsetHost(pIns, pvBuffer);
808
809 if (offBuffer != HGSMIOFFSET_VOID)
810 {
811 unsigned uScreenId;
812 for (uScreenId = 0; uScreenId < pCtx->cViews; uScreenId++)
813 {
814 const VBVAINFOVIEW *pView = &pCtx->aViews[uScreenId].view;
815
816 if ( pView->u32ViewSize > 0
817 && pView->u32ViewOffset <= offBuffer
818 && offBuffer <= pView->u32ViewOffset + pView->u32ViewSize - 1)
819 {
820 return pView->u32ViewIndex;
821 }
822 }
823 }
824
825 return UINT32_C(~0);
826}
827
828#ifdef DEBUG_sunlover
829static void dumpctx(const VBVACONTEXT *pCtx)
830{
831 Log(("VBVACONTEXT dump: cViews %d\n", pCtx->cViews));
832
833 uint32_t iView;
834 for (iView = 0; iView < pCtx->cViews; iView++)
835 {
836 const VBVAVIEW *pView = &pCtx->aViews[iView];
837
838 Log((" view %d o 0x%x s 0x%x m 0x%x\n",
839 pView->view.u32ViewIndex,
840 pView->view.u32ViewOffset,
841 pView->view.u32ViewSize,
842 pView->view.u32MaxScreenSize));
843
844 Log((" screen %d @%d,%d s 0x%x l 0x%x %dx%d bpp %d f 0x%x\n",
845 pView->screen.u32ViewIndex,
846 pView->screen.i32OriginX,
847 pView->screen.i32OriginY,
848 pView->screen.u32StartOffset,
849 pView->screen.u32LineSize,
850 pView->screen.u32Width,
851 pView->screen.u32Height,
852 pView->screen.u16BitsPerPixel,
853 pView->screen.u16Flags));
854
855 Log((" VBVA o 0x%x p %p\n",
856 pView->vbva.u32VBVAOffset,
857 pView->vbva.guest.pVBVA));
858
859 Log((" PR cb 0x%x p %p\n",
860 pView->vbva.partialRecord.cb,
861 pView->vbva.partialRecord.pu8));
862 }
863
864 dumpMouseShapeInfo(&pCtx->mouseShapeInfo);
865}
866#endif /* DEBUG_sunlover */
867
868#define VBOXVBVASAVEDSTATE_VHWAAVAILABLE_MAGIC 0x12345678
869#define VBOXVBVASAVEDSTATE_VHWAUNAVAILABLE_MAGIC 0x9abcdef0
870
871#ifdef VBOX_WITH_VIDEOHWACCEL
872static void vbvaVHWAHHCommandReinit(VBOXVHWACMD* pHdr, VBOXVHWACMD_TYPE enmCmd, int32_t iDisplay)
873{
874 memset(pHdr, 0, VBOXVHWACMD_HEADSIZE());
875 pHdr->cRefs = 1;
876 pHdr->iDisplay = iDisplay;
877 pHdr->rc = VERR_NOT_IMPLEMENTED;
878 pHdr->enmCmd = enmCmd;
879 pHdr->Flags = VBOXVHWACMD_FLAG_HH_CMD;
880}
881
882static VBOXVHWACMD* vbvaVHWAHHCommandCreate (PVGASTATE pVGAState, VBOXVHWACMD_TYPE enmCmd, int32_t iDisplay, VBOXVHWACMD_LENGTH cbCmd)
883{
884 VBOXVHWACMD* pHdr = (VBOXVHWACMD*)RTMemAllocZ(cbCmd + VBOXVHWACMD_HEADSIZE());
885 Assert(pHdr);
886 if (pHdr)
887 vbvaVHWAHHCommandReinit(pHdr, enmCmd, iDisplay);
888
889 return pHdr;
890}
891
892DECLINLINE(void) vbvaVHWAHHCommandRelease (VBOXVHWACMD* pCmd)
893{
894 uint32_t cRefs = ASMAtomicDecU32(&pCmd->cRefs);
895 if(!cRefs)
896 {
897 RTMemFree(pCmd);
898 }
899}
900
901DECLINLINE(void) vbvaVHWAHHCommandRetain (VBOXVHWACMD* pCmd)
902{
903 ASMAtomicIncU32(&pCmd->cRefs);
904}
905
906static void vbvaVHWACommandComplete(PVGASTATE pVGAState, PVBOXVHWACMD pCommand, bool fAsyncCommand)
907{
908 if (fAsyncCommand)
909 {
910 Assert(pCommand->Flags & VBOXVHWACMD_FLAG_HG_ASYNCH);
911 vbvaVHWACommandCompleteAsync(&pVGAState->IVBVACallbacks, pCommand);
912 }
913 else
914 {
915 Log(("VGA Command <<< Sync rc %d %#p, %d\n", pCommand->rc, pCommand, pCommand->enmCmd));
916 pCommand->Flags &= (~VBOXVHWACMD_FLAG_HG_ASYNCH);
917 }
918
919}
920
921static void vbvaVHWACommandCompleteAllPending(PVGASTATE pVGAState, int rc)
922{
923 if (!ASMAtomicUoReadU32(&pVGAState->pendingVhwaCommands.cPending))
924 return;
925
926 VBOX_VHWA_PENDINGCMD *pIter, *pNext;
927
928 PDMCritSectEnter(&pVGAState->CritSect, VERR_SEM_BUSY);
929
930 RTListForEachSafe(&pVGAState->pendingVhwaCommands.PendingList, pIter, pNext, VBOX_VHWA_PENDINGCMD, Node)
931 {
932 pIter->pCommand->rc = rc;
933 vbvaVHWACommandComplete(pVGAState, pIter->pCommand, true);
934
935 /* the command is submitted/processed, remove from the pend list */
936 RTListNodeRemove(&pIter->Node);
937 ASMAtomicDecU32(&pVGAState->pendingVhwaCommands.cPending);
938 RTMemFree(pIter);
939 }
940
941 PDMCritSectLeave(&pVGAState->CritSect);
942}
943
944static void vbvaVHWACommandClearAllPending(PVGASTATE pVGAState)
945{
946 if (!ASMAtomicUoReadU32(&pVGAState->pendingVhwaCommands.cPending))
947 return;
948
949 VBOX_VHWA_PENDINGCMD *pIter, *pNext;
950
951 PDMCritSectEnter(&pVGAState->CritSect, VERR_SEM_BUSY);
952
953 RTListForEachSafe(&pVGAState->pendingVhwaCommands.PendingList, pIter, pNext, VBOX_VHWA_PENDINGCMD, Node)
954 {
955 RTListNodeRemove(&pIter->Node);
956 ASMAtomicDecU32(&pVGAState->pendingVhwaCommands.cPending);
957 RTMemFree(pIter);
958 }
959
960 PDMCritSectLeave(&pVGAState->CritSect);
961}
962
963static void vbvaVHWACommandPend(PVGASTATE pVGAState, PVBOXVHWACMD pCommand)
964{
965 int rc = VERR_BUFFER_OVERFLOW;
966
967 if (ASMAtomicUoReadU32(&pVGAState->pendingVhwaCommands.cPending) < VBOX_VHWA_MAX_PENDING_COMMANDS)
968 {
969 VBOX_VHWA_PENDINGCMD *pPend = (VBOX_VHWA_PENDINGCMD*)RTMemAlloc(sizeof (*pPend));
970 if (pPend)
971 {
972 pCommand->Flags |= VBOXVHWACMD_FLAG_HG_ASYNCH;
973 pPend->pCommand = pCommand;
974 PDMCritSectEnter(&pVGAState->CritSect, VERR_SEM_BUSY);
975 if (ASMAtomicUoReadU32(&pVGAState->pendingVhwaCommands.cPending) < VBOX_VHWA_MAX_PENDING_COMMANDS)
976 {
977 RTListAppend(&pVGAState->pendingVhwaCommands.PendingList, &pPend->Node);
978 ASMAtomicIncU32(&pVGAState->pendingVhwaCommands.cPending);
979 PDMCritSectLeave(&pVGAState->CritSect);
980 return;
981 }
982 PDMCritSectLeave(&pVGAState->CritSect);
983 LogRel(("VBVA: Pending command count has reached its threshold.. completing them all.."));
984 RTMemFree(pPend);
985 }
986 else
987 rc = VERR_NO_MEMORY;
988 }
989 else
990 LogRel(("VBVA: Pending command count has reached its threshold, completing them all.."));
991
992 vbvaVHWACommandCompleteAllPending(pVGAState, rc);
993
994 pCommand->rc = rc;
995
996 vbvaVHWACommandComplete(pVGAState, pCommand, false);
997}
998
999static bool vbvaVHWACommandCanPend(PVBOXVHWACMD pCommand)
1000{
1001 switch (pCommand->enmCmd)
1002 {
1003 case VBOXVHWACMD_TYPE_HH_CONSTRUCT:
1004 case VBOXVHWACMD_TYPE_HH_SAVESTATE_SAVEBEGIN:
1005 case VBOXVHWACMD_TYPE_HH_SAVESTATE_SAVEEND:
1006 case VBOXVHWACMD_TYPE_HH_SAVESTATE_SAVEPERFORM:
1007 case VBOXVHWACMD_TYPE_HH_SAVESTATE_LOADPERFORM:
1008 return false;
1009 default:
1010 return true;
1011 }
1012}
1013
1014static int vbvaVHWACommandSavePending(PVGASTATE pVGAState, PSSMHANDLE pSSM)
1015{
1016 int rc = SSMR3PutU32(pSSM, pVGAState->pendingVhwaCommands.cPending);
1017 AssertRCReturn(rc, rc);
1018 VBOX_VHWA_PENDINGCMD *pIter;
1019 RTListForEach(&pVGAState->pendingVhwaCommands.PendingList, pIter, VBOX_VHWA_PENDINGCMD, Node)
1020 {
1021 rc = SSMR3PutU32(pSSM, (uint32_t)(((uint8_t*)pIter->pCommand) - ((uint8_t*)pVGAState->vram_ptrR3)));
1022 AssertRCReturn(rc, rc);
1023 }
1024 return rc;
1025}
1026
1027static int vbvaVHWACommandLoadPending(PVGASTATE pVGAState, PSSMHANDLE pSSM, uint32_t u32Version)
1028{
1029 if (u32Version < VGA_SAVEDSTATE_VERSION_WITH_PENDVHWA)
1030 return VINF_SUCCESS;
1031
1032 int rc;
1033 uint32_t u32;
1034 rc = SSMR3GetU32(pSSM, &u32);
1035 AssertRCReturn(rc, rc);
1036 for (uint32_t i = 0; i < u32; ++i)
1037 {
1038 uint32_t off32;
1039 rc = SSMR3GetU32(pSSM, &off32);
1040 AssertRCReturn(rc, rc);
1041 PVBOXVHWACMD pCommand = (PVBOXVHWACMD)(((uint8_t*)pVGAState->vram_ptrR3) + off32);
1042 vbvaVHWACommandPend(pVGAState, pCommand);
1043 }
1044 return rc;
1045}
1046
1047
1048static bool vbvaVHWACommandSubmit(PVGASTATE pVGAState, PVBOXVHWACMD pCommand, bool fAsyncCommand)
1049{
1050 unsigned id = (unsigned)pCommand->iDisplay;
1051 bool fPend = false;
1052
1053 if (pVGAState->pDrv->pfnVHWACommandProcess)
1054 {
1055 Log(("VGA Command >>> %#p, %d\n", pCommand, pCommand->enmCmd));
1056 int rc = pVGAState->pDrv->pfnVHWACommandProcess(pVGAState->pDrv, pCommand);
1057 if (rc == VINF_CALLBACK_RETURN)
1058 {
1059 Log(("VGA Command --- Going Async %#p, %d\n", pCommand, pCommand->enmCmd));
1060 return true; /* command will be completed asynchronously, return right away */
1061 }
1062 else if (rc == VERR_INVALID_STATE)
1063 {
1064 Log(("VGA Command --- Trying Pend %#p, %d\n", pCommand, pCommand->enmCmd));
1065 fPend = vbvaVHWACommandCanPend(pCommand);
1066 if (!fPend)
1067 {
1068 Log(("VGA Command --- Can NOT Pend %#p, %d\n", pCommand, pCommand->enmCmd));
1069 pCommand->rc = rc;
1070 }
1071 else
1072 Log(("VGA Command --- Can Pend %#p, %d\n", pCommand, pCommand->enmCmd));
1073 }
1074 else
1075 {
1076 Log(("VGA Command --- Going Complete Sync rc %d %#p, %d\n", rc, pCommand, pCommand->enmCmd));
1077 pCommand->rc = rc;
1078 }
1079
1080 /* the command was completed, take a special care about it (seee below) */
1081 }
1082 else
1083 {
1084 AssertFailed();
1085 pCommand->rc = VERR_INVALID_STATE;
1086 }
1087
1088 if (fPend)
1089 return false;
1090
1091 vbvaVHWACommandComplete(pVGAState, pCommand, fAsyncCommand);
1092
1093 return true;
1094}
1095
1096static bool vbvaVHWACheckPendingCommands(PVGASTATE pVGAState)
1097{
1098 if (!ASMAtomicUoReadU32(&pVGAState->pendingVhwaCommands.cPending))
1099 return true;
1100
1101 VBOX_VHWA_PENDINGCMD *pIter, *pNext;
1102
1103 PDMCritSectEnter(&pVGAState->CritSect, VERR_SEM_BUSY);
1104
1105 RTListForEachSafe(&pVGAState->pendingVhwaCommands.PendingList, pIter, pNext, VBOX_VHWA_PENDINGCMD, Node)
1106 {
1107 if (!vbvaVHWACommandSubmit(pVGAState, pIter->pCommand, true))
1108 {
1109 PDMCritSectLeave(&pVGAState->CritSect);
1110 return false; /* the command should be pended still */
1111 }
1112
1113 /* the command is submitted/processed, remove from the pend list */
1114 RTListNodeRemove(&pIter->Node);
1115 ASMAtomicDecU32(&pVGAState->pendingVhwaCommands.cPending);
1116 RTMemFree(pIter);
1117 }
1118
1119 PDMCritSectLeave(&pVGAState->CritSect);
1120
1121 return true;
1122}
1123
1124void vbvaTimerCb(PVGASTATE pVGAState)
1125{
1126 vbvaVHWACheckPendingCommands(pVGAState);
1127}
1128static void vbvaVHWAHandleCommand(PVGASTATE pVGAState, PVBOXVHWACMD pCmd)
1129{
1130 if (vbvaVHWACheckPendingCommands(pVGAState))
1131 {
1132 if (vbvaVHWACommandSubmit(pVGAState, pCmd, false))
1133 return;
1134 }
1135
1136 vbvaVHWACommandPend(pVGAState, pCmd);
1137}
1138
1139static DECLCALLBACK(void) vbvaVHWAHHCommandSetEventCallback(void * pContext)
1140{
1141 RTSemEventSignal((RTSEMEVENT)pContext);
1142}
1143
1144static int vbvaVHWAHHCommandPost(PVGASTATE pVGAState, VBOXVHWACMD* pCmd)
1145{
1146 RTSEMEVENT hComplEvent;
1147 int rc = RTSemEventCreate(&hComplEvent);
1148 AssertRC(rc);
1149 if(RT_SUCCESS(rc))
1150 {
1151 /* ensure the cmd is not deleted until we process it */
1152 vbvaVHWAHHCommandRetain (pCmd);
1153 VBOXVHWA_HH_CALLBACK_SET(pCmd, vbvaVHWAHHCommandSetEventCallback, (void*)hComplEvent);
1154 vbvaVHWAHandleCommand(pVGAState, pCmd);
1155 if((ASMAtomicReadU32((volatile uint32_t *)&pCmd->Flags) & VBOXVHWACMD_FLAG_HG_ASYNCH) != 0)
1156 {
1157 rc = RTSemEventWaitNoResume(hComplEvent, RT_INDEFINITE_WAIT);
1158 }
1159 else
1160 {
1161 /* the command is completed */
1162 }
1163
1164 AssertRC(rc);
1165 if(RT_SUCCESS(rc))
1166 {
1167 RTSemEventDestroy(hComplEvent);
1168 }
1169 vbvaVHWAHHCommandRelease(pCmd);
1170 }
1171 return rc;
1172}
1173
1174int vbvaVHWAConstruct (PVGASTATE pVGAState)
1175{
1176 pVGAState->pendingVhwaCommands.cPending = 0;
1177 RTListInit(&pVGAState->pendingVhwaCommands.PendingList);
1178
1179 VBOXVHWACMD *pCmd = vbvaVHWAHHCommandCreate(pVGAState, VBOXVHWACMD_TYPE_HH_CONSTRUCT, 0, sizeof(VBOXVHWACMD_HH_CONSTRUCT));
1180 Assert(pCmd);
1181 if(pCmd)
1182 {
1183 uint32_t iDisplay = 0;
1184 int rc = VINF_SUCCESS;
1185 VBOXVHWACMD_HH_CONSTRUCT * pBody = VBOXVHWACMD_BODY(pCmd, VBOXVHWACMD_HH_CONSTRUCT);
1186
1187 do
1188 {
1189 memset(pBody, 0, sizeof(VBOXVHWACMD_HH_CONSTRUCT));
1190
1191 PPDMDEVINS pDevIns = pVGAState->pDevInsR3;
1192 PVM pVM = PDMDevHlpGetVM(pDevIns);
1193
1194 pBody->pVM = pVM;
1195 pBody->pvVRAM = pVGAState->vram_ptrR3;
1196 pBody->cbVRAM = pVGAState->vram_size;
1197
1198 rc = vbvaVHWAHHCommandPost(pVGAState, pCmd);
1199 AssertRC(rc);
1200 if(RT_SUCCESS(rc))
1201 {
1202 rc = pCmd->rc;
1203 AssertMsg(RT_SUCCESS(rc) || rc == VERR_NOT_IMPLEMENTED, ("%Rrc\n", rc));
1204 if(rc == VERR_NOT_IMPLEMENTED)
1205 {
1206 /* @todo: set some flag in pVGAState indicating VHWA is not supported */
1207 /* VERR_NOT_IMPLEMENTED is not a failure, we just do not support it */
1208 rc = VINF_SUCCESS;
1209 }
1210
1211 if (!RT_SUCCESS(rc))
1212 break;
1213 }
1214 else
1215 break;
1216
1217 ++iDisplay;
1218 if (iDisplay >= pVGAState->cMonitors)
1219 break;
1220 vbvaVHWAHHCommandReinit(pCmd, VBOXVHWACMD_TYPE_HH_CONSTRUCT, (int32_t)iDisplay);
1221 } while (true);
1222
1223 vbvaVHWAHHCommandRelease(pCmd);
1224
1225 return rc;
1226 }
1227 return VERR_OUT_OF_RESOURCES;
1228}
1229
1230int vbvaVHWAReset (PVGASTATE pVGAState)
1231{
1232 vbvaVHWACommandClearAllPending(pVGAState);
1233
1234 /* ensure we have all pending cmds processed and h->g cmds disabled */
1235 VBOXVHWACMD *pCmd = vbvaVHWAHHCommandCreate(pVGAState, VBOXVHWACMD_TYPE_HH_RESET, 0, 0);
1236 Assert(pCmd);
1237 if(pCmd)
1238 {
1239 int rc = VINF_SUCCESS;
1240 uint32_t iDisplay = 0;
1241
1242 do
1243 {
1244 rc =vbvaVHWAHHCommandPost(pVGAState, pCmd);
1245 AssertRC(rc);
1246 if(RT_SUCCESS(rc))
1247 {
1248 rc = pCmd->rc;
1249 AssertMsg(RT_SUCCESS(rc) || rc == VERR_NOT_IMPLEMENTED, ("%Rrc\n", rc));
1250 if (rc == VERR_NOT_IMPLEMENTED)
1251 rc = VINF_SUCCESS;
1252 }
1253
1254 if (!RT_SUCCESS(rc))
1255 break;
1256
1257 ++iDisplay;
1258 if (iDisplay >= pVGAState->cMonitors)
1259 break;
1260 vbvaVHWAHHCommandReinit(pCmd, VBOXVHWACMD_TYPE_HH_RESET, (int32_t)iDisplay);
1261
1262 } while (true);
1263
1264 vbvaVHWAHHCommandRelease(pCmd);
1265
1266 return rc;
1267 }
1268 return VERR_OUT_OF_RESOURCES;
1269}
1270
1271typedef DECLCALLBACK(bool) FNVBOXVHWAHHCMDPRECB(PVGASTATE pVGAState, VBOXVHWACMD *pCmd, uint32_t iDisplay, void *pvContext);
1272typedef FNVBOXVHWAHHCMDPRECB *PFNVBOXVHWAHHCMDPRECB;
1273
1274typedef DECLCALLBACK(bool) FNVBOXVHWAHHCMDPOSTCB(PVGASTATE pVGAState, VBOXVHWACMD *pCmd, uint32_t iDisplay, int rc, void *pvContext);
1275typedef FNVBOXVHWAHHCMDPOSTCB *PFNVBOXVHWAHHCMDPOSTCB;
1276
1277int vbvaVHWAHHPost(PVGASTATE pVGAState, VBOXVHWACMD *pCmd, PFNVBOXVHWAHHCMDPRECB pfnPre, PFNVBOXVHWAHHCMDPOSTCB pfnPost, void *pvContext)
1278{
1279 const VBOXVHWACMD_TYPE enmType = pCmd->enmCmd;
1280 int rc = VINF_SUCCESS;
1281 uint32_t iDisplay = 0;
1282
1283 do
1284 {
1285 if (!pfnPre || pfnPre(pVGAState, pCmd, iDisplay, pvContext))
1286 {
1287 rc = vbvaVHWAHHCommandPost(pVGAState, pCmd);
1288 AssertRC(rc);
1289 if (pfnPost)
1290 {
1291 if (!pfnPost(pVGAState, pCmd, iDisplay, rc, pvContext))
1292 {
1293 rc = VINF_SUCCESS;
1294 break;
1295 }
1296 rc = VINF_SUCCESS;
1297 }
1298 else if(RT_SUCCESS(rc))
1299 {
1300 rc = pCmd->rc;
1301 AssertMsg(RT_SUCCESS(rc) || rc == VERR_NOT_IMPLEMENTED, ("%Rrc\n", rc));
1302 if(rc == VERR_NOT_IMPLEMENTED)
1303 {
1304 rc = VINF_SUCCESS;
1305 }
1306 }
1307
1308 if (!RT_SUCCESS(rc))
1309 break;
1310 }
1311
1312 ++iDisplay;
1313 if (iDisplay >= pVGAState->cMonitors)
1314 break;
1315 vbvaVHWAHHCommandReinit(pCmd, enmType, (int32_t)iDisplay);
1316 } while (true);
1317
1318 return rc;
1319}
1320
1321/* @todo call this also on reset? */
1322int vbvaVHWAEnable (PVGASTATE pVGAState, bool bEnable)
1323{
1324 const VBOXVHWACMD_TYPE enmType = bEnable ? VBOXVHWACMD_TYPE_HH_ENABLE : VBOXVHWACMD_TYPE_HH_DISABLE;
1325 VBOXVHWACMD *pCmd = vbvaVHWAHHCommandCreate(pVGAState,
1326 enmType,
1327 0, 0);
1328 Assert(pCmd);
1329 if(pCmd)
1330 {
1331 int rc = vbvaVHWAHHPost (pVGAState, pCmd, NULL, NULL, NULL);
1332 vbvaVHWAHHCommandRelease(pCmd);
1333 return rc;
1334 }
1335 return VERR_OUT_OF_RESOURCES;
1336}
1337
1338int vboxVBVASaveStatePrep (PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
1339{
1340 /* ensure we have no pending commands */
1341 return vbvaVHWAEnable(PDMINS_2_DATA(pDevIns, PVGASTATE), false);
1342}
1343
1344int vboxVBVASaveStateDone (PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
1345{
1346 /* ensure we have no pending commands */
1347 return vbvaVHWAEnable(PDMINS_2_DATA(pDevIns, PVGASTATE), true);
1348}
1349
1350int vbvaVHWACommandCompleteAsync(PPDMIDISPLAYVBVACALLBACKS pInterface, PVBOXVHWACMD pCmd)
1351{
1352 int rc;
1353 Log(("VGA Command <<< Async rc %d %#p, %d\n", pCmd->rc, pCmd, pCmd->enmCmd));
1354
1355 if((pCmd->Flags & VBOXVHWACMD_FLAG_HH_CMD) == 0)
1356 {
1357 PVGASTATE pVGAState = PPDMIDISPLAYVBVACALLBACKS_2_PVGASTATE(pInterface);
1358 PHGSMIINSTANCE pIns = pVGAState->pHGSMI;
1359
1360 Assert(pCmd->Flags & VBOXVHWACMD_FLAG_HG_ASYNCH);
1361#ifdef VBOX_WITH_WDDM
1362 if (pVGAState->fGuestCaps & VBVACAPS_COMPLETEGCMD_BY_IOREAD)
1363 {
1364 rc = HGSMICompleteGuestCommand(pIns, pCmd, !!(pCmd->Flags & VBOXVHWACMD_FLAG_GH_ASYNCH_IRQ));
1365 AssertRC(rc);
1366 }
1367 else
1368#endif
1369 {
1370 VBVAHOSTCMD *pHostCmd;
1371 int32_t iDisplay = pCmd->iDisplay;
1372
1373 if(pCmd->Flags & VBOXVHWACMD_FLAG_GH_ASYNCH_EVENT)
1374 {
1375 rc = HGSMIHostCommandAlloc (pIns,
1376 (void**)&pHostCmd,
1377 VBVAHOSTCMD_SIZE(sizeof(VBVAHOSTCMDEVENT)),
1378 HGSMI_CH_VBVA,
1379 VBVAHG_EVENT);
1380 AssertRC(rc);
1381 if(RT_SUCCESS(rc))
1382 {
1383 memset(pHostCmd, 0 , VBVAHOSTCMD_SIZE(sizeof(VBVAHOSTCMDEVENT)));
1384 pHostCmd->iDstID = pCmd->iDisplay;
1385 pHostCmd->customOpCode = 0;
1386 VBVAHOSTCMDEVENT *pBody = VBVAHOSTCMD_BODY(pHostCmd, VBVAHOSTCMDEVENT);
1387 pBody->pEvent = pCmd->GuestVBVAReserved1;
1388 }
1389 }
1390 else
1391 {
1392 HGSMIOFFSET offCmd = HGSMIPointerToOffsetHost (pIns, pCmd);
1393 Assert(offCmd != HGSMIOFFSET_VOID);
1394 if(offCmd != HGSMIOFFSET_VOID)
1395 {
1396 rc = HGSMIHostCommandAlloc (pIns,
1397 (void**)&pHostCmd,
1398 VBVAHOSTCMD_SIZE(sizeof(VBVAHOSTCMDVHWACMDCOMPLETE)),
1399 HGSMI_CH_VBVA,
1400 VBVAHG_DISPLAY_CUSTOM);
1401 AssertRC(rc);
1402 if(RT_SUCCESS(rc))
1403 {
1404 memset(pHostCmd, 0 , VBVAHOSTCMD_SIZE(sizeof(VBVAHOSTCMDVHWACMDCOMPLETE)));
1405 pHostCmd->iDstID = pCmd->iDisplay;
1406 pHostCmd->customOpCode = VBVAHG_DCUSTOM_VHWA_CMDCOMPLETE;
1407 VBVAHOSTCMDVHWACMDCOMPLETE *pBody = VBVAHOSTCMD_BODY(pHostCmd, VBVAHOSTCMDVHWACMDCOMPLETE);
1408 pBody->offCmd = offCmd;
1409 }
1410 }
1411 else
1412 {
1413 rc = VERR_INVALID_PARAMETER;
1414 }
1415 }
1416
1417 if(RT_SUCCESS(rc))
1418 {
1419 rc = HGSMIHostCommandSubmitAndFreeAsynch(pIns, pHostCmd, RT_BOOL(pCmd->Flags & VBOXVHWACMD_FLAG_GH_ASYNCH_IRQ));
1420 AssertRC(rc);
1421 if(RT_SUCCESS(rc))
1422 {
1423 return rc;
1424 }
1425 HGSMIHostCommandFree (pIns, pHostCmd);
1426 }
1427 }
1428 }
1429 else
1430 {
1431 PFNVBOXVHWA_HH_CALLBACK pfn = VBOXVHWA_HH_CALLBACK_GET(pCmd);
1432 if(pfn)
1433 {
1434 pfn(VBOXVHWA_HH_CALLBACK_GET_ARG(pCmd));
1435 }
1436 rc = VINF_SUCCESS;
1437 }
1438 return rc;
1439}
1440
1441typedef struct VBOXVBVASAVEDSTATECBDATA
1442{
1443 PSSMHANDLE pSSM;
1444 int rc;
1445 bool ab2DOn[VBOX_VIDEO_MAX_SCREENS];
1446} VBOXVBVASAVEDSTATECBDATA, *PVBOXVBVASAVEDSTATECBDATA;
1447
1448static DECLCALLBACK(bool) vboxVBVASaveStateBeginPostCb(PVGASTATE pVGAState, VBOXVHWACMD *pCmd, uint32_t iDisplay, int rc, void *pvContext)
1449{
1450 PVBOXVBVASAVEDSTATECBDATA pData = (PVBOXVBVASAVEDSTATECBDATA)pvContext;
1451 if (RT_FAILURE(pData->rc))
1452 return false;
1453 if (RT_FAILURE(rc))
1454 {
1455 pData->rc = rc;
1456 return false;
1457 }
1458
1459 Assert(iDisplay < RT_ELEMENTS(pData->ab2DOn));
1460 if (iDisplay >= RT_ELEMENTS(pData->ab2DOn))
1461 {
1462 pData->rc = VERR_INVALID_PARAMETER;
1463 return false;
1464 }
1465
1466 Assert(RT_SUCCESS(pCmd->rc) || pCmd->rc == VERR_NOT_IMPLEMENTED);
1467 if (RT_SUCCESS(pCmd->rc))
1468 {
1469 pData->ab2DOn[iDisplay] = true;
1470 }
1471 else if (pCmd->rc != VERR_NOT_IMPLEMENTED)
1472 {
1473 pData->rc = pCmd->rc;
1474 return false;
1475 }
1476
1477 return true;
1478}
1479
1480static DECLCALLBACK(bool) vboxVBVASaveStatePerformPreCb(PVGASTATE pVGAState, VBOXVHWACMD *pCmd, uint32_t iDisplay, void *pvContext)
1481{
1482 PVBOXVBVASAVEDSTATECBDATA pData = (PVBOXVBVASAVEDSTATECBDATA)pvContext;
1483 if (RT_FAILURE(pData->rc))
1484 return false;
1485
1486 Assert(iDisplay < RT_ELEMENTS(pData->ab2DOn));
1487 if (iDisplay >= RT_ELEMENTS(pData->ab2DOn))
1488 {
1489 pData->rc = VERR_INVALID_PARAMETER;
1490 return false;
1491 }
1492
1493 int rc;
1494
1495 if (pData->ab2DOn[iDisplay])
1496 {
1497 rc = SSMR3PutU32 (pData->pSSM, VBOXVBVASAVEDSTATE_VHWAAVAILABLE_MAGIC); AssertRC(rc);
1498 if (RT_FAILURE(rc))
1499 {
1500 pData->rc = rc;
1501 return false;
1502 }
1503 return true;
1504 }
1505
1506 rc = SSMR3PutU32 (pData->pSSM, VBOXVBVASAVEDSTATE_VHWAUNAVAILABLE_MAGIC); AssertRC(rc);
1507 if (RT_FAILURE(rc))
1508 {
1509 pData->rc = rc;
1510 return false;
1511 }
1512
1513 return false;
1514}
1515
1516static DECLCALLBACK(bool) vboxVBVASaveStateEndPreCb(PVGASTATE pVGAState, VBOXVHWACMD *pCmd, uint32_t iDisplay, void *pvContext)
1517{
1518 PVBOXVBVASAVEDSTATECBDATA pData = (PVBOXVBVASAVEDSTATECBDATA)pvContext;
1519 Assert(iDisplay < RT_ELEMENTS(pData->ab2DOn));
1520 if (pData->ab2DOn[iDisplay])
1521 {
1522 return true;
1523 }
1524
1525 return false;
1526}
1527
1528static DECLCALLBACK(bool) vboxVBVALoadStatePerformPostCb(PVGASTATE pVGAState, VBOXVHWACMD *pCmd, uint32_t iDisplay, int rc, void *pvContext)
1529{
1530 PVBOXVBVASAVEDSTATECBDATA pData = (PVBOXVBVASAVEDSTATECBDATA)pvContext;
1531 if (RT_FAILURE(pData->rc))
1532 return false;
1533 if (RT_FAILURE(rc))
1534 {
1535 pData->rc = rc;
1536 return false;
1537 }
1538
1539 Assert(iDisplay < RT_ELEMENTS(pData->ab2DOn));
1540 if (iDisplay >= RT_ELEMENTS(pData->ab2DOn))
1541 {
1542 pData->rc = VERR_INVALID_PARAMETER;
1543 return false;
1544 }
1545
1546 Assert(RT_SUCCESS(pCmd->rc) || pCmd->rc == VERR_NOT_IMPLEMENTED);
1547 if (pCmd->rc == VERR_NOT_IMPLEMENTED)
1548 {
1549 pData->rc = SSMR3SkipToEndOfUnit(pData->pSSM);
1550 AssertRC(pData->rc);
1551 return false;
1552 }
1553 if (RT_FAILURE(pCmd->rc))
1554 {
1555 pData->rc = pCmd->rc;
1556 return false;
1557 }
1558
1559 return true;
1560}
1561
1562static DECLCALLBACK(bool) vboxVBVALoadStatePerformPreCb(PVGASTATE pVGAState, VBOXVHWACMD *pCmd, uint32_t iDisplay, void *pvContext)
1563{
1564 PVBOXVBVASAVEDSTATECBDATA pData = (PVBOXVBVASAVEDSTATECBDATA)pvContext;
1565 if (RT_FAILURE(pData->rc))
1566 return false;
1567
1568 Assert(iDisplay < RT_ELEMENTS(pData->ab2DOn));
1569 if (iDisplay >= RT_ELEMENTS(pData->ab2DOn))
1570 {
1571 pData->rc = VERR_INVALID_PARAMETER;
1572 return false;
1573 }
1574
1575 int rc;
1576 uint32_t u32;
1577 rc = SSMR3GetU32(pData->pSSM, &u32); AssertRC(rc);
1578 if (RT_FAILURE(rc))
1579 {
1580 pData->rc = rc;
1581 return false;
1582 }
1583
1584 switch (u32)
1585 {
1586 case VBOXVBVASAVEDSTATE_VHWAAVAILABLE_MAGIC:
1587 pData->ab2DOn[iDisplay] = true;
1588 return true;
1589 case VBOXVBVASAVEDSTATE_VHWAUNAVAILABLE_MAGIC:
1590 pData->ab2DOn[iDisplay] = false;
1591 return false;
1592 default:
1593 pData->rc = VERR_INVALID_STATE;
1594 return false;
1595 }
1596}
1597#endif /* #ifdef VBOX_WITH_VIDEOHWACCEL */
1598
1599int vboxVBVASaveDevStateExec (PVGASTATE pVGAState, PSSMHANDLE pSSM)
1600{
1601 PHGSMIINSTANCE pIns = pVGAState->pHGSMI;
1602 int rc = HGSMIHostSaveStateExec (pIns, pSSM);
1603 if (RT_SUCCESS(rc))
1604 {
1605 /* Save VBVACONTEXT. */
1606 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext (pIns);
1607
1608 if (!pCtx)
1609 {
1610 AssertFailed();
1611
1612 /* Still write a valid value to the SSM. */
1613 rc = SSMR3PutU32 (pSSM, 0);
1614 AssertRCReturn(rc, rc);
1615 }
1616 else
1617 {
1618#ifdef DEBUG_sunlover
1619 dumpctx(pCtx);
1620#endif
1621
1622 rc = SSMR3PutU32 (pSSM, pCtx->cViews);
1623 AssertRCReturn(rc, rc);
1624
1625 uint32_t iView;
1626 for (iView = 0; iView < pCtx->cViews; iView++)
1627 {
1628 VBVAVIEW *pView = &pCtx->aViews[iView];
1629
1630 rc = SSMR3PutU32 (pSSM, pView->view.u32ViewIndex);
1631 AssertRCReturn(rc, rc);
1632 rc = SSMR3PutU32 (pSSM, pView->view.u32ViewOffset);
1633 AssertRCReturn(rc, rc);
1634 rc = SSMR3PutU32 (pSSM, pView->view.u32ViewSize);
1635 AssertRCReturn(rc, rc);
1636 rc = SSMR3PutU32 (pSSM, pView->view.u32MaxScreenSize);
1637 AssertRCReturn(rc, rc);
1638
1639 rc = SSMR3PutU32 (pSSM, pView->screen.u32ViewIndex);
1640 AssertRCReturn(rc, rc);
1641 rc = SSMR3PutS32 (pSSM, pView->screen.i32OriginX);
1642 AssertRCReturn(rc, rc);
1643 rc = SSMR3PutS32 (pSSM, pView->screen.i32OriginY);
1644 AssertRCReturn(rc, rc);
1645 rc = SSMR3PutU32 (pSSM, pView->screen.u32StartOffset);
1646 AssertRCReturn(rc, rc);
1647 rc = SSMR3PutU32 (pSSM, pView->screen.u32LineSize);
1648 AssertRCReturn(rc, rc);
1649 rc = SSMR3PutU32 (pSSM, pView->screen.u32Width);
1650 AssertRCReturn(rc, rc);
1651 rc = SSMR3PutU32 (pSSM, pView->screen.u32Height);
1652 AssertRCReturn(rc, rc);
1653 rc = SSMR3PutU16 (pSSM, pView->screen.u16BitsPerPixel);
1654 AssertRCReturn(rc, rc);
1655 rc = SSMR3PutU16 (pSSM, pView->screen.u16Flags);
1656 AssertRCReturn(rc, rc);
1657
1658 rc = SSMR3PutU32 (pSSM, pView->vbva.guest.pVBVA? pView->vbva.u32VBVAOffset: HGSMIOFFSET_VOID);
1659 AssertRCReturn(rc, rc);
1660
1661 rc = SSMR3PutU32 (pSSM, pView->vbva.partialRecord.cb);
1662 AssertRCReturn(rc, rc);
1663
1664 if (pView->vbva.partialRecord.cb > 0)
1665 {
1666 rc = SSMR3PutMem (pSSM, pView->vbva.partialRecord.pu8, pView->vbva.partialRecord.cb);
1667 AssertRCReturn(rc, rc);
1668 }
1669 }
1670
1671 /* Save mouse pointer shape information. */
1672 rc = SSMR3PutBool (pSSM, pCtx->mouseShapeInfo.fSet);
1673 AssertRCReturn(rc, rc);
1674 rc = SSMR3PutBool (pSSM, pCtx->mouseShapeInfo.fVisible);
1675 AssertRCReturn(rc, rc);
1676 rc = SSMR3PutBool (pSSM, pCtx->mouseShapeInfo.fAlpha);
1677 AssertRCReturn(rc, rc);
1678 rc = SSMR3PutU32 (pSSM, pCtx->mouseShapeInfo.u32HotX);
1679 AssertRCReturn(rc, rc);
1680 rc = SSMR3PutU32 (pSSM, pCtx->mouseShapeInfo.u32HotY);
1681 AssertRCReturn(rc, rc);
1682 rc = SSMR3PutU32 (pSSM, pCtx->mouseShapeInfo.u32Width);
1683 AssertRCReturn(rc, rc);
1684 rc = SSMR3PutU32 (pSSM, pCtx->mouseShapeInfo.u32Height);
1685 AssertRCReturn(rc, rc);
1686 rc = SSMR3PutU32 (pSSM, pCtx->mouseShapeInfo.cbShape);
1687 AssertRCReturn(rc, rc);
1688 if (pCtx->mouseShapeInfo.cbShape)
1689 {
1690 rc = SSMR3PutMem (pSSM, pCtx->mouseShapeInfo.pu8Shape, pCtx->mouseShapeInfo.cbShape);
1691 AssertRCReturn(rc, rc);
1692 }
1693
1694#ifdef VBOX_WITH_WDDM
1695 /* Size of some additional data. For future extensions. */
1696 rc = SSMR3PutU32 (pSSM, 4);
1697 AssertRCReturn(rc, rc);
1698 rc = SSMR3PutU32 (pSSM, pVGAState->fGuestCaps);
1699 AssertRCReturn(rc, rc);
1700#else
1701 /* Size of some additional data. For future extensions. */
1702 rc = SSMR3PutU32 (pSSM, 0);
1703 AssertRCReturn(rc, rc);
1704#endif
1705 rc = SSMR3PutU32 (pSSM, RT_ELEMENTS(pCtx->aModeHints));
1706 AssertRCReturn(rc, rc);
1707 rc = SSMR3PutU32 (pSSM, sizeof(VBVAMODEHINT));
1708 AssertRCReturn(rc, rc);
1709 for (unsigned i = 0; i < RT_ELEMENTS(pCtx->aModeHints); ++i)
1710 {
1711 rc = SSMR3PutMem (pSSM, &pCtx->aModeHints[i],
1712 sizeof(VBVAMODEHINT));
1713 AssertRCReturn(rc, rc);
1714 }
1715 }
1716 }
1717
1718 return rc;
1719}
1720
1721int vboxVBVASaveStateExec (PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
1722{
1723 PVGASTATE pVGAState = PDMINS_2_DATA(pDevIns, PVGASTATE);
1724 int rc;
1725#ifdef VBOX_WITH_VIDEOHWACCEL
1726 VBOXVBVASAVEDSTATECBDATA VhwaData = {0};
1727 VhwaData.pSSM = pSSM;
1728 uint32_t cbCmd = sizeof (VBOXVHWACMD_HH_SAVESTATE_SAVEPERFORM); /* maximum cmd size */
1729 VBOXVHWACMD *pCmd = vbvaVHWAHHCommandCreate(pVGAState, VBOXVHWACMD_TYPE_HH_SAVESTATE_SAVEBEGIN, 0, cbCmd);
1730 Assert(pCmd);
1731 if(pCmd)
1732 {
1733 vbvaVHWAHHPost (pVGAState, pCmd, NULL, vboxVBVASaveStateBeginPostCb, &VhwaData);
1734 rc = VhwaData.rc;
1735 AssertRC(rc);
1736 if (RT_SUCCESS(rc))
1737 {
1738#endif
1739 rc = vboxVBVASaveDevStateExec (pVGAState, pSSM);
1740 AssertRC(rc);
1741#ifdef VBOX_WITH_VIDEOHWACCEL
1742 if (RT_SUCCESS(rc))
1743 {
1744 vbvaVHWAHHCommandReinit(pCmd, VBOXVHWACMD_TYPE_HH_SAVESTATE_SAVEPERFORM, 0);
1745 VBOXVHWACMD_HH_SAVESTATE_SAVEPERFORM *pSave = VBOXVHWACMD_BODY(pCmd, VBOXVHWACMD_HH_SAVESTATE_SAVEPERFORM);
1746 pSave->pSSM = pSSM;
1747 vbvaVHWAHHPost (pVGAState, pCmd, vboxVBVASaveStatePerformPreCb, NULL, &VhwaData);
1748 rc = VhwaData.rc;
1749 AssertRC(rc);
1750 if (RT_SUCCESS(rc))
1751 {
1752 rc = vbvaVHWACommandSavePending(pVGAState, pSSM);
1753 AssertRCReturn(rc, rc);
1754
1755 vbvaVHWAHHCommandReinit(pCmd, VBOXVHWACMD_TYPE_HH_SAVESTATE_SAVEEND, 0);
1756 vbvaVHWAHHPost (pVGAState, pCmd, vboxVBVASaveStateEndPreCb, NULL, &VhwaData);
1757 rc = VhwaData.rc;
1758 AssertRC(rc);
1759 }
1760 }
1761 }
1762
1763 vbvaVHWAHHCommandRelease(pCmd);
1764 }
1765 else
1766 rc = VERR_OUT_OF_RESOURCES;
1767#else
1768 if (RT_SUCCESS(rc))
1769 {
1770 for (uint32_t i = 0; i < pVGAState->cMonitors; ++i)
1771 {
1772 rc = SSMR3PutU32 (pSSM, VBOXVBVASAVEDSTATE_VHWAUNAVAILABLE_MAGIC);
1773 AssertRCReturn(rc, rc);
1774 }
1775 }
1776
1777 /* no pending commands */
1778 SSMR3PutU32(pSSM, 0);
1779#endif
1780 return rc;
1781}
1782
1783int vboxVBVALoadStateExec (PPDMDEVINS pDevIns, PSSMHANDLE pSSM, uint32_t u32Version)
1784{
1785 if (u32Version < VGA_SAVEDSTATE_VERSION_HGSMI)
1786 {
1787 /* Nothing was saved. */
1788 return VINF_SUCCESS;
1789 }
1790
1791 PVGASTATE pVGAState = PDMINS_2_DATA(pDevIns, PVGASTATE);
1792 PHGSMIINSTANCE pIns = pVGAState->pHGSMI;
1793 int rc = HGSMIHostLoadStateExec (pIns, pSSM, u32Version);
1794 if (RT_SUCCESS(rc))
1795 {
1796 /* Load VBVACONTEXT. */
1797 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext (pIns);
1798
1799 if (!pCtx)
1800 {
1801 /* This should not happen. */
1802 AssertFailed();
1803 rc = VERR_INVALID_PARAMETER;
1804 }
1805 else
1806 {
1807 uint32_t cViews = 0;
1808 rc = SSMR3GetU32 (pSSM, &cViews);
1809 AssertRCReturn(rc, rc);
1810
1811 uint32_t iView;
1812 for (iView = 0; iView < cViews; iView++)
1813 {
1814 VBVAVIEW *pView = &pCtx->aViews[iView];
1815
1816 rc = SSMR3GetU32 (pSSM, &pView->view.u32ViewIndex);
1817 AssertRCReturn(rc, rc);
1818 rc = SSMR3GetU32 (pSSM, &pView->view.u32ViewOffset);
1819 AssertRCReturn(rc, rc);
1820 rc = SSMR3GetU32 (pSSM, &pView->view.u32ViewSize);
1821 AssertRCReturn(rc, rc);
1822 rc = SSMR3GetU32 (pSSM, &pView->view.u32MaxScreenSize);
1823 AssertRCReturn(rc, rc);
1824
1825 rc = SSMR3GetU32 (pSSM, &pView->screen.u32ViewIndex);
1826 AssertRCReturn(rc, rc);
1827 rc = SSMR3GetS32 (pSSM, &pView->screen.i32OriginX);
1828 AssertRCReturn(rc, rc);
1829 rc = SSMR3GetS32 (pSSM, &pView->screen.i32OriginY);
1830 AssertRCReturn(rc, rc);
1831 rc = SSMR3GetU32 (pSSM, &pView->screen.u32StartOffset);
1832 AssertRCReturn(rc, rc);
1833 rc = SSMR3GetU32 (pSSM, &pView->screen.u32LineSize);
1834 AssertRCReturn(rc, rc);
1835 rc = SSMR3GetU32 (pSSM, &pView->screen.u32Width);
1836 AssertRCReturn(rc, rc);
1837 rc = SSMR3GetU32 (pSSM, &pView->screen.u32Height);
1838 AssertRCReturn(rc, rc);
1839 rc = SSMR3GetU16 (pSSM, &pView->screen.u16BitsPerPixel);
1840 AssertRCReturn(rc, rc);
1841 rc = SSMR3GetU16 (pSSM, &pView->screen.u16Flags);
1842 AssertRCReturn(rc, rc);
1843
1844 rc = SSMR3GetU32 (pSSM, &pView->vbva.u32VBVAOffset);
1845 AssertRCReturn(rc, rc);
1846
1847 rc = SSMR3GetU32 (pSSM, &pView->vbva.partialRecord.cb);
1848 AssertRCReturn(rc, rc);
1849
1850 if (pView->vbva.partialRecord.cb == 0)
1851 {
1852 pView->vbva.partialRecord.pu8 = NULL;
1853 }
1854 else
1855 {
1856 Assert(pView->vbva.partialRecord.pu8 == NULL); /* Should be it. */
1857
1858 uint8_t *pu8 = (uint8_t *)RTMemAlloc(pView->vbva.partialRecord.cb);
1859
1860 if (!pu8)
1861 {
1862 return VERR_NO_MEMORY;
1863 }
1864
1865 pView->vbva.partialRecord.pu8 = pu8;
1866
1867 rc = SSMR3GetMem (pSSM, pView->vbva.partialRecord.pu8, pView->vbva.partialRecord.cb);
1868 AssertRCReturn(rc, rc);
1869 }
1870
1871 if (pView->vbva.u32VBVAOffset == HGSMIOFFSET_VOID)
1872 {
1873 pView->vbva.guest.pVBVA = NULL;
1874 }
1875 else
1876 {
1877 pView->vbva.guest.pVBVA = (VBVABUFFER *)HGSMIOffsetToPointerHost(pIns, pView->vbva.u32VBVAOffset);
1878 }
1879 }
1880
1881 if (u32Version > VGA_SAVEDSTATE_VERSION_WITH_CONFIG)
1882 {
1883 /* Read mouse pointer shape information. */
1884 rc = SSMR3GetBool (pSSM, &pCtx->mouseShapeInfo.fSet);
1885 AssertRCReturn(rc, rc);
1886 rc = SSMR3GetBool (pSSM, &pCtx->mouseShapeInfo.fVisible);
1887 AssertRCReturn(rc, rc);
1888 rc = SSMR3GetBool (pSSM, &pCtx->mouseShapeInfo.fAlpha);
1889 AssertRCReturn(rc, rc);
1890 rc = SSMR3GetU32 (pSSM, &pCtx->mouseShapeInfo.u32HotX);
1891 AssertRCReturn(rc, rc);
1892 rc = SSMR3GetU32 (pSSM, &pCtx->mouseShapeInfo.u32HotY);
1893 AssertRCReturn(rc, rc);
1894 rc = SSMR3GetU32 (pSSM, &pCtx->mouseShapeInfo.u32Width);
1895 AssertRCReturn(rc, rc);
1896 rc = SSMR3GetU32 (pSSM, &pCtx->mouseShapeInfo.u32Height);
1897 AssertRCReturn(rc, rc);
1898 rc = SSMR3GetU32 (pSSM, &pCtx->mouseShapeInfo.cbShape);
1899 AssertRCReturn(rc, rc);
1900 if (pCtx->mouseShapeInfo.cbShape)
1901 {
1902 pCtx->mouseShapeInfo.pu8Shape = (uint8_t *)RTMemAlloc(pCtx->mouseShapeInfo.cbShape);
1903 if (pCtx->mouseShapeInfo.pu8Shape == NULL)
1904 {
1905 return VERR_NO_MEMORY;
1906 }
1907 pCtx->mouseShapeInfo.cbAllocated = pCtx->mouseShapeInfo.cbShape;
1908 rc = SSMR3GetMem (pSSM, pCtx->mouseShapeInfo.pu8Shape, pCtx->mouseShapeInfo.cbShape);
1909 AssertRCReturn(rc, rc);
1910 }
1911 else
1912 {
1913 pCtx->mouseShapeInfo.pu8Shape = NULL;
1914 }
1915
1916 /* Size of some additional data. For future extensions. */
1917 uint32_t cbExtra = 0;
1918 rc = SSMR3GetU32 (pSSM, &cbExtra);
1919 AssertRCReturn(rc, rc);
1920#ifdef VBOX_WITH_WDDM
1921 if (cbExtra >= 4)
1922 {
1923 rc = SSMR3GetU32 (pSSM, &pVGAState->fGuestCaps);
1924 AssertRCReturn(rc, rc);
1925 pVGAState->pDrv->pfnVBVAGuestCapabilityUpdate(pVGAState->pDrv, pVGAState->fGuestCaps);
1926 cbExtra -= 4;
1927 }
1928#endif
1929 if (cbExtra > 0)
1930 {
1931 rc = SSMR3Skip(pSSM, cbExtra);
1932 AssertRCReturn(rc, rc);
1933 }
1934
1935 if (u32Version >= VGA_SAVEDSTATE_VERSION_MODE_HINTS)
1936 {
1937 uint32_t cModeHints, cbModeHints;
1938 rc = SSMR3GetU32 (pSSM, &cModeHints);
1939 AssertRCReturn(rc, rc);
1940 rc = SSMR3GetU32 (pSSM, &cbModeHints);
1941 AssertRCReturn(rc, rc);
1942 memset(&pCtx->aModeHints, ~0, sizeof(pCtx->aModeHints));
1943 unsigned iHint;
1944 for (iHint = 0; iHint < cModeHints; ++iHint)
1945 {
1946 if ( cbModeHints <= sizeof(VBVAMODEHINT)
1947 && iHint < RT_ELEMENTS(pCtx->aModeHints))
1948 rc = SSMR3GetMem(pSSM, &pCtx->aModeHints[iHint],
1949 cbModeHints);
1950 else
1951 rc = SSMR3Skip(pSSM, cbModeHints);
1952 AssertRCReturn(rc, rc);
1953 }
1954 }
1955 }
1956
1957 pCtx->cViews = iView;
1958 LogFlowFunc(("%d views loaded\n", pCtx->cViews));
1959
1960 if (u32Version > VGA_SAVEDSTATE_VERSION_WDDM)
1961 {
1962 bool fLoadCommands;
1963
1964 if (u32Version < VGA_SAVEDSTATE_VERSION_FIXED_PENDVHWA)
1965 {
1966 const char *pcszOsArch = SSMR3HandleHostOSAndArch(pSSM);
1967 Assert(pcszOsArch);
1968 fLoadCommands = !pcszOsArch || RTStrNCmp(pcszOsArch, RT_STR_TUPLE("solaris"));
1969 }
1970 else
1971 fLoadCommands = true;
1972
1973#ifdef VBOX_WITH_VIDEOHWACCEL
1974 uint32_t cbCmd = sizeof (VBOXVHWACMD_HH_SAVESTATE_LOADPERFORM); /* maximum cmd size */
1975 VBOXVHWACMD *pCmd = vbvaVHWAHHCommandCreate(pVGAState, VBOXVHWACMD_TYPE_HH_SAVESTATE_LOADPERFORM, 0, cbCmd);
1976 Assert(pCmd);
1977 if(pCmd)
1978 {
1979 VBOXVBVASAVEDSTATECBDATA VhwaData = {0};
1980 VhwaData.pSSM = pSSM;
1981 VBOXVHWACMD_HH_SAVESTATE_LOADPERFORM *pLoad = VBOXVHWACMD_BODY(pCmd, VBOXVHWACMD_HH_SAVESTATE_LOADPERFORM);
1982 pLoad->pSSM = pSSM;
1983 vbvaVHWAHHPost (pVGAState, pCmd, vboxVBVALoadStatePerformPreCb, vboxVBVALoadStatePerformPostCb, &VhwaData);
1984 rc = VhwaData.rc;
1985 vbvaVHWAHHCommandRelease(pCmd);
1986 AssertRCReturn(rc, rc);
1987
1988 if (fLoadCommands)
1989 {
1990 rc = vbvaVHWACommandLoadPending(pVGAState, pSSM, u32Version);
1991 AssertRCReturn(rc, rc);
1992 }
1993 }
1994 else
1995 {
1996 rc = VERR_OUT_OF_RESOURCES;
1997 }
1998#else
1999 uint32_t u32;
2000
2001 for (uint32_t i = 0; i < pVGAState->cMonitors; ++i)
2002 {
2003 rc = SSMR3GetU32(pSSM, &u32);
2004 AssertRCReturn(rc, rc);
2005
2006 if (u32 != VBOXVBVASAVEDSTATE_VHWAUNAVAILABLE_MAGIC)
2007 {
2008 LogRel(("VBVA: 2D data while 2D is not supported\n"));
2009 return VERR_NOT_SUPPORTED;
2010 }
2011 }
2012
2013 if (fLoadCommands)
2014 {
2015 rc = SSMR3GetU32(pSSM, &u32);
2016 AssertRCReturn(rc, rc);
2017
2018 if (u32)
2019 {
2020 LogRel(("VBVA: 2D pending command while 2D is not supported\n"));
2021 return VERR_NOT_SUPPORTED;
2022 }
2023 }
2024#endif
2025 }
2026
2027#ifdef DEBUG_sunlover
2028 dumpctx(pCtx);
2029#endif
2030 }
2031 }
2032
2033 return rc;
2034}
2035
2036int vboxVBVALoadStateDone (PPDMDEVINS pDevIns, PSSMHANDLE pSSM)
2037{
2038 PVGASTATE pVGAState = PDMINS_2_DATA(pDevIns, PVGASTATE);
2039 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext (pVGAState->pHGSMI);
2040
2041 if (pCtx)
2042 {
2043 uint32_t iView;
2044 for (iView = 0; iView < pCtx->cViews; iView++)
2045 {
2046 VBVAVIEW *pView = &pCtx->aViews[iView];
2047
2048 if (pView->vbva.guest.pVBVA)
2049 {
2050#ifdef VBOX_WITH_CRHGSMI
2051 Assert(!vboxCmdVBVAIsEnabled(pVGAState));
2052#endif
2053 int rc = vbvaEnable(iView, pVGAState, pCtx, pView->vbva.guest.pVBVA, pView->vbva.u32VBVAOffset, true /* fRestored */);
2054 if (RT_SUCCESS(rc))
2055 {
2056 vbvaResize(pVGAState, pView, &pView->screen);
2057 }
2058 else
2059 {
2060 LogRel(("VBVA: can not restore: %Rrc\n", rc));
2061 }
2062 }
2063 }
2064
2065 if (pCtx->mouseShapeInfo.fSet)
2066 {
2067 vbvaUpdateMousePointerShape(pVGAState, &pCtx->mouseShapeInfo, true);
2068 }
2069 }
2070
2071 return VINF_SUCCESS;
2072}
2073
2074void VBVARaiseIrq (PVGASTATE pVGAState, uint32_t fFlags)
2075{
2076 PPDMDEVINS pDevIns = pVGAState->pDevInsR3;
2077
2078 PDMCritSectEnter(&pVGAState->CritSect, VERR_SEM_BUSY);
2079 HGSMISetHostGuestFlags(pVGAState->pHGSMI, HGSMIHOSTFLAGS_IRQ | fFlags);
2080 PDMCritSectLeave(&pVGAState->CritSect);
2081
2082 PDMDevHlpPCISetIrq(pDevIns, 0, PDM_IRQ_LEVEL_HIGH);
2083}
2084
2085static DECLCALLBACK(int) vbvaRaiseIrqEMT(PVGASTATE pVGAState, uint32_t fFlags)
2086{
2087 VBVARaiseIrq(pVGAState, fFlags);
2088 return VINF_SUCCESS;
2089}
2090
2091void VBVARaiseIrqNoWait(PVGASTATE pVGAState, uint32_t fFlags)
2092{
2093 /* we can not use PDMDevHlpPCISetIrqNoWait here, because we need to set IRG host flag and raise IRQ atomically,
2094 * otherwise there might be a situation, when:
2095 * 1. Flag is set
2096 * 2. guest issues an IRQ clean request, that cleans up the flag and the interrupt
2097 * 3. IRQ is set */
2098 VMR3ReqCallNoWait(PDMDevHlpGetVM(pVGAState->pDevInsR3), VMCPUID_ANY, (PFNRT)vbvaRaiseIrqEMT, 2, pVGAState, fFlags);
2099}
2100
2101static int vbvaHandleQueryConf32(PVGASTATE pVGAState, VBVACONF32 *pConf32)
2102{
2103 int rc = VINF_SUCCESS;
2104 PHGSMIINSTANCE pIns = pVGAState->pHGSMI;
2105 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext(pIns);
2106
2107 const uint32_t u32Index = pConf32->u32Index;
2108
2109 LogFlowFunc(("VBVA_QUERY_CONF32: u32Index %d, u32Value 0x%x\n",
2110 u32Index, pConf32->u32Value));
2111
2112 if (u32Index == VBOX_VBVA_CONF32_MONITOR_COUNT)
2113 {
2114 pConf32->u32Value = pCtx->cViews;
2115 }
2116 else if (u32Index == VBOX_VBVA_CONF32_HOST_HEAP_SIZE)
2117 {
2118 /* @todo a value calculated from the vram size */
2119 pConf32->u32Value = 64*_1K;
2120 }
2121 else if ( u32Index == VBOX_VBVA_CONF32_MODE_HINT_REPORTING
2122 || u32Index == VBOX_VBVA_CONF32_GUEST_CURSOR_REPORTING)
2123 {
2124 pConf32->u32Value = VINF_SUCCESS;
2125 }
2126 else if (u32Index == VBOX_VBVA_CONF32_CURSOR_CAPABILITIES)
2127 {
2128 pConf32->u32Value = pVGAState->fHostCursorCapabilities;
2129 }
2130 else if (u32Index == VBOX_VBVA_CONF32_SCREEN_FLAGS)
2131 {
2132 pConf32->u32Value = VBVA_SCREEN_F_ACTIVE | VBVA_SCREEN_F_DISABLED | VBVA_SCREEN_F_BLANK;
2133 }
2134 else if (u32Index == VBOX_VBVA_CONF32_MAX_RECORD_SIZE)
2135 {
2136 pConf32->u32Value = VBVA_MAX_RECORD_SIZE;
2137 }
2138 else
2139 {
2140 Log(("Unsupported VBVA_QUERY_CONF32 index %d!!!\n",
2141 u32Index));
2142 rc = VERR_INVALID_PARAMETER;
2143 }
2144
2145 return rc;
2146}
2147
2148static int vbvaHandleSetConf32(PVGASTATE pVGAState, VBVACONF32 *pConf32)
2149{
2150 NOREF(pVGAState);
2151
2152 int rc = VINF_SUCCESS;
2153 const VBVACONF32 parms = *pConf32;
2154
2155 LogFlowFunc(("VBVA_SET_CONF32: u32Index %d, u32Value 0x%x\n",
2156 parms.u32Index, parms.u32Value));
2157
2158 if (parms.u32Index == VBOX_VBVA_CONF32_MONITOR_COUNT)
2159 {
2160 /* do nothing. this is a const. */
2161 }
2162 else if (parms.u32Index == VBOX_VBVA_CONF32_HOST_HEAP_SIZE)
2163 {
2164 /* do nothing. this is a const. */
2165 }
2166 else
2167 {
2168 Log(("Unsupported VBVA_SET_CONF32 index %d!!!\n",
2169 parms.u32Index));
2170 rc = VERR_INVALID_PARAMETER;
2171 }
2172
2173 return rc;
2174}
2175
2176static int vbvaHandleInfoHeap(PVGASTATE pVGAState, const VBVAINFOHEAP *pInfoHeap)
2177{
2178 PHGSMIINSTANCE pIns = pVGAState->pHGSMI;
2179
2180 const VBVAINFOHEAP parms = *pInfoHeap;
2181 LogFlowFunc(("VBVA_INFO_HEAP: offset 0x%x, size 0x%x\n",
2182 parms.u32HeapOffset, parms.u32HeapSize));
2183
2184 return HGSMIHostHeapSetup(pIns, parms.u32HeapOffset, parms.u32HeapSize);
2185}
2186
2187int VBVAInfoView(PVGASTATE pVGAState, const VBVAINFOVIEW *pView)
2188{
2189 const VBVAINFOVIEW view = *pView;
2190
2191 LogFlowFunc(("VBVA_INFO_VIEW: u32ViewIndex %d, u32ViewOffset 0x%x, u32ViewSize 0x%x, u32MaxScreenSize 0x%x\n",
2192 view.u32ViewIndex, view.u32ViewOffset, view.u32ViewSize, view.u32MaxScreenSize));
2193
2194 PHGSMIINSTANCE pIns = pVGAState->pHGSMI;
2195 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext(pIns);
2196
2197 if ( view.u32ViewIndex < pCtx->cViews
2198 && view.u32ViewOffset <= pVGAState->vram_size
2199 && view.u32ViewSize <= pVGAState->vram_size
2200 && view.u32ViewOffset <= pVGAState->vram_size - view.u32ViewSize
2201 && view.u32MaxScreenSize <= view.u32ViewSize)
2202 {
2203 pCtx->aViews[view.u32ViewIndex].view = view;
2204 return VINF_SUCCESS;
2205 }
2206
2207 LogRelFlow(("VBVA: InfoView: invalid data! index %d(%d), offset 0x%x, size 0x%x, max 0x%x, vram size 0x%x\n",
2208 view.u32ViewIndex, pCtx->cViews, view.u32ViewOffset, view.u32ViewSize,
2209 view.u32MaxScreenSize, pVGAState->vram_size));
2210 return VERR_INVALID_PARAMETER;
2211}
2212
2213int VBVAInfoScreen(PVGASTATE pVGAState, const VBVAINFOSCREEN *pScreen)
2214{
2215 const VBVAINFOSCREEN screen = *pScreen;
2216
2217 LogRel(("VBVA: InfoScreen: [%d] @%d,%d %dx%d, line 0x%x, BPP %d, flags 0x%x\n",
2218 screen.u32ViewIndex, screen.i32OriginX, screen.i32OriginY,
2219 screen.u32Width, screen.u32Height,
2220 screen.u32LineSize, screen.u16BitsPerPixel, screen.u16Flags));
2221
2222 PHGSMIINSTANCE pIns = pVGAState->pHGSMI;
2223 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext(pIns);
2224
2225 /* Allow screen.u16BitsPerPixel == 0 because legacy guest code used it for screen blanking. */
2226 if ( screen.u32ViewIndex < pCtx->cViews
2227 && screen.u16BitsPerPixel <= 32
2228 && screen.u32Width <= UINT16_MAX
2229 && screen.u32Height <= UINT16_MAX
2230 && screen.u32LineSize <= UINT16_MAX * 4)
2231 {
2232 const VBVAINFOVIEW *pView = &pCtx->aViews[screen.u32ViewIndex].view;
2233 const uint32_t u32BytesPerPixel = (screen.u16BitsPerPixel + 7) / 8;
2234 if (screen.u32Width <= screen.u32LineSize / (u32BytesPerPixel? u32BytesPerPixel: 1))
2235 {
2236 const uint64_t u64ScreenSize = (uint64_t)screen.u32LineSize * screen.u32Height;
2237 if ( screen.u32StartOffset <= pView->u32ViewSize
2238 && u64ScreenSize <= pView->u32MaxScreenSize
2239 && screen.u32StartOffset <= pView->u32ViewSize - (uint32_t)u64ScreenSize)
2240 {
2241 vbvaResize(pVGAState, &pCtx->aViews[screen.u32ViewIndex], &screen);
2242 return VINF_SUCCESS;
2243 }
2244
2245 /** @todo why not use "%#RX" instead of "0x%RX"? */
2246 LogRelFlow(("VBVA: InfoScreen: invalid data! size 0x%RX64, max 0x%RX32\n",
2247 u64ScreenSize, pView->u32MaxScreenSize));
2248 }
2249 }
2250 else
2251 {
2252 LogRelFlow(("VBVA: InfoScreen: invalid data! index %RU32(%RU32)\n", screen.u32ViewIndex,
2253 pCtx->cViews));
2254 }
2255
2256 return VERR_INVALID_PARAMETER;
2257}
2258
2259int VBVAGetInfoViewAndScreen(PVGASTATE pVGAState, uint32_t u32ViewIndex, VBVAINFOVIEW *pView, VBVAINFOSCREEN *pScreen)
2260{
2261 if (u32ViewIndex >= pVGAState->cMonitors)
2262 return VERR_INVALID_PARAMETER;
2263
2264 PHGSMIINSTANCE pIns = pVGAState->pHGSMI;
2265 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext (pIns);
2266
2267 if (pView)
2268 *pView = pCtx->aViews[u32ViewIndex].view;
2269
2270 if (pScreen)
2271 *pScreen = pCtx->aViews[u32ViewIndex].screen;
2272
2273 return VINF_SUCCESS;
2274}
2275
2276static int vbvaHandleEnable(PVGASTATE pVGAState, const VBVAENABLE *pVbvaEnable, uint32_t u32ScreenId)
2277{
2278 int rc = VINF_SUCCESS;
2279 PHGSMIINSTANCE pIns = pVGAState->pHGSMI;
2280 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext(pIns);
2281
2282 if (u32ScreenId > pCtx->cViews)
2283 {
2284 return VERR_INVALID_PARAMETER;
2285 }
2286
2287 const VBVAENABLE parms = *pVbvaEnable;
2288
2289 LogFlowFunc(("VBVA_ENABLE[%d]: u32Flags 0x%x u32Offset 0x%x\n",
2290 u32ScreenId, parms.u32Flags, parms.u32Offset));
2291
2292 if ((parms.u32Flags & (VBVA_F_ENABLE | VBVA_F_DISABLE)) == VBVA_F_ENABLE)
2293 {
2294 uint32_t u32Offset = parms.u32Offset;
2295 if (u32Offset < pVGAState->vram_size)
2296 {
2297 /* Guest reported offset either absolute or relative to view. */
2298 if (parms.u32Flags & VBVA_F_ABSOFFSET)
2299 {
2300 /* Offset from VRAM start. */
2301 if ( pVGAState->vram_size < RT_UOFFSETOF(VBVABUFFER, au8Data)
2302 || u32Offset > pVGAState->vram_size - RT_UOFFSETOF(VBVABUFFER, au8Data))
2303 {
2304 rc = VERR_INVALID_PARAMETER;
2305 }
2306 }
2307 else
2308 {
2309 /* Offset from the view start. */
2310 const VBVAINFOVIEW *pView = &pCtx->aViews[u32ScreenId].view;
2311 if ( pVGAState->vram_size - u32Offset < pView->u32ViewOffset
2312 || pView->u32ViewSize < RT_UOFFSETOF(VBVABUFFER, au8Data)
2313 || u32Offset > pView->u32ViewSize - RT_UOFFSETOF(VBVABUFFER, au8Data))
2314 {
2315 rc = VERR_INVALID_PARAMETER;
2316 }
2317 else
2318 {
2319 u32Offset += pView->u32ViewOffset;
2320 }
2321 }
2322 }
2323 else
2324 {
2325 rc = VERR_INVALID_PARAMETER;
2326 }
2327
2328 if (RT_SUCCESS(rc))
2329 {
2330 VBVABUFFER *pVBVA = (VBVABUFFER *)HGSMIOffsetToPointerHost(pIns, u32Offset);
2331 if (pVBVA)
2332 {
2333 /* Process any pending orders and empty the VBVA ring buffer. */
2334 vbvaFlush(pVGAState, pCtx);
2335
2336 rc = vbvaEnable(u32ScreenId, pVGAState, pCtx, pVBVA, u32Offset, false /* fRestored */);
2337 }
2338 else
2339 {
2340 Log(("Invalid VBVABUFFER offset 0x%x!!!\n",
2341 parms.u32Offset));
2342 rc = VERR_INVALID_PARAMETER;
2343 }
2344 }
2345
2346 if (RT_FAILURE(rc))
2347 {
2348 LogRelMax(8, ("VBVA: can not enable: %Rrc\n", rc));
2349 }
2350 }
2351 else if ((parms.u32Flags & (VBVA_F_ENABLE | VBVA_F_DISABLE)) == VBVA_F_DISABLE)
2352 {
2353 rc = vbvaDisable(u32ScreenId, pVGAState, pCtx);
2354 }
2355 else
2356 {
2357 Log(("Invalid VBVA_ENABLE flags 0x%x!!!\n",
2358 parms.u32Flags));
2359 rc = VERR_INVALID_PARAMETER;
2360 }
2361
2362 return rc;
2363}
2364
2365static int vbvaHandleQueryModeHints(PVGASTATE pVGAState, const VBVAQUERYMODEHINTS *pQueryModeHints, HGSMISIZE cbBuffer)
2366{
2367 PHGSMIINSTANCE pIns = pVGAState->pHGSMI;
2368 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext(pIns);
2369
2370 const VBVAQUERYMODEHINTS parms = *pQueryModeHints;
2371
2372 LogRelFlowFunc(("VBVA: HandleQueryModeHints: cHintsQueried=%RU16, cbHintStructureGuest=%RU16\n",
2373 parms.cHintsQueried, parms.cbHintStructureGuest));
2374
2375 if (cbBuffer < sizeof(VBVAQUERYMODEHINTS)
2376 + (uint64_t)parms.cHintsQueried * parms.cbHintStructureGuest)
2377 {
2378 return VERR_INVALID_PARAMETER;
2379 }
2380
2381 uint8_t *pbHint = (uint8_t *)pQueryModeHints + sizeof(VBVAQUERYMODEHINTS);
2382 memset(pbHint, ~0, cbBuffer - sizeof(VBVAQUERYMODEHINTS));
2383
2384 unsigned iHint;
2385 for (iHint = 0; iHint < parms.cHintsQueried
2386 && iHint < VBOX_VIDEO_MAX_SCREENS; ++iHint)
2387 {
2388 memcpy(pbHint, &pCtx->aModeHints[iHint],
2389 RT_MIN(parms.cbHintStructureGuest, sizeof(VBVAMODEHINT)));
2390 pbHint += parms.cbHintStructureGuest;
2391 Assert(pbHint - (uint8_t *)pQueryModeHints <= cbBuffer);
2392 }
2393
2394 return VINF_SUCCESS;
2395}
2396
2397/*
2398 *
2399 * New VBVA uses a new interface id: #define VBE_DISPI_ID_VBOX_VIDEO 0xBE01
2400 *
2401 * VBVA uses two 32 bits IO ports to write VRAM offsets of shared memory blocks for commands.
2402 * Read Write
2403 * Host port 0x3b0 to process completed
2404 * Guest port 0x3d0 control value? to process
2405 *
2406 */
2407
2408static DECLCALLBACK(void) vbvaNotifyGuest (void *pvCallback)
2409{
2410#if defined(VBOX_WITH_HGSMI) && (defined(VBOX_WITH_VIDEOHWACCEL) || defined(VBOX_WITH_VDMA) || defined(VBOX_WITH_WDDM))
2411 PVGASTATE pVGAState = (PVGASTATE)pvCallback;
2412 VBVARaiseIrqNoWait (pVGAState, 0);
2413#else
2414 NOREF(pvCallback);
2415 /* Do nothing. Later the VMMDev/VGA IRQ can be used for the notification. */
2416#endif
2417}
2418
2419/** The guest submitted a command buffer. Verify the buffer size and invoke corresponding handler.
2420 *
2421 * @return VBox status.
2422 * @param pvHandler The VBVA channel context.
2423 * @param u16ChannelInfo Command code.
2424 * @param pvBuffer HGSMI buffer with command data.
2425 * @param cbBuffer Size of command data.
2426 */
2427static DECLCALLBACK(int) vbvaChannelHandler(void *pvHandler, uint16_t u16ChannelInfo, void *pvBuffer, HGSMISIZE cbBuffer)
2428{
2429 int rc = VINF_SUCCESS;
2430
2431 LogFlowFunc(("pvHandler %p, u16ChannelInfo %d, pvBuffer %p, cbBuffer %u\n",
2432 pvHandler, u16ChannelInfo, pvBuffer, cbBuffer));
2433
2434 PVGASTATE pVGAState = (PVGASTATE)pvHandler;
2435 PHGSMIINSTANCE pIns = pVGAState->pHGSMI;
2436 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext(pIns);
2437
2438 switch (u16ChannelInfo)
2439 {
2440#ifdef VBOX_WITH_CRHGSMI
2441 case VBVA_CMDVBVA_SUBMIT:
2442 {
2443 rc = vboxCmdVBVACmdSubmit(pVGAState);
2444 } break;
2445
2446 case VBVA_CMDVBVA_FLUSH:
2447 {
2448 rc = vboxCmdVBVACmdFlush(pVGAState);
2449 } break;
2450
2451 case VBVA_CMDVBVA_CTL:
2452 {
2453 if (cbBuffer < VBoxSHGSMIBufferHeaderSize() + sizeof(VBOXCMDVBVA_CTL))
2454 {
2455 rc = VERR_INVALID_PARAMETER;
2456 break;
2457 }
2458
2459 VBOXCMDVBVA_CTL *pCtl = (VBOXCMDVBVA_CTL*)VBoxSHGSMIBufferData((PVBOXSHGSMIHEADER)pvBuffer);
2460 rc = vboxCmdVBVACmdCtl(pVGAState, pCtl, cbBuffer - VBoxSHGSMIBufferHeaderSize());
2461 } break;
2462#endif /* VBOX_WITH_CRHGSMI */
2463
2464#ifdef VBOX_WITH_VDMA
2465 case VBVA_VDMA_CMD:
2466 {
2467 if (cbBuffer < VBoxSHGSMIBufferHeaderSize() + sizeof(VBOXVDMACBUF_DR))
2468 {
2469 rc = VERR_INVALID_PARAMETER;
2470 break;
2471 }
2472
2473 PVBOXVDMACBUF_DR pCmd = (PVBOXVDMACBUF_DR)VBoxSHGSMIBufferData((PVBOXSHGSMIHEADER)pvBuffer);
2474 vboxVDMACommand(pVGAState->pVdma, pCmd, cbBuffer - VBoxSHGSMIBufferHeaderSize());
2475 } break;
2476
2477 case VBVA_VDMA_CTL:
2478 {
2479 if (cbBuffer < VBoxSHGSMIBufferHeaderSize() + sizeof(VBOXVDMA_CTL))
2480 {
2481 rc = VERR_INVALID_PARAMETER;
2482 break;
2483 }
2484
2485 PVBOXVDMA_CTL pCmd = (PVBOXVDMA_CTL)VBoxSHGSMIBufferData((PVBOXSHGSMIHEADER)pvBuffer);
2486 vboxVDMAControl(pVGAState->pVdma, pCmd, cbBuffer - VBoxSHGSMIBufferHeaderSize());
2487 } break;
2488#endif /* VBOX_WITH_VDMA */
2489
2490 case VBVA_QUERY_CONF32:
2491 {
2492 if (cbBuffer < sizeof(VBVACONF32))
2493 {
2494 rc = VERR_INVALID_PARAMETER;
2495 break;
2496 }
2497
2498 VBVACONF32 *pConf32 = (VBVACONF32 *)pvBuffer;
2499 rc = vbvaHandleQueryConf32(pVGAState, pConf32);
2500 } break;
2501
2502 case VBVA_SET_CONF32:
2503 {
2504 if (cbBuffer < sizeof(VBVACONF32))
2505 {
2506 rc = VERR_INVALID_PARAMETER;
2507 break;
2508 }
2509
2510 VBVACONF32 *pConf32 = (VBVACONF32 *)pvBuffer;
2511 rc = vbvaHandleSetConf32(pVGAState, pConf32);
2512 } break;
2513
2514 case VBVA_INFO_VIEW:
2515 {
2516#ifdef VBOX_WITH_CRHGSMI
2517 if (vboxCmdVBVAIsEnabled(pVGAState))
2518 {
2519 AssertMsgFailed(("VBVA_INFO_VIEW is not acceptible for CmdVbva\n"));
2520 rc = VERR_INVALID_PARAMETER;
2521 break;
2522 }
2523#endif /* VBOX_WITH_CRHGSMI */
2524
2525 /* Expect at least one VBVAINFOVIEW structure. */
2526 if (cbBuffer < sizeof(VBVAINFOVIEW))
2527 {
2528 rc = VERR_INVALID_PARAMETER;
2529 break;
2530 }
2531
2532 /* Guest submits an array of VBVAINFOVIEW structures. */
2533 const VBVAINFOVIEW *pView = (VBVAINFOVIEW *)pvBuffer;
2534 for (;
2535 cbBuffer >= sizeof(VBVAINFOVIEW);
2536 ++pView, cbBuffer -= sizeof(VBVAINFOVIEW))
2537 {
2538 rc = VBVAInfoView(pVGAState, pView);
2539 if (RT_FAILURE(rc))
2540 break;
2541 }
2542 } break;
2543
2544 case VBVA_INFO_HEAP:
2545 {
2546 if (cbBuffer < sizeof(VBVAINFOHEAP))
2547 {
2548 rc = VERR_INVALID_PARAMETER;
2549 break;
2550 }
2551
2552 const VBVAINFOHEAP *pInfoHeap = (VBVAINFOHEAP *)pvBuffer;
2553 rc = vbvaHandleInfoHeap(pVGAState, pInfoHeap);
2554 } break;
2555
2556 case VBVA_FLUSH:
2557 {
2558 if (cbBuffer < sizeof(VBVAFLUSH))
2559 {
2560 rc = VERR_INVALID_PARAMETER;
2561 break;
2562 }
2563
2564 // const VBVAFLUSH *pVbvaFlush = (VBVAFLUSH *)pvBuffer;
2565 rc = vbvaFlush(pVGAState, pCtx);
2566 } break;
2567
2568 case VBVA_INFO_SCREEN:
2569 {
2570#ifdef VBOX_WITH_CRHGSMI
2571 if (vboxCmdVBVAIsEnabled(pVGAState))
2572 {
2573 AssertMsgFailed(("VBVA_INFO_SCREEN is not acceptible for CmdVbva\n"));
2574 rc = VERR_INVALID_PARAMETER;
2575 break;
2576 }
2577#endif /* VBOX_WITH_CRHGSMI */
2578
2579 if (cbBuffer < sizeof(VBVAINFOSCREEN))
2580 {
2581 rc = VERR_INVALID_PARAMETER;
2582 break;
2583 }
2584
2585 const VBVAINFOSCREEN *pInfoScreen = (VBVAINFOSCREEN *)pvBuffer;
2586 rc = VBVAInfoScreen(pVGAState, pInfoScreen);
2587 } break;
2588
2589 case VBVA_ENABLE:
2590 {
2591#ifdef VBOX_WITH_CRHGSMI
2592 if (vboxCmdVBVAIsEnabled(pVGAState))
2593 {
2594 AssertMsgFailed(("VBVA_ENABLE is not acceptible for CmdVbva\n"));
2595 rc = VERR_INVALID_PARAMETER;
2596 break;
2597 }
2598#endif /* VBOX_WITH_CRHGSMI */
2599
2600 if (cbBuffer < sizeof(VBVAENABLE))
2601 {
2602 rc = VERR_INVALID_PARAMETER;
2603 break;
2604 }
2605
2606 VBVAENABLE *pVbvaEnable = (VBVAENABLE *)pvBuffer;
2607
2608 uint32_t u32ScreenId;
2609 const uint32_t u32Flags = pVbvaEnable->u32Flags;
2610 if (u32Flags & VBVA_F_EXTENDED)
2611 {
2612 if (cbBuffer < sizeof(VBVAENABLE_EX))
2613 {
2614 rc = VERR_INVALID_PARAMETER;
2615 break;
2616 }
2617
2618 const VBVAENABLE_EX *pEnableEx = (VBVAENABLE_EX *)pvBuffer;
2619 u32ScreenId = pEnableEx->u32ScreenId;
2620 }
2621 else
2622 {
2623 u32ScreenId = vbvaViewFromBufferPtr(pIns, pCtx, pvBuffer);
2624 }
2625
2626 rc = vbvaHandleEnable(pVGAState, pVbvaEnable, u32ScreenId);
2627
2628 pVbvaEnable->i32Result = rc;
2629 } break;
2630
2631 case VBVA_MOUSE_POINTER_SHAPE:
2632 {
2633 if (cbBuffer < sizeof(VBVAMOUSEPOINTERSHAPE))
2634 {
2635 rc = VERR_INVALID_PARAMETER;
2636 break;
2637 }
2638
2639 VBVAMOUSEPOINTERSHAPE *pShape = (VBVAMOUSEPOINTERSHAPE *)pvBuffer;
2640 rc = vbvaMousePointerShape(pVGAState, pCtx, pShape, cbBuffer);
2641
2642 pShape->i32Result = rc;
2643 } break;
2644
2645
2646#ifdef VBOX_WITH_VIDEOHWACCEL
2647 case VBVA_VHWA_CMD:
2648 {
2649 if (cbBuffer < sizeof(VBOXVHWACMD))
2650 {
2651 rc = VERR_INVALID_PARAMETER;
2652 break;
2653 }
2654 vbvaVHWAHandleCommand(pVGAState, (PVBOXVHWACMD)pvBuffer);
2655 } break;
2656#endif /* VBOX_WITH_VIDEOHWACCEL */
2657
2658#ifdef VBOX_WITH_WDDM
2659 case VBVA_INFO_CAPS:
2660 {
2661 if (cbBuffer < sizeof(VBVACAPS))
2662 {
2663 rc = VERR_INVALID_PARAMETER;
2664 break;
2665 }
2666
2667 VBVACAPS *pCaps = (VBVACAPS*)pvBuffer;
2668 pVGAState->fGuestCaps = pCaps->fCaps;
2669 pVGAState->pDrv->pfnVBVAGuestCapabilityUpdate(pVGAState->pDrv,
2670 pVGAState->fGuestCaps);
2671 pCaps->rc = VINF_SUCCESS;
2672 } break;
2673#endif /* VBOX_WITH_WDDM */
2674
2675 case VBVA_SCANLINE_CFG:
2676 {
2677 if (cbBuffer < sizeof(VBVASCANLINECFG))
2678 {
2679 rc = VERR_INVALID_PARAMETER;
2680 break;
2681 }
2682
2683 VBVASCANLINECFG *pCfg = (VBVASCANLINECFG*)pvBuffer;
2684 pVGAState->fScanLineCfg = pCfg->fFlags;
2685 pCfg->rc = VINF_SUCCESS;
2686 } break;
2687
2688 case VBVA_QUERY_MODE_HINTS:
2689 {
2690 if (cbBuffer < sizeof(VBVAQUERYMODEHINTS))
2691 {
2692 rc = VERR_INVALID_PARAMETER;
2693 break;
2694 }
2695
2696 VBVAQUERYMODEHINTS *pQueryModeHints = (VBVAQUERYMODEHINTS*)pvBuffer;
2697 rc = vbvaHandleQueryModeHints(pVGAState, pQueryModeHints, cbBuffer);
2698 pQueryModeHints->rc = rc;
2699 } break;
2700
2701 case VBVA_REPORT_INPUT_MAPPING:
2702 {
2703 if (cbBuffer < sizeof(VBVAREPORTINPUTMAPPING))
2704 {
2705 rc = VERR_INVALID_PARAMETER;
2706 break;
2707 }
2708
2709 const VBVAREPORTINPUTMAPPING inputMapping = *(VBVAREPORTINPUTMAPPING *)pvBuffer;
2710 LogRelFlowFunc(("VBVA: ChannelHandler: VBVA_REPORT_INPUT_MAPPING: x=%RI32, y=%RI32, cx=%RU32, cy=%RU32\n",
2711 inputMapping.x, inputMapping.y, inputMapping.cx, inputMapping.cy));
2712 pVGAState->pDrv->pfnVBVAInputMappingUpdate(pVGAState->pDrv,
2713 inputMapping.x, inputMapping.y,
2714 inputMapping.cx, inputMapping.cy);
2715 } break;
2716
2717 case VBVA_CURSOR_POSITION:
2718 {
2719 if (cbBuffer < sizeof(VBVACURSORPOSITION))
2720 {
2721 rc = VERR_INVALID_PARAMETER;
2722 break;
2723 }
2724
2725 VBVACURSORPOSITION *pReport = (VBVACURSORPOSITION *)pvBuffer;
2726
2727 LogRelFlowFunc(("VBVA: ChannelHandler: VBVA_CURSOR_POSITION: fReportPosition=%RTbool, x=%RU32, y=%RU32\n",
2728 RT_BOOL(pReport->fReportPosition), pReport->x, pReport->y));
2729
2730 pReport->x = pCtx->xCursor;
2731 pReport->y = pCtx->yCursor;
2732 } break;
2733
2734 default:
2735 Log(("Unsupported VBVA guest command %d!!!\n",
2736 u16ChannelInfo));
2737 break;
2738 }
2739
2740 return rc;
2741}
2742
2743/* When VBVA is paused, then VGA device is allowed to work but
2744 * no HGSMI etc state is changed.
2745 */
2746void VBVAPause(PVGASTATE pVGAState, bool fPause)
2747{
2748 if (!pVGAState || !pVGAState->pHGSMI)
2749 {
2750 return;
2751 }
2752
2753 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext(pVGAState->pHGSMI);
2754
2755 if (pCtx)
2756 {
2757 pCtx->fPaused = fPause;
2758 }
2759}
2760
2761void VBVAReset (PVGASTATE pVGAState)
2762{
2763 if (!pVGAState || !pVGAState->pHGSMI)
2764 {
2765 return;
2766 }
2767
2768 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext (pVGAState->pHGSMI);
2769
2770#ifdef VBOX_WITH_VIDEOHWACCEL
2771 vbvaVHWAReset (pVGAState);
2772#endif
2773
2774 uint32_t HgFlags = HGSMIReset (pVGAState->pHGSMI);
2775 if(HgFlags & HGSMIHOSTFLAGS_IRQ)
2776 {
2777 /* this means the IRQ is LEVEL_HIGH, need to reset it */
2778 PDMDevHlpPCISetIrq(pVGAState->pDevInsR3, 0, PDM_IRQ_LEVEL_LOW);
2779 }
2780
2781 if (pCtx)
2782 {
2783 vbvaFlush (pVGAState, pCtx);
2784
2785 unsigned uScreenId;
2786
2787 for (uScreenId = 0; uScreenId < pCtx->cViews; uScreenId++)
2788 {
2789 vbvaDisable (uScreenId, pVGAState, pCtx);
2790 }
2791
2792 pCtx->mouseShapeInfo.fSet = false;
2793 RTMemFree(pCtx->mouseShapeInfo.pu8Shape);
2794 pCtx->mouseShapeInfo.pu8Shape = NULL;
2795 pCtx->mouseShapeInfo.cbAllocated = 0;
2796 pCtx->mouseShapeInfo.cbShape = 0;
2797 }
2798
2799}
2800
2801int VBVAUpdateDisplay (PVGASTATE pVGAState)
2802{
2803 int rc = VERR_NOT_SUPPORTED; /* Assuming that the VGA device will have to do updates. */
2804
2805 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext (pVGAState->pHGSMI);
2806
2807 if (pCtx)
2808 {
2809 if (!pCtx->fPaused)
2810 {
2811 rc = vbvaFlush (pVGAState, pCtx);
2812
2813 if (RT_SUCCESS (rc))
2814 {
2815 if (!pCtx->aViews[0].vbva.guest.pVBVA)
2816 {
2817 /* VBVA is not enabled for the first view, so VGA device must do updates. */
2818 rc = VERR_NOT_SUPPORTED;
2819 }
2820 }
2821 }
2822 }
2823
2824 return rc;
2825}
2826
2827static int vbvaSendModeHintWorker(PVGASTATE pThis, uint32_t cx, uint32_t cy,
2828 uint32_t cBPP, uint32_t iDisplay, uint32_t dx,
2829 uint32_t dy, uint32_t fEnabled,
2830 uint32_t fNotifyGuest)
2831{
2832 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext(pThis->pHGSMI);
2833 /** @note See Display::setVideoModeHint: "It is up to the guest to decide
2834 * whether the hint is valid. Therefore don't do any VRAM sanity checks
2835 * here! */
2836 if (iDisplay >= RT_MIN(pThis->cMonitors, RT_ELEMENTS(pCtx->aModeHints)))
2837 return VERR_OUT_OF_RANGE;
2838 pCtx->aModeHints[iDisplay].magic = VBVAMODEHINT_MAGIC;
2839 pCtx->aModeHints[iDisplay].cx = cx;
2840 pCtx->aModeHints[iDisplay].cy = cy;
2841 pCtx->aModeHints[iDisplay].cBPP = cBPP;
2842 pCtx->aModeHints[iDisplay].dx = dx;
2843 pCtx->aModeHints[iDisplay].dy = dy;
2844 pCtx->aModeHints[iDisplay].fEnabled = fEnabled;
2845 if (fNotifyGuest && pThis->fGuestCaps & VBVACAPS_IRQ && pThis->fGuestCaps & VBVACAPS_VIDEO_MODE_HINTS)
2846 VBVARaiseIrq(pThis, HGSMIHOSTFLAGS_HOTPLUG);
2847 return VINF_SUCCESS;
2848}
2849
2850/** Converts a display port interface pointer to a vga state pointer. */
2851#define IDISPLAYPORT_2_VGASTATE(pInterface) ( (PVGASTATE)((uintptr_t)pInterface - RT_OFFSETOF(VGASTATE, IPort)) )
2852
2853DECLCALLBACK(int) vbvaPortSendModeHint(PPDMIDISPLAYPORT pInterface, uint32_t cx,
2854 uint32_t cy, uint32_t cBPP,
2855 uint32_t iDisplay, uint32_t dx,
2856 uint32_t dy, uint32_t fEnabled,
2857 uint32_t fNotifyGuest)
2858{
2859 PVGASTATE pThis;
2860 int rc;
2861
2862 pThis = IDISPLAYPORT_2_VGASTATE(pInterface);
2863 rc = PDMCritSectEnter(&pThis->CritSect, VERR_SEM_BUSY);
2864 AssertRC(rc);
2865 rc = vbvaSendModeHintWorker(pThis, cx, cy, cBPP, iDisplay, dx, dy, fEnabled,
2866 fNotifyGuest);
2867 PDMCritSectLeave(&pThis->CritSect);
2868 return rc;
2869}
2870
2871DECLCALLBACK(void) vbvaPortReportHostCursorCapabilities(PPDMIDISPLAYPORT pInterface, uint32_t fCapabilitiesAdded,
2872 uint32_t fCapabilitiesRemoved)
2873{
2874 PVGASTATE pThis = IDISPLAYPORT_2_VGASTATE(pInterface);
2875 int rc = PDMCritSectEnter(&pThis->CritSect, VERR_SEM_BUSY);
2876 AssertRC(rc);
2877 pThis->fHostCursorCapabilities |= fCapabilitiesAdded;
2878 pThis->fHostCursorCapabilities &= ~fCapabilitiesRemoved;
2879 if (pThis->fGuestCaps & VBVACAPS_IRQ && pThis->fGuestCaps & VBVACAPS_DISABLE_CURSOR_INTEGRATION)
2880 VBVARaiseIrqNoWait(pThis, HGSMIHOSTFLAGS_CURSOR_CAPABILITIES);
2881 PDMCritSectLeave(&pThis->CritSect);
2882}
2883
2884DECLCALLBACK(void) vbvaPortReportHostCursorPosition
2885 (PPDMIDISPLAYPORT pInterface, uint32_t x, uint32_t y)
2886{
2887 PVGASTATE pThis = IDISPLAYPORT_2_VGASTATE(pInterface);
2888 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext(pThis->pHGSMI);
2889 int rc = PDMCritSectEnter(&pThis->CritSect, VERR_SEM_BUSY);
2890 AssertRC(rc);
2891 pCtx->xCursor = x;
2892 pCtx->yCursor = y;
2893 PDMCritSectLeave(&pThis->CritSect);
2894}
2895
2896int VBVAInit (PVGASTATE pVGAState)
2897{
2898 PPDMDEVINS pDevIns = pVGAState->pDevInsR3;
2899
2900 PVM pVM = PDMDevHlpGetVM(pDevIns);
2901
2902 int rc = HGSMICreate (&pVGAState->pHGSMI,
2903 pVM,
2904 "VBVA",
2905 0,
2906 pVGAState->vram_ptrR3,
2907 pVGAState->vram_size,
2908 vbvaNotifyGuest,
2909 pVGAState,
2910 sizeof (VBVACONTEXT));
2911
2912 if (RT_SUCCESS (rc))
2913 {
2914 rc = HGSMIHostChannelRegister (pVGAState->pHGSMI,
2915 HGSMI_CH_VBVA,
2916 vbvaChannelHandler,
2917 pVGAState);
2918 if (RT_SUCCESS (rc))
2919 {
2920 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext (pVGAState->pHGSMI);
2921 pCtx->cViews = pVGAState->cMonitors;
2922 pCtx->fPaused = true;
2923 memset(pCtx->aModeHints, ~0, sizeof(pCtx->aModeHints));
2924 pVGAState->fHostCursorCapabilities = 0;
2925 }
2926 }
2927
2928 return rc;
2929
2930}
2931
2932void VBVADestroy (PVGASTATE pVGAState)
2933{
2934 VBVACONTEXT *pCtx = (VBVACONTEXT *)HGSMIContext (pVGAState->pHGSMI);
2935
2936 if (pCtx)
2937 {
2938 pCtx->mouseShapeInfo.fSet = false;
2939 RTMemFree(pCtx->mouseShapeInfo.pu8Shape);
2940 pCtx->mouseShapeInfo.pu8Shape = NULL;
2941 pCtx->mouseShapeInfo.cbAllocated = 0;
2942 pCtx->mouseShapeInfo.cbShape = 0;
2943 }
2944
2945 HGSMIDestroy (pVGAState->pHGSMI);
2946 pVGAState->pHGSMI = NULL;
2947}
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