VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/Graphics/Video/mp/wddm/VBoxMPVbva.cpp@ 50947

Last change on this file since 50947 was 50947, checked in by vboxsync, 11 years ago

wddm: preemption fixes

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 42.8 KB
Line 
1/* $Id: VBoxMPVbva.cpp 50947 2014-04-01 17:31:27Z vboxsync $ */
2
3/** @file
4 * VBox WDDM Miniport driver
5 */
6
7/*
8 * Copyright (C) 2011-2012 Oracle Corporation
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18
19#include "VBoxMPWddm.h"
20#include "common/VBoxMPCommon.h"
21
22/*
23 * Public hardware buffer methods.
24 */
25int vboxVbvaEnable (PVBOXMP_DEVEXT pDevExt, VBOXVBVAINFO *pVbva)
26{
27 if (VBoxVBVAEnable(&pVbva->Vbva, &VBoxCommonFromDeviceExt(pDevExt)->guestCtx,
28 pVbva->Vbva.pVBVA, pVbva->srcId))
29 return VINF_SUCCESS;
30
31 WARN(("VBoxVBVAEnable failed!"));
32 return VERR_GENERAL_FAILURE;
33}
34
35int vboxVbvaDisable (PVBOXMP_DEVEXT pDevExt, VBOXVBVAINFO *pVbva)
36{
37 VBoxVBVADisable(&pVbva->Vbva, &VBoxCommonFromDeviceExt(pDevExt)->guestCtx, pVbva->srcId);
38 return VINF_SUCCESS;
39}
40
41int vboxVbvaCreate(PVBOXMP_DEVEXT pDevExt, VBOXVBVAINFO *pVbva, ULONG offBuffer, ULONG cbBuffer, D3DDDI_VIDEO_PRESENT_SOURCE_ID srcId)
42{
43 memset(pVbva, 0, sizeof(VBOXVBVAINFO));
44
45 KeInitializeSpinLock(&pVbva->Lock);
46
47 int rc = VBoxMPCmnMapAdapterMemory(VBoxCommonFromDeviceExt(pDevExt),
48 (void**)&pVbva->Vbva.pVBVA,
49 offBuffer,
50 cbBuffer);
51 if (RT_SUCCESS(rc))
52 {
53 Assert(pVbva->Vbva.pVBVA);
54 VBoxVBVASetupBufferContext(&pVbva->Vbva, offBuffer, cbBuffer);
55 pVbva->srcId = srcId;
56 }
57 else
58 {
59 WARN(("VBoxMPCmnMapAdapterMemory failed rc %d", rc));
60 }
61
62
63 return rc;
64}
65
66int vboxVbvaDestroy(PVBOXMP_DEVEXT pDevExt, VBOXVBVAINFO *pVbva)
67{
68 int rc = VINF_SUCCESS;
69 VBoxMPCmnUnmapAdapterMemory(VBoxCommonFromDeviceExt(pDevExt), (void**)&pVbva->Vbva.pVBVA);
70 memset(pVbva, 0, sizeof (VBOXVBVAINFO));
71 return rc;
72}
73
74int vboxVbvaReportDirtyRect (PVBOXMP_DEVEXT pDevExt, PVBOXWDDM_SOURCE pSrc, RECT *pRectOrig)
75{
76 VBVACMDHDR hdr;
77
78 RECT rect = *pRectOrig;
79
80// if (rect.left < 0) rect.left = 0;
81// if (rect.top < 0) rect.top = 0;
82// if (rect.right > (int)ppdev->cxScreen) rect.right = ppdev->cxScreen;
83// if (rect.bottom > (int)ppdev->cyScreen) rect.bottom = ppdev->cyScreen;
84
85 hdr.x = (int16_t)rect.left;
86 hdr.y = (int16_t)rect.top;
87 hdr.w = (uint16_t)(rect.right - rect.left);
88 hdr.h = (uint16_t)(rect.bottom - rect.top);
89
90 hdr.x += (int16_t)pSrc->VScreenPos.x;
91 hdr.y += (int16_t)pSrc->VScreenPos.y;
92
93 if (VBoxVBVAWrite(&pSrc->Vbva.Vbva, &VBoxCommonFromDeviceExt(pDevExt)->guestCtx, &hdr, sizeof(hdr)))
94 return VINF_SUCCESS;
95
96 WARN(("VBoxVBVAWrite failed"));
97 return VERR_GENERAL_FAILURE;
98}
99
100#ifdef VBOX_WITH_CROGL
101/* command vbva ring buffer */
102
103/* customized VBVA implementation */
104
105/* Forward declarations of internal functions. */
106static void vboxHwBufferPlaceDataAt(PVBVAEXBUFFERCONTEXT pCtx, const void *p,
107 uint32_t cb, uint32_t offset);
108static bool vboxHwBufferWrite(PVBVAEXBUFFERCONTEXT pCtx,
109 PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx,
110 const void *p, uint32_t cb);
111
112DECLINLINE(void) vboxVBVAExFlush(struct VBVAEXBUFFERCONTEXT *pCtx, PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx)
113{
114 pCtx->pfnFlush(pCtx, pHGSMICtx, pCtx->pvFlush);
115}
116
117static int vboxCmdVbvaSubmitHgsmi(PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx, HGSMIOFFSET offDr)
118{
119 VBoxVideoCmnPortWriteUlong(pHGSMICtx->port, offDr);
120 return VINF_SUCCESS;
121}
122#define vboxCmdVbvaSubmit vboxCmdVbvaSubmitHgsmi
123
124static VBOXCMDVBVA_CTL * vboxCmdVbvaCtlCreate(PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx, uint32_t cbCtl)
125{
126 Assert(cbCtl >= sizeof (VBOXCMDVBVA_CTL));
127 return (VBOXCMDVBVA_CTL*)VBoxSHGSMICommandAlloc(&pHGSMICtx->heapCtx, cbCtl, HGSMI_CH_VBVA, VBVA_CMDVBVA_CTL);
128}
129
130static void vboxCmdVbvaCtlFree(PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx, VBOXCMDVBVA_CTL * pCtl)
131{
132 VBoxSHGSMICommandFree(&pHGSMICtx->heapCtx, pCtl);
133}
134
135static int vboxCmdVbvaCtlSubmitSync(PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx, VBOXCMDVBVA_CTL * pCtl)
136{
137 const VBOXSHGSMIHEADER* pHdr = VBoxSHGSMICommandPrepSynch(&pHGSMICtx->heapCtx, pCtl);
138 if (!pHdr)
139 {
140 WARN(("VBoxSHGSMICommandPrepSynch returnd NULL"));
141 return VERR_INVALID_PARAMETER;
142 }
143
144 HGSMIOFFSET offCmd = VBoxSHGSMICommandOffset(&pHGSMICtx->heapCtx, pHdr);
145 if (offCmd == HGSMIOFFSET_VOID)
146 {
147 WARN(("VBoxSHGSMICommandOffset returnd NULL"));
148 VBoxSHGSMICommandCancelSynch(&pHGSMICtx->heapCtx, pHdr);
149 return VERR_INVALID_PARAMETER;
150 }
151
152 int rc = vboxCmdVbvaSubmit(pHGSMICtx, offCmd);
153 if (RT_SUCCESS(rc))
154 {
155 rc = VBoxSHGSMICommandDoneSynch(&pHGSMICtx->heapCtx, pHdr);
156 if (RT_SUCCESS(rc))
157 {
158 rc = pCtl->i32Result;
159 if (!RT_SUCCESS(rc))
160 WARN(("pCtl->i32Result %d", pCtl->i32Result));
161
162 return rc;
163 }
164 else
165 WARN(("VBoxSHGSMICommandDoneSynch returnd %d", rc));
166 }
167 else
168 WARN(("vboxCmdVbvaSubmit returnd %d", rc));
169
170 VBoxSHGSMICommandCancelSynch(&pHGSMICtx->heapCtx, pHdr);
171
172 return rc;
173}
174
175static int vboxCmdVbvaCtlSubmitAsync(PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx, VBOXCMDVBVA_CTL * pCtl, FNVBOXSHGSMICMDCOMPLETION pfnCompletion, void *pvCompletion)
176{
177 const VBOXSHGSMIHEADER* pHdr = VBoxSHGSMICommandPrepAsynch(&pHGSMICtx->heapCtx, pCtl, pfnCompletion, pvCompletion, VBOXSHGSMI_FLAG_GH_ASYNCH_IRQ);
178 HGSMIOFFSET offCmd = VBoxSHGSMICommandOffset(&pHGSMICtx->heapCtx, pHdr);
179 if (offCmd == HGSMIOFFSET_VOID)
180 {
181 WARN(("VBoxSHGSMICommandOffset returnd NULL"));
182 VBoxSHGSMICommandCancelAsynch(&pHGSMICtx->heapCtx, pHdr);
183 return VERR_INVALID_PARAMETER;
184 }
185
186 int rc = vboxCmdVbvaSubmit(pHGSMICtx, offCmd);
187 if (RT_SUCCESS(rc))
188 {
189 VBoxSHGSMICommandDoneAsynch(&pHGSMICtx->heapCtx, pHdr);
190 return rc;
191 }
192 else
193 WARN(("vboxCmdVbvaSubmit returnd %d", rc));
194
195 VBoxSHGSMICommandCancelAsynch(&pHGSMICtx->heapCtx, pHdr);
196
197 return rc;
198}
199
200static int vboxVBVAExCtlSubmitEnableDisable(PVBVAEXBUFFERCONTEXT pCtx, PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx, bool fEnable)
201{
202 VBOXCMDVBVA_CTL_ENABLE *pCtl = (VBOXCMDVBVA_CTL_ENABLE*)vboxCmdVbvaCtlCreate(pHGSMICtx, sizeof (*pCtl));
203 if (!pCtl)
204 {
205 WARN(("vboxCmdVbvaCtlCreate failed"));
206 return VERR_NO_MEMORY;
207 }
208
209 pCtl->Hdr.u32Type = VBOXCMDVBVACTL_TYPE_ENABLE;
210 pCtl->Hdr.i32Result = VERR_NOT_IMPLEMENTED;
211 memset(&pCtl->Enable, 0, sizeof (pCtl->Enable));
212 pCtl->Enable.u32Flags = fEnable? VBVA_F_ENABLE: VBVA_F_DISABLE;
213 pCtl->Enable.u32Offset = pCtx->offVRAMBuffer;
214 pCtl->Enable.i32Result = VERR_NOT_SUPPORTED;
215 pCtl->Enable.u32Flags |= VBVA_F_ABSOFFSET;
216
217 int rc = vboxCmdVbvaCtlSubmitSync(pHGSMICtx, &pCtl->Hdr);
218 if (RT_SUCCESS(rc))
219 {
220 rc = pCtl->Hdr.i32Result;
221 if (!RT_SUCCESS(rc))
222 WARN(("vboxCmdVbvaCtlSubmitSync Disable failed %d", rc));
223 }
224 else
225 WARN(("vboxCmdVbvaCtlSubmitSync returnd %d", rc));
226
227 vboxCmdVbvaCtlFree(pHGSMICtx, &pCtl->Hdr);
228
229 return rc;
230}
231
232/*
233 * Public hardware buffer methods.
234 */
235RTDECL(int) VBoxVBVAExEnable(PVBVAEXBUFFERCONTEXT pCtx,
236 PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx,
237 VBVABUFFER *pVBVA)
238{
239 int rc = VERR_GENERAL_FAILURE;
240
241 LogFlowFunc(("pVBVA %p\n", pVBVA));
242
243#if 0 /* All callers check this */
244 if (ppdev->bHGSMISupported)
245#endif
246 {
247 LogFunc(("pVBVA %p vbva off 0x%x\n", pVBVA, pCtx->offVRAMBuffer));
248
249 pVBVA->hostFlags.u32HostEvents = 0;
250 pVBVA->hostFlags.u32SupportedOrders = 0;
251 pVBVA->off32Data = 0;
252 pVBVA->off32Free = 0;
253 memset(pVBVA->aRecords, 0, sizeof (pVBVA->aRecords));
254 pVBVA->indexRecordFirst = 0;
255 pVBVA->indexRecordFree = 0;
256 pVBVA->cbPartialWriteThreshold = 256;
257 pVBVA->cbData = pCtx->cbBuffer - sizeof (VBVABUFFER) + sizeof (pVBVA->au8Data);
258
259 pCtx->fHwBufferOverflow = false;
260 pCtx->pRecord = NULL;
261 pCtx->pVBVA = pVBVA;
262
263 rc = vboxVBVAExCtlSubmitEnableDisable(pCtx, pHGSMICtx, true);
264 }
265
266 if (!RT_SUCCESS(rc))
267 {
268 WARN(("enable failed %d", rc));
269 VBoxVBVAExDisable(pCtx, pHGSMICtx);
270 }
271
272 return rc;
273}
274
275RTDECL(void) VBoxVBVAExDisable(PVBVAEXBUFFERCONTEXT pCtx,
276 PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx)
277{
278 LogFlowFunc(("\n"));
279
280 pCtx->fHwBufferOverflow = false;
281 pCtx->pRecord = NULL;
282 pCtx->pVBVA = NULL;
283
284 vboxVBVAExCtlSubmitEnableDisable(pCtx, pHGSMICtx, false);
285
286 return;
287}
288
289RTDECL(bool) VBoxVBVAExBufferBeginUpdate(PVBVAEXBUFFERCONTEXT pCtx,
290 PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx)
291{
292 bool bRc = false;
293
294 // LogFunc(("flags = 0x%08X\n", pCtx->pVBVA? pCtx->pVBVA->u32HostEvents: -1));
295
296 Assert(pCtx->pVBVA);
297 /* we do not use u32HostEvents & VBVA_F_MODE_ENABLED,
298 * VBVA stays enabled once ENABLE call succeeds, until it is disabled with DISABLED call */
299// if ( pCtx->pVBVA
300// && (pCtx->pVBVA->hostFlags.u32HostEvents & VBVA_F_MODE_ENABLED))
301 {
302 uint32_t indexRecordNext;
303
304 Assert(!pCtx->fHwBufferOverflow);
305 Assert(pCtx->pRecord == NULL);
306
307 indexRecordNext = (pCtx->pVBVA->indexRecordFree + 1) % VBVA_MAX_RECORDS;
308
309 if (indexRecordNext == pCtx->indexRecordFirstUncompleted)
310 {
311 /* All slots in the records queue are used. */
312 vboxVBVAExFlush (pCtx, pHGSMICtx);
313 }
314
315 if (indexRecordNext == pCtx->indexRecordFirstUncompleted)
316 {
317 /* Even after flush there is no place. Fail the request. */
318 LogFunc(("no space in the queue of records!!! first %d, last %d\n",
319 indexRecordNext, pCtx->pVBVA->indexRecordFree));
320 }
321 else
322 {
323 /* Initialize the record. */
324 VBVARECORD *pRecord = &pCtx->pVBVA->aRecords[pCtx->pVBVA->indexRecordFree];
325
326 pRecord->cbRecord = VBVA_F_RECORD_PARTIAL;
327
328 pCtx->pVBVA->indexRecordFree = indexRecordNext;
329
330 // LogFunc(("indexRecordNext = %d\n", indexRecordNext));
331
332 /* Remember which record we are using. */
333 pCtx->pRecord = pRecord;
334
335 bRc = true;
336 }
337 }
338
339 return bRc;
340}
341
342RTDECL(void) VBoxVBVAExBufferEndUpdate(PVBVAEXBUFFERCONTEXT pCtx)
343{
344 VBVARECORD *pRecord;
345
346 // LogFunc(("\n"));
347
348 Assert(pCtx->pVBVA);
349
350 pRecord = pCtx->pRecord;
351 Assert(pRecord && (pRecord->cbRecord & VBVA_F_RECORD_PARTIAL));
352
353 /* Mark the record completed. */
354 pRecord->cbRecord &= ~VBVA_F_RECORD_PARTIAL;
355
356 pCtx->fHwBufferOverflow = false;
357 pCtx->pRecord = NULL;
358
359 return;
360}
361
362DECLINLINE(bool) vboxVBVAExIsEntryInRange(uint32_t u32First, uint32_t u32Entry, uint32_t u32Free)
363{
364 return ( u32First != u32Free
365 && (
366 (u32First < u32Free && u32Entry >= u32First && u32Entry < u32Free)
367 || (u32First > u32Free && (u32Entry >= u32First || u32Entry < u32Free))
368 )
369 );
370}
371
372DECLINLINE(bool) vboxVBVAExIsEntryInRangeOrEmpty(uint32_t u32First, uint32_t u32Entry, uint32_t u32Free)
373{
374 return vboxVBVAExIsEntryInRange(u32First, u32Entry, u32Free)
375 || ( u32First == u32Entry
376 && u32Entry == u32Free);
377}
378#ifdef DEBUG
379
380DECLINLINE(void) vboxHwBufferVerifyCompleted(PVBVAEXBUFFERCONTEXT pCtx)
381{
382 VBVABUFFER *pVBVA = pCtx->pVBVA;
383 if (!vboxVBVAExIsEntryInRangeOrEmpty(pCtx->indexRecordFirstUncompleted, pVBVA->indexRecordFirst, pVBVA->indexRecordFree))
384 {
385 WARN(("invalid record set"));
386 }
387
388 if (!vboxVBVAExIsEntryInRangeOrEmpty(pCtx->off32DataUncompleted, pVBVA->off32Data, pVBVA->off32Free))
389 {
390 WARN(("invalid data set"));
391 }
392}
393#endif
394
395/*
396 * Private operations.
397 */
398static uint32_t vboxHwBufferAvail(PVBVAEXBUFFERCONTEXT pCtx, const VBVABUFFER *pVBVA)
399{
400 int32_t i32Diff = pCtx->off32DataUncompleted - pVBVA->off32Free;
401
402 return i32Diff > 0? i32Diff: pVBVA->cbData + i32Diff;
403}
404
405static uint32_t vboxHwBufferContiguousAvail(PVBVAEXBUFFERCONTEXT pCtx, const VBVABUFFER *pVBVA)
406{
407 int32_t i32Diff = pCtx->off32DataUncompleted - pVBVA->off32Free;
408
409 return i32Diff > 0 ? i32Diff: pVBVA->cbData - pVBVA->off32Free;
410}
411
412static void vboxHwBufferPlaceDataAt(PVBVAEXBUFFERCONTEXT pCtx, const void *p,
413 uint32_t cb, uint32_t offset)
414{
415 VBVABUFFER *pVBVA = pCtx->pVBVA;
416 uint32_t u32BytesTillBoundary = pVBVA->cbData - offset;
417 uint8_t *dst = &pVBVA->au8Data[offset];
418 int32_t i32Diff = cb - u32BytesTillBoundary;
419
420 if (i32Diff <= 0)
421 {
422 /* Chunk will not cross buffer boundary. */
423 memcpy (dst, p, cb);
424 }
425 else
426 {
427 /* Chunk crosses buffer boundary. */
428 memcpy (dst, p, u32BytesTillBoundary);
429 memcpy (&pVBVA->au8Data[0], (uint8_t *)p + u32BytesTillBoundary, i32Diff);
430 }
431
432 return;
433}
434
435static bool vboxHwBufferWrite(PVBVAEXBUFFERCONTEXT pCtx,
436 PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx,
437 const void *p, uint32_t cb)
438{
439 VBVARECORD *pRecord;
440 uint32_t cbHwBufferAvail;
441
442 uint32_t cbWritten = 0;
443
444 VBVABUFFER *pVBVA = pCtx->pVBVA;
445 Assert(pVBVA);
446
447 if (!pVBVA || pCtx->fHwBufferOverflow)
448 {
449 return false;
450 }
451
452 Assert(pVBVA->indexRecordFirst != pVBVA->indexRecordFree);
453 Assert(pCtx->indexRecordFirstUncompleted != pVBVA->indexRecordFree);
454
455 pRecord = pCtx->pRecord;
456 Assert(pRecord && (pRecord->cbRecord & VBVA_F_RECORD_PARTIAL));
457
458 // LogFunc(("%d\n", cb));
459
460 cbHwBufferAvail = vboxHwBufferAvail(pCtx, pVBVA);
461
462 while (cb > 0)
463 {
464 uint32_t cbChunk = cb;
465
466 // LogFunc(("pVBVA->off32Free %d, pRecord->cbRecord 0x%08X, cbHwBufferAvail %d, cb %d, cbWritten %d\n",
467 // pVBVA->off32Free, pRecord->cbRecord, cbHwBufferAvail, cb, cbWritten));
468
469 if (cbChunk >= cbHwBufferAvail)
470 {
471 LogFunc(("1) avail %d, chunk %d\n", cbHwBufferAvail, cbChunk));
472
473 vboxVBVAExFlush(pCtx, pHGSMICtx);
474
475 cbHwBufferAvail = vboxHwBufferAvail(pCtx, pVBVA);
476
477 if (cbChunk >= cbHwBufferAvail)
478 {
479 WARN(("no place for %d bytes. Only %d bytes available after flush. Going to partial writes.\n",
480 cb, cbHwBufferAvail));
481
482 if (cbHwBufferAvail <= pVBVA->cbPartialWriteThreshold)
483 {
484 WARN(("Buffer overflow!!!\n"));
485 pCtx->fHwBufferOverflow = true;
486 Assert(false);
487 return false;
488 }
489
490 cbChunk = cbHwBufferAvail - pVBVA->cbPartialWriteThreshold;
491 }
492 }
493
494 Assert(cbChunk <= cb);
495 Assert(cbChunk <= vboxHwBufferAvail(pCtx, pVBVA));
496
497 vboxHwBufferPlaceDataAt (pCtx, (uint8_t *)p + cbWritten, cbChunk, pVBVA->off32Free);
498
499 pVBVA->off32Free = (pVBVA->off32Free + cbChunk) % pVBVA->cbData;
500 pRecord->cbRecord += cbChunk;
501 cbHwBufferAvail -= cbChunk;
502
503 cb -= cbChunk;
504 cbWritten += cbChunk;
505 }
506
507 return true;
508}
509
510/*
511 * Public writer to the hardware buffer.
512 */
513RTDECL(uint32_t) VBoxVBVAExGetFreeTail(PVBVAEXBUFFERCONTEXT pCtx)
514{
515 VBVABUFFER *pVBVA = pCtx->pVBVA;
516 if (pVBVA->off32Data <= pVBVA->off32Free)
517 return pVBVA->cbData - pVBVA->off32Free;
518 return 0;
519}
520
521RTDECL(void*) VBoxVBVAExAllocContiguous(PVBVAEXBUFFERCONTEXT pCtx, PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx, uint32_t cb)
522{
523 VBVARECORD *pRecord;
524 uint32_t cbHwBufferContiguousAvail;
525 uint32_t offset;
526
527 VBVABUFFER *pVBVA = pCtx->pVBVA;
528 Assert(pVBVA);
529
530 if (!pVBVA || pCtx->fHwBufferOverflow)
531 {
532 return NULL;
533 }
534
535 Assert(pVBVA->indexRecordFirst != pVBVA->indexRecordFree);
536 Assert(pCtx->indexRecordFirstUncompleted != pVBVA->indexRecordFree);
537
538 pRecord = pCtx->pRecord;
539 Assert(pRecord && (pRecord->cbRecord & VBVA_F_RECORD_PARTIAL));
540
541 // LogFunc(("%d\n", cb));
542
543 if (pVBVA->cbData < cb)
544 {
545 WARN(("requested to allocate buffer of size %d bigger than the VBVA ring buffer size %d", cb, pVBVA->cbData));
546 return NULL;
547 }
548
549 cbHwBufferContiguousAvail = vboxHwBufferContiguousAvail(pCtx, pVBVA);
550
551 if (cbHwBufferContiguousAvail < cb)
552 {
553 if (cb < pVBVA->cbData - pVBVA->off32Free)
554 {
555 /* the entire contiguous part is smaller than the requested buffer */
556 return NULL;
557 }
558
559 vboxVBVAExFlush(pCtx, pHGSMICtx);
560
561 cbHwBufferContiguousAvail = vboxHwBufferContiguousAvail(pCtx, pVBVA);
562 if (cbHwBufferContiguousAvail < cb)
563 {
564 /* this is really bad - the host did not clean up buffer even after we requested it to flush */
565 WARN(("Host did not clean up the buffer!"));
566 return NULL;
567 }
568 }
569
570 offset = pVBVA->off32Free;
571
572 pVBVA->off32Free = (pVBVA->off32Free + cb) % pVBVA->cbData;
573 pRecord->cbRecord += cb;
574
575 return &pVBVA->au8Data[offset];
576}
577
578RTDECL(bool) VBoxVBVAExIsProcessing(PVBVAEXBUFFERCONTEXT pCtx)
579{
580 uint32_t u32HostEvents = pCtx->pVBVA->hostFlags.u32HostEvents;
581 return !!(u32HostEvents & VBVA_F_STATE_PROCESSING);
582}
583
584RTDECL(void) VBoxVBVAExCBufferCompleted(PVBVAEXBUFFERCONTEXT pCtx)
585{
586 VBVABUFFER *pVBVA = pCtx->pVBVA;
587 uint32_t cbBuffer = pVBVA->aRecords[pCtx->indexRecordFirstUncompleted].cbRecord;
588 pCtx->indexRecordFirstUncompleted = (pCtx->indexRecordFirstUncompleted + 1) % VBVA_MAX_RECORDS;
589 pCtx->off32DataUncompleted = (pCtx->off32DataUncompleted + cbBuffer) % pVBVA->cbData;
590}
591
592RTDECL(bool) VBoxVBVAExWrite(PVBVAEXBUFFERCONTEXT pCtx,
593 PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx,
594 const void *pv, uint32_t cb)
595{
596 return vboxHwBufferWrite(pCtx, pHGSMICtx, pv, cb);
597}
598
599RTDECL(bool) VBoxVBVAExOrderSupported(PVBVAEXBUFFERCONTEXT pCtx, unsigned code)
600{
601 VBVABUFFER *pVBVA = pCtx->pVBVA;
602
603 if (!pVBVA)
604 {
605 return false;
606 }
607
608 if (pVBVA->hostFlags.u32SupportedOrders & (1 << code))
609 {
610 return true;
611 }
612
613 return false;
614}
615
616RTDECL(void) VBoxVBVAExSetupBufferContext(PVBVAEXBUFFERCONTEXT pCtx,
617 uint32_t offVRAMBuffer,
618 uint32_t cbBuffer,
619 PFNVBVAEXBUFFERFLUSH pfnFlush,
620 void *pvFlush)
621{
622 memset(pCtx, 0, RT_OFFSETOF(VBVAEXBUFFERCONTEXT, pVBVA));
623 pCtx->offVRAMBuffer = offVRAMBuffer;
624 pCtx->cbBuffer = cbBuffer;
625 pCtx->pfnFlush = pfnFlush;
626 pCtx->pvFlush = pvFlush;
627}
628
629static void* vboxVBVAExIterCur(PVBVAEXBUFFERITERBASE pIter, struct VBVABUFFER *pVBVA, uint32_t *pcbBuffer, bool *pfProcessed)
630{
631 uint32_t cbRecord = pVBVA->aRecords[pIter->iCurRecord].cbRecord;
632 if (cbRecord == VBVA_F_RECORD_PARTIAL)
633 return NULL;
634 if (pcbBuffer)
635 *pcbBuffer = cbRecord;
636 if (pfProcessed)
637 *pfProcessed = !vboxVBVAExIsEntryInRange(pVBVA->indexRecordFirst, pIter->iCurRecord, pVBVA->indexRecordFree);
638 return &pVBVA->au8Data[pIter->off32CurCmd];
639}
640
641DECLINLINE(uint32_t) vboxVBVAExSubst(uint32_t x, uint32_t val, uint32_t maxVal)
642{
643 int32_t result = (int32_t)(x - val);
644 return result >= 0 ? (uint32_t)result : maxVal - (((uint32_t)(-result)) % maxVal);
645}
646
647RTDECL(void) VBoxVBVAExBIterInit(PVBVAEXBUFFERCONTEXT pCtx, PVBVAEXBUFFERBACKWARDITER pIter)
648{
649 struct VBVABUFFER *pVBVA = pCtx->pVBVA;
650 pIter->Base.pCtx = pCtx;
651 uint32_t iCurRecord = vboxVBVAExSubst(pVBVA->indexRecordFree, 1, VBVA_MAX_RECORDS);
652 if (vboxVBVAExIsEntryInRange(pCtx->indexRecordFirstUncompleted, iCurRecord, pVBVA->indexRecordFree))
653 {
654 /* even if the command gets completed by the time we're doing the pCtx->pVBVA->aRecords[iCurRecord].cbRecord below,
655 * the pCtx->pVBVA->aRecords[iCurRecord].cbRecord will still be valid, as it can only be modified by a submitter,
656 * and we are in a submitter context now */
657 pIter->Base.iCurRecord = iCurRecord;
658 pIter->Base.off32CurCmd = vboxVBVAExSubst(pVBVA->off32Free, pCtx->pVBVA->aRecords[iCurRecord].cbRecord, pVBVA->cbData);
659 }
660 else
661 {
662 /* no data */
663 pIter->Base.iCurRecord = pVBVA->indexRecordFree;
664 pIter->Base.off32CurCmd = pVBVA->off32Free;
665 }
666}
667
668RTDECL(void*) VBoxVBVAExBIterNext(PVBVAEXBUFFERBACKWARDITER pIter, uint32_t *pcbBuffer, bool *pfProcessed)
669{
670 PVBVAEXBUFFERCONTEXT pCtx = pIter->Base.pCtx;
671 struct VBVABUFFER *pVBVA = pCtx->pVBVA;
672 uint32_t indexRecordFirstUncompleted = pCtx->indexRecordFirstUncompleted;
673 if (!vboxVBVAExIsEntryInRange(indexRecordFirstUncompleted, pIter->Base.iCurRecord, pVBVA->indexRecordFree))
674 return NULL;
675
676 void *pvBuffer = vboxVBVAExIterCur(&pIter->Base, pVBVA, pcbBuffer, pfProcessed);
677 AssertRelease(pvBuffer);
678
679 /* even if the command gets completed by the time we're doing the pCtx->pVBVA->aRecords[pIter->Base.iCurRecord].cbRecord below,
680 * the pCtx->pVBVA->aRecords[pIter->Base.iCurRecord].cbRecord will still be valid, as it can only be modified by a submitter,
681 * and we are in a submitter context now */
682 pIter->Base.iCurRecord = vboxVBVAExSubst(pIter->Base.iCurRecord, 1, VBVA_MAX_RECORDS);
683 pIter->Base.off32CurCmd = vboxVBVAExSubst(pIter->Base.off32CurCmd, pCtx->pVBVA->aRecords[pIter->Base.iCurRecord].cbRecord, pVBVA->cbData);
684
685 return pvBuffer;
686}
687
688RTDECL(void) VBoxVBVAExCFIterInit(PVBVAEXBUFFERCONTEXT pCtx, PVBVAEXBUFFERFORWARDITER pIter)
689{
690 pIter->Base.pCtx = pCtx;
691 pIter->Base.iCurRecord = pCtx->indexRecordFirstUncompleted;
692 pIter->Base.off32CurCmd = pCtx->off32DataUncompleted;
693}
694
695RTDECL(void*) VBoxVBVAExCFIterNext(PVBVAEXBUFFERFORWARDITER pIter, uint32_t *pcbBuffer, bool *pfProcessed)
696{
697 PVBVAEXBUFFERCONTEXT pCtx = pIter->Base.pCtx;
698 struct VBVABUFFER *pVBVA = pCtx->pVBVA;
699 uint32_t indexRecordFree = pVBVA->indexRecordFree;
700 if (!vboxVBVAExIsEntryInRange(pCtx->indexRecordFirstUncompleted, pIter->Base.iCurRecord, indexRecordFree))
701 return NULL;
702
703 uint32_t cbBuffer;
704 void *pvData = vboxVBVAExIterCur(&pIter->Base, pVBVA, &cbBuffer, pfProcessed);
705 if (!pvData)
706 return NULL;
707
708 pIter->Base.iCurRecord = (pIter->Base.iCurRecord + 1) % VBVA_MAX_RECORDS;
709 pIter->Base.off32CurCmd = (pIter->Base.off32CurCmd + cbBuffer) % pVBVA->cbData;
710
711 if (pcbBuffer)
712 *pcbBuffer = cbBuffer;
713
714 return pvData;
715}
716
717/**/
718
719int VBoxCmdVbvaEnable(PVBOXMP_DEVEXT pDevExt, VBOXCMDVBVA *pVbva)
720{
721 return VBoxVBVAExEnable(&pVbva->Vbva, &VBoxCommonFromDeviceExt(pDevExt)->guestCtx, pVbva->Vbva.pVBVA);
722}
723
724int VBoxCmdVbvaDisable(PVBOXMP_DEVEXT pDevExt, VBOXCMDVBVA *pVbva)
725{
726 VBoxVBVAExDisable(&pVbva->Vbva, &VBoxCommonFromDeviceExt(pDevExt)->guestCtx);
727 return VINF_SUCCESS;
728}
729
730int VBoxCmdVbvaDestroy(PVBOXMP_DEVEXT pDevExt, VBOXCMDVBVA *pVbva)
731{
732 int rc = VINF_SUCCESS;
733 VBoxMPCmnUnmapAdapterMemory(VBoxCommonFromDeviceExt(pDevExt), (void**)&pVbva->Vbva.pVBVA);
734 memset(pVbva, 0, sizeof (*pVbva));
735 return rc;
736}
737
738static void vboxCmdVbvaDdiNotifyCompleteIrq(PVBOXMP_DEVEXT pDevExt, VBOXCMDVBVA *pVbva, UINT u32FenceId, DXGK_INTERRUPT_TYPE enmComplType)
739{
740 DXGKARGCB_NOTIFY_INTERRUPT_DATA notify;
741 memset(&notify, 0, sizeof(DXGKARGCB_NOTIFY_INTERRUPT_DATA));
742 switch (enmComplType)
743 {
744 case DXGK_INTERRUPT_DMA_COMPLETED:
745 notify.InterruptType = DXGK_INTERRUPT_DMA_COMPLETED;
746 notify.DmaCompleted.SubmissionFenceId = u32FenceId;
747 notify.DmaCompleted.NodeOrdinal = pVbva->idNode;
748 break;
749
750 case DXGK_INTERRUPT_DMA_PREEMPTED:
751 notify.InterruptType = DXGK_INTERRUPT_DMA_PREEMPTED;
752 notify.DmaPreempted.PreemptionFenceId = u32FenceId;
753 notify.DmaPreempted.NodeOrdinal = pVbva->idNode;
754 notify.DmaPreempted.LastCompletedFenceId = pVbva->u32FenceCompleted;
755 break;
756
757 case DXGK_INTERRUPT_DMA_FAULTED:
758 Assert(0);
759 notify.InterruptType = DXGK_INTERRUPT_DMA_FAULTED;
760 notify.DmaFaulted.FaultedFenceId = u32FenceId;
761 notify.DmaFaulted.Status = STATUS_UNSUCCESSFUL; /* @todo: better status ? */
762 notify.DmaFaulted.NodeOrdinal = pVbva->idNode;
763 break;
764
765 default:
766 WARN(("unrecognized completion type %d", enmComplType));
767 break;
768 }
769
770 pDevExt->u.primary.DxgkInterface.DxgkCbNotifyInterrupt(pDevExt->u.primary.DxgkInterface.DeviceHandle, &notify);
771}
772
773typedef struct VBOXCMDVBVA_NOTIFYPREEMPT_CB
774{
775 PVBOXMP_DEVEXT pDevExt;
776 VBOXCMDVBVA *pVbva;
777 int rc;
778 UINT u32SubmitFenceId;
779 UINT u32PreemptFenceId;
780} VBOXCMDVBVA_NOTIFYPREEMPT_CB;
781
782static BOOLEAN vboxCmdVbvaDdiNotifyPreemptCb(PVOID pvContext)
783{
784 VBOXCMDVBVA_NOTIFYPREEMPT_CB* pData = (VBOXCMDVBVA_NOTIFYPREEMPT_CB*)pvContext;
785 PVBOXMP_DEVEXT pDevExt = pData->pDevExt;
786 VBOXCMDVBVA *pVbva = pData->pVbva;
787 if (!pData->u32SubmitFenceId || pVbva->u32FenceCompleted == pData->u32SubmitFenceId)
788 {
789 vboxCmdVbvaDdiNotifyCompleteIrq(pDevExt, pVbva, pData->u32PreemptFenceId, DXGK_INTERRUPT_DMA_PREEMPTED);
790
791 pDevExt->u.primary.DxgkInterface.DxgkCbQueueDpc(pDevExt->u.primary.DxgkInterface.DeviceHandle);
792 }
793 else
794 {
795 Assert(pVbva->u32FenceCompleted < pData->u32SubmitFenceId);
796 Assert(pVbva->cPreempt <= VBOXCMDVBVA_PREEMPT_EL_SIZE);
797 if (pVbva->cPreempt == VBOXCMDVBVA_PREEMPT_EL_SIZE)
798 {
799 WARN(("no more free elements in preempt map"));
800 pData->rc = VERR_BUFFER_OVERFLOW;
801 return FALSE;
802 }
803 uint32_t iNewEl = (pVbva->iCurPreempt + pVbva->cPreempt) % VBOXCMDVBVA_PREEMPT_EL_SIZE;
804 Assert(iNewEl < VBOXCMDVBVA_PREEMPT_EL_SIZE);
805 pVbva->aPreempt[iNewEl].u32SubmitFence = pData->u32SubmitFenceId;
806 pVbva->aPreempt[iNewEl].u32PreemptFence = pData->u32PreemptFenceId;
807 ++pVbva->cPreempt;
808 }
809
810 pData->rc = VINF_SUCCESS;
811 return TRUE;
812}
813
814static int vboxCmdVbvaDdiNotifyPreempt(PVBOXMP_DEVEXT pDevExt, VBOXCMDVBVA *pVbva, UINT u32SubmitFenceId, UINT u32PreemptFenceId)
815{
816 VBOXCMDVBVA_NOTIFYPREEMPT_CB Data;
817 Data.pDevExt = pDevExt;
818 Data.pVbva = pVbva;
819 Data.rc = VERR_INTERNAL_ERROR;
820 Data.u32SubmitFenceId = u32SubmitFenceId;
821 Data.u32PreemptFenceId = u32PreemptFenceId;
822 BOOLEAN bDummy;
823 NTSTATUS Status = pDevExt->u.primary.DxgkInterface.DxgkCbSynchronizeExecution(
824 pDevExt->u.primary.DxgkInterface.DeviceHandle,
825 vboxCmdVbvaDdiNotifyPreemptCb,
826 &Data,
827 0, /* IN ULONG MessageNumber */
828 &bDummy);
829 if (!NT_SUCCESS(Status))
830 {
831 WARN(("DxgkCbSynchronizeExecution failed Status %#x", Status));
832 return VERR_GENERAL_FAILURE;
833 }
834
835 if (!RT_SUCCESS(Data.rc))
836 {
837 WARN(("vboxCmdVbvaDdiNotifyPreemptCb failed rc %d", Data.rc));
838 return Data.rc;
839 }
840
841 return VINF_SUCCESS;
842}
843
844static int vboxCmdVbvaFlush(PVBOXMP_DEVEXT pDevExt, HGSMIGUESTCOMMANDCONTEXT *pCtx, bool fBufferOverflow)
845{
846 /* Issue the flush command. */
847 VBVACMDVBVAFLUSH *pFlush = (VBVACMDVBVAFLUSH*)VBoxHGSMIBufferAlloc(pCtx,
848 sizeof (VBVACMDVBVAFLUSH),
849 HGSMI_CH_VBVA,
850 VBVA_CMDVBVA_FLUSH);
851 if (!pFlush)
852 {
853 WARN(("VBoxHGSMIBufferAlloc failed\n"));
854 return VERR_OUT_OF_RESOURCES;
855 }
856
857 pFlush->u32Flags = fBufferOverflow ? VBVACMDVBVAFLUSH_F_GUEST_BUFFER_OVERFLOW : 0;
858
859 VBoxHGSMIBufferSubmit(pCtx, pFlush);
860
861 VBoxHGSMIBufferFree(pCtx, pFlush);
862
863 return VINF_SUCCESS;
864}
865
866typedef struct VBOXCMDVBVA_CHECK_COMPLETED_CB
867{
868 PVBOXMP_DEVEXT pDevExt;
869 VBOXCMDVBVA *pVbva;
870 uint32_t u32FenceID;
871} VBOXCMDVBVA_CHECK_COMPLETED_CB;
872
873static BOOLEAN vboxCmdVbvaCheckCompletedIrqCb(PVOID pContext)
874{
875 VBOXCMDVBVA_CHECK_COMPLETED_CB *pCompleted = (VBOXCMDVBVA_CHECK_COMPLETED_CB*)pContext;
876 BOOLEAN bRc = DxgkDdiInterruptRoutineNew(pCompleted->pDevExt, 0);
877 if (pCompleted->pVbva)
878 pCompleted->u32FenceID = pCompleted->pVbva->u32FenceCompleted;
879 return bRc;
880}
881
882
883static uint32_t vboxCmdVbvaCheckCompleted(PVBOXMP_DEVEXT pDevExt, VBOXCMDVBVA *pVbva, bool fPingHost, HGSMIGUESTCOMMANDCONTEXT *pCtx, bool fBufferOverflow)
884{
885 if (fPingHost)
886 vboxCmdVbvaFlush(pDevExt, pCtx, fBufferOverflow);
887
888 VBOXCMDVBVA_CHECK_COMPLETED_CB context;
889 context.pDevExt = pDevExt;
890 context.pVbva = pVbva;
891 context.u32FenceID = 0;
892 BOOLEAN bRet;
893 NTSTATUS Status = pDevExt->u.primary.DxgkInterface.DxgkCbSynchronizeExecution(
894 pDevExt->u.primary.DxgkInterface.DeviceHandle,
895 vboxCmdVbvaCheckCompletedIrqCb,
896 &context,
897 0, /* IN ULONG MessageNumber */
898 &bRet);
899 Assert(Status == STATUS_SUCCESS);
900
901 return context.u32FenceID;
902}
903
904DECLCALLBACK(void) voxCmdVbvaFlushCb(struct VBVAEXBUFFERCONTEXT *pCtx, PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx, void *pvFlush)
905{
906 PVBOXMP_DEVEXT pDevExt = (PVBOXMP_DEVEXT)pvFlush;
907
908 vboxCmdVbvaCheckCompleted(pDevExt, NULL, true /*fPingHost*/, pHGSMICtx, true /*fBufferOverflow*/);
909}
910
911int VBoxCmdVbvaCreate(PVBOXMP_DEVEXT pDevExt, VBOXCMDVBVA *pVbva, ULONG offBuffer, ULONG cbBuffer)
912{
913 memset(pVbva, 0, sizeof (*pVbva));
914
915 int rc = VBoxMPCmnMapAdapterMemory(VBoxCommonFromDeviceExt(pDevExt),
916 (void**)&pVbva->Vbva.pVBVA,
917 offBuffer,
918 cbBuffer);
919 if (RT_SUCCESS(rc))
920 {
921 Assert(pVbva->Vbva.pVBVA);
922 VBoxVBVAExSetupBufferContext(&pVbva->Vbva, offBuffer, cbBuffer, voxCmdVbvaFlushCb, pDevExt);
923 }
924 else
925 {
926 WARN(("VBoxMPCmnMapAdapterMemory failed rc %d", rc));
927 }
928
929 return rc;
930}
931
932int VBoxCmdVbvaSubmit(PVBOXMP_DEVEXT pDevExt, VBOXCMDVBVA *pVbva, struct VBOXCMDVBVA_HDR *pCmd, uint32_t cbCmd)
933{
934 int rc = VINF_SUCCESS;
935
936 Assert(pCmd->u32FenceID);
937
938 pCmd->u8State = VBOXCMDVBVA_STATE_SUBMITTED;
939#ifdef DEBUG_misha
940 Assert(pCmd->u32FenceID == pVbva->u32FenceSubmitted + 1);
941#endif
942 pVbva->u32FenceSubmitted = pCmd->u32FenceID;
943
944 if (VBoxVBVAExGetSize(&pVbva->Vbva) < cbCmd)
945 {
946 WARN(("buffer does not fit the vbva buffer, we do not support splitting buffers"));
947 return VERR_NOT_SUPPORTED;
948 }
949
950 if (!VBoxVBVAExBufferBeginUpdate(&pVbva->Vbva, &VBoxCommonFromDeviceExt(pDevExt)->guestCtx))
951 {
952 WARN(("VBoxVBVAExBufferBeginUpdate failed!"));
953 return VERR_GENERAL_FAILURE;
954 }
955
956 void* pvBuffer = VBoxVBVAExAllocContiguous(&pVbva->Vbva, &VBoxCommonFromDeviceExt(pDevExt)->guestCtx, cbCmd);
957 if (!pvBuffer)
958 {
959 WARN(("failed to allocate contiguous buffer, trying nopping the tail"));
960 uint32_t cbTail = VBoxVBVAExGetFreeTail(&pVbva->Vbva);
961 if (!cbTail)
962 {
963 WARN(("this is not a free tail case, cbTail is NULL"));
964 return VERR_BUFFER_OVERFLOW;
965 }
966
967 Assert(cbTail < cbCmd);
968
969 pvBuffer = VBoxVBVAExAllocContiguous(&pVbva->Vbva, &VBoxCommonFromDeviceExt(pDevExt)->guestCtx, cbTail);
970
971 Assert(pvBuffer);
972
973 *((uint8_t*)pvBuffer) = VBOXCMDVBVA_OPTYPE_NOP;
974
975 VBoxVBVAExBufferEndUpdate(&pVbva->Vbva);
976
977 if (!VBoxVBVAExBufferBeginUpdate(&pVbva->Vbva, &VBoxCommonFromDeviceExt(pDevExt)->guestCtx))
978 {
979 WARN(("VBoxVBVAExBufferBeginUpdate 2 failed!"));
980 return VERR_GENERAL_FAILURE;
981 }
982
983 pvBuffer = VBoxVBVAExAllocContiguous(&pVbva->Vbva, &VBoxCommonFromDeviceExt(pDevExt)->guestCtx, cbCmd);
984 if (!pvBuffer)
985 {
986 WARN(("failed to allocate contiguous buffer, failing"));
987 return VERR_GENERAL_FAILURE;
988 }
989 }
990
991 Assert(pvBuffer);
992
993 memcpy(pvBuffer, pCmd, cbCmd);
994
995 VBoxVBVAExBufferEndUpdate(&pVbva->Vbva);
996
997 if (!VBoxVBVAExIsProcessing(&pVbva->Vbva))
998 {
999 /* Issue the submit command. */
1000 HGSMIGUESTCOMMANDCONTEXT *pCtx = &VBoxCommonFromDeviceExt(pDevExt)->guestCtx;
1001 VBVACMDVBVASUBMIT *pSubmit = (VBVACMDVBVASUBMIT*)VBoxHGSMIBufferAlloc(pCtx,
1002 sizeof (VBVACMDVBVASUBMIT),
1003 HGSMI_CH_VBVA,
1004 VBVA_CMDVBVA_SUBMIT);
1005 if (!pSubmit)
1006 {
1007 WARN(("VBoxHGSMIBufferAlloc failed\n"));
1008 return VERR_OUT_OF_RESOURCES;
1009 }
1010
1011 pSubmit->u32Reserved = 0;
1012
1013 VBoxHGSMIBufferSubmit(pCtx, pSubmit);
1014
1015 VBoxHGSMIBufferFree(pCtx, pSubmit);
1016 }
1017
1018 return VINF_SUCCESS;
1019}
1020
1021bool VBoxCmdVbvaPreempt(PVBOXMP_DEVEXT pDevExt, VBOXCMDVBVA *pVbva, uint32_t u32FenceID)
1022{
1023 VBVAEXBUFFERBACKWARDITER Iter;
1024 VBoxVBVAExBIterInit(&pVbva->Vbva, &Iter);
1025
1026 uint32_t cbBuffer;
1027 bool fProcessed;
1028 uint8_t* pu8Cmd;
1029 uint32_t u32SubmitFence = 0;
1030
1031 /* we can do it right here */
1032 while ((pu8Cmd = (uint8_t*)VBoxVBVAExBIterNext(&Iter, &cbBuffer, &fProcessed)) != NULL)
1033 {
1034 if (*pu8Cmd == VBOXCMDVBVA_OPTYPE_NOP)
1035 continue;
1036
1037 VBOXCMDVBVA_HDR *pCmd = (VBOXCMDVBVA_HDR*)pu8Cmd;
1038
1039 if (ASMAtomicCmpXchgU8(&pCmd->u8State, VBOXCMDVBVA_STATE_CANCELLED, VBOXCMDVBVA_STATE_SUBMITTED)
1040 || pCmd->u8State == VBOXCMDVBVA_STATE_CANCELLED)
1041 continue;
1042
1043 Assert(pCmd->u8State == VBOXCMDVBVA_STATE_IN_PROGRESS);
1044
1045 u32SubmitFence = ASMAtomicUoReadU32(&pCmd->u32FenceID);
1046 break;
1047 }
1048
1049 vboxCmdVbvaDdiNotifyPreempt(pDevExt, pVbva, u32SubmitFence, u32FenceID);
1050
1051 return false;
1052}
1053
1054bool VBoxCmdVbvaCheckCompletedIrq(PVBOXMP_DEVEXT pDevExt, VBOXCMDVBVA *pVbva)
1055{
1056 VBVAEXBUFFERFORWARDITER Iter;
1057 VBoxVBVAExCFIterInit(&pVbva->Vbva, &Iter);
1058
1059 bool fHasCommandsCompletedPreempted = false;
1060 bool fProcessed;
1061 uint8_t* pu8Cmd;
1062
1063
1064 while ((pu8Cmd = (uint8_t*)VBoxVBVAExCFIterNext(&Iter, NULL, &fProcessed)) != NULL)
1065 {
1066 if (!fProcessed)
1067 break;
1068
1069 if (*pu8Cmd == VBOXCMDVBVA_OPTYPE_NOP)
1070 continue;
1071
1072 VBOXCMDVBVA_HDR *pCmd = (VBOXCMDVBVA_HDR*)pu8Cmd;
1073 uint8_t u8State = pCmd->u8State;
1074 uint32_t u32FenceID = pCmd->u32FenceID;
1075
1076 Assert(u8State == VBOXCMDVBVA_STATE_IN_PROGRESS
1077 || u8State == VBOXCMDVBVA_STATE_CANCELLED);
1078 Assert(u32FenceID);
1079 VBoxVBVAExCBufferCompleted(&pVbva->Vbva);
1080
1081 if (u8State == VBOXCMDVBVA_STATE_IN_PROGRESS)
1082 {
1083 Assert(u32FenceID);
1084 if (!u32FenceID)
1085 continue;
1086
1087#ifdef DEBUG_misha
1088 Assert(u32FenceID == pVbva->u32FenceCompleted + 1);
1089#endif
1090 pVbva->u32FenceCompleted = u32FenceID;
1091 }
1092 else
1093 {
1094 Assert(u8State == VBOXCMDVBVA_STATE_CANCELLED);
1095 continue;
1096 }
1097
1098 Assert(u32FenceID);
1099 vboxCmdVbvaDdiNotifyCompleteIrq(pDevExt, pVbva, u32FenceID, DXGK_INTERRUPT_DMA_COMPLETED);
1100
1101 if (pVbva->cPreempt && pVbva->aPreempt[pVbva->iCurPreempt].u32SubmitFence == u32FenceID)
1102 {
1103 Assert(pVbva->aPreempt[pVbva->iCurPreempt].u32PreemptFence);
1104 vboxCmdVbvaDdiNotifyCompleteIrq(pDevExt, pVbva, pVbva->aPreempt[pVbva->iCurPreempt].u32PreemptFence, DXGK_INTERRUPT_DMA_PREEMPTED);
1105 --pVbva->cPreempt;
1106 if (!pVbva->cPreempt)
1107 pVbva->iCurPreempt = 0;
1108 else
1109 {
1110 ++pVbva->iCurPreempt;
1111 pVbva->iCurPreempt %= VBOXCMDVBVA_PREEMPT_EL_SIZE;
1112 }
1113 }
1114
1115 fHasCommandsCompletedPreempted = true;
1116 }
1117
1118#ifdef DEBUG
1119 vboxHwBufferVerifyCompleted(&pVbva->Vbva);
1120#endif
1121
1122 return fHasCommandsCompletedPreempted;
1123}
1124
1125uint32_t VBoxCmdVbvaCheckCompleted(PVBOXMP_DEVEXT pDevExt, VBOXCMDVBVA *pVbva, bool fPingHost)
1126{
1127 return vboxCmdVbvaCheckCompleted(pDevExt, pVbva, fPingHost, &VBoxCommonFromDeviceExt(pDevExt)->guestCtx, false /* fBufferOverflow */);
1128}
1129
1130#if 0
1131static uint32_t vboxCVDdiSysMemElBuild(VBOXCMDVBVA_SYSMEMEL *pEl, PMDL pMdl, uint32_t iPfn, uint32_t cPages)
1132{
1133 PFN_NUMBER cur = MmGetMdlPfnArray(pMdl)[iPfn];
1134 uint32_t cbEl = sizeof (*pEl);
1135 uint32_t cStoredPages = 1;
1136 PFN_NUMBER next;
1137 pEl->iPage1 = (uint32_t)(cur & 0xfffff);
1138 pEl->iPage2 = (uint32_t)(cur >> 20);
1139 --cPages;
1140 for ( ; cPages && cStoredPages < VBOXCMDVBVA_SYSMEMEL_CPAGES_MAX; --cPages, ++cStoredPages, cur = next)
1141 {
1142 next = MmGetMdlPfnArray(pMdl)[iPfn+cStoredPages];
1143 if (next != cur+1)
1144 break;
1145 }
1146
1147 Assert(cStoredPages);
1148 pEl->cPagesAfterFirst = cStoredPages - 1;
1149
1150 return cPages;
1151}
1152
1153uint32_t VBoxCVDdiPTransferVRamSysBuildEls(VBOXCMDVBVA_PAGING_TRANSFER *pCmd, PMDL pMdl, uint32_t iPfn, uint32_t cPages, uint32_t cbBuffer, uint32_t *pcPagesWritten)
1154{
1155 uint32_t cInitPages = cPages;
1156 uint32_t cbInitBuffer = cbBuffer;
1157 uint32_t cEls = 0;
1158 VBOXCMDVBVA_SYSMEMEL *pEl = pCmd->aSysMem;
1159
1160 Assert(cbBuffer >= sizeof (VBOXCMDVBVA_PAGING_TRANSFER));
1161
1162 cbBuffer -= RT_OFFSETOF(VBOXCMDVBVA_PAGING_TRANSFER, aSysMem);
1163
1164 for (; cPages && cbBuffer >= sizeof (VBOXCMDVBVA_SYSMEMEL); ++cEls, cbBuffer-=sizeof (VBOXCMDVBVA_SYSMEMEL), ++pEl)
1165 {
1166 cPages = vboxCVDdiSysMemElBuild(pEl, pMdl, iPfn + cInitPages - cPages, cPages);
1167 }
1168
1169 *pcPagesWritten = cInitPages - cPages;
1170 return cbInitBuffer - cbBuffer;
1171}
1172#endif
1173
1174uint32_t VBoxCVDdiPTransferVRamSysBuildEls(VBOXCMDVBVA_PAGING_TRANSFER *pCmd, PMDL pMdl, uint32_t iPfn, uint32_t cPages, uint32_t cbBuffer, uint32_t *pcPagesWritten)
1175{
1176 uint32_t cbInitBuffer = cbBuffer;
1177 uint32_t i = 0;
1178 VBOXCMDVBVAPAGEIDX *pPageNumbers = pCmd->Data.aPageNumbers;
1179
1180 cbBuffer -= RT_OFFSETOF(VBOXCMDVBVA_PAGING_TRANSFER, Data.aPageNumbers);
1181
1182 for (; i < cPages && cbBuffer >= sizeof (*pPageNumbers); ++i, cbBuffer -= sizeof (*pPageNumbers))
1183 {
1184 pPageNumbers[i] = (VBOXCMDVBVAPAGEIDX)(MmGetMdlPfnArray(pMdl)[iPfn + i]);
1185 }
1186
1187 *pcPagesWritten = i;
1188 Assert(cbInitBuffer - cbBuffer == RT_OFFSETOF(VBOXCMDVBVA_PAGING_TRANSFER, Data.aPageNumbers[i]));
1189 return cbInitBuffer - cbBuffer;
1190}
1191
1192
1193int vboxCmdVbvaConConnect(PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx,
1194 uint32_t crVersionMajor, uint32_t crVersionMinor,
1195 uint32_t *pu32ClientID)
1196{
1197 VBOXCMDVBVA_CTL_3DCTL_CONNECT *pConnect = (VBOXCMDVBVA_CTL_3DCTL_CONNECT*)vboxCmdVbvaCtlCreate(pHGSMICtx, sizeof (VBOXCMDVBVA_CTL_3DCTL_CONNECT));
1198 if (!pConnect)
1199 {
1200 WARN(("vboxCmdVbvaCtlCreate failed"));
1201 return VERR_OUT_OF_RESOURCES;
1202 }
1203 pConnect->Hdr.u32Type = VBOXCMDVBVACTL_TYPE_3DCTL;
1204 pConnect->Hdr.i32Result = VERR_NOT_SUPPORTED;
1205 pConnect->Connect.Hdr.u32Type = VBOXCMDVBVA3DCTL_TYPE_CONNECT;
1206 pConnect->Connect.Hdr.u32CmdClientId = 0;
1207 pConnect->Connect.u32MajorVersion = crVersionMajor;
1208 pConnect->Connect.u32MinorVersion = crVersionMinor;
1209 pConnect->Connect.u64Pid = (uint64_t)PsGetCurrentProcessId();
1210
1211 int rc = vboxCmdVbvaCtlSubmitSync(pHGSMICtx, &pConnect->Hdr);
1212 if (RT_SUCCESS(rc))
1213 {
1214 rc = pConnect->Hdr.i32Result;
1215 if (RT_SUCCESS(rc))
1216 {
1217 Assert(pConnect->Connect.Hdr.u32CmdClientId);
1218 *pu32ClientID = pConnect->Connect.Hdr.u32CmdClientId;
1219 }
1220 else
1221 WARN(("VBOXCMDVBVA3DCTL_TYPE_CONNECT Disable failed %d", rc));
1222 }
1223 else
1224 WARN(("vboxCmdVbvaCtlSubmitSync returnd %d", rc));
1225
1226 vboxCmdVbvaCtlFree(pHGSMICtx, &pConnect->Hdr);
1227
1228 return rc;
1229}
1230
1231int vboxCmdVbvaConDisconnect(PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx, uint32_t u32ClientID)
1232{
1233 VBOXCMDVBVA_CTL_3DCTL *pDisconnect = (VBOXCMDVBVA_CTL_3DCTL*)vboxCmdVbvaCtlCreate(pHGSMICtx, sizeof (VBOXCMDVBVA_CTL_3DCTL));
1234 if (!pDisconnect)
1235 {
1236 WARN(("vboxCmdVbvaCtlCreate failed"));
1237 return VERR_OUT_OF_RESOURCES;
1238 }
1239 pDisconnect->Hdr.u32Type = VBOXCMDVBVACTL_TYPE_3DCTL;
1240 pDisconnect->Hdr.i32Result = VERR_NOT_SUPPORTED;
1241 pDisconnect->Ctl.u32Type = VBOXCMDVBVA3DCTL_TYPE_DISCONNECT;
1242 pDisconnect->Ctl.u32CmdClientId = u32ClientID;
1243
1244 int rc = vboxCmdVbvaCtlSubmitSync(pHGSMICtx, &pDisconnect->Hdr);
1245 if (RT_SUCCESS(rc))
1246 {
1247 rc = pDisconnect->Hdr.i32Result;
1248 if (!RT_SUCCESS(rc))
1249 WARN(("VBOXCMDVBVA3DCTL_TYPE_DISCONNECT Disable failed %d", rc));
1250 }
1251 else
1252 WARN(("vboxCmdVbvaCtlSubmitSync returnd %d", rc));
1253
1254 vboxCmdVbvaCtlFree(pHGSMICtx, &pDisconnect->Hdr);
1255
1256 return rc;
1257}
1258
1259int VBoxCmdVbvaConConnect(PVBOXMP_DEVEXT pDevExt, VBOXCMDVBVA *pVbva,
1260 uint32_t crVersionMajor, uint32_t crVersionMinor,
1261 uint32_t *pu32ClientID)
1262{
1263 return vboxCmdVbvaConConnect(&VBoxCommonFromDeviceExt(pDevExt)->guestCtx, crVersionMajor, crVersionMinor, pu32ClientID);
1264}
1265
1266int VBoxCmdVbvaConDisconnect(PVBOXMP_DEVEXT pDevExt, VBOXCMDVBVA *pVbva, uint32_t u32ClientID)
1267{
1268 return vboxCmdVbvaConDisconnect(&VBoxCommonFromDeviceExt(pDevExt)->guestCtx, u32ClientID);
1269}
1270
1271VBOXCMDVBVA_CRCMD_CMD* vboxCmdVbvaConCmdAlloc(PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx, uint32_t cbCmd)
1272{
1273 VBOXCMDVBVA_CTL_3DCTL_CMD *pCmd = (VBOXCMDVBVA_CTL_3DCTL_CMD*)vboxCmdVbvaCtlCreate(pHGSMICtx, sizeof (VBOXCMDVBVA_CTL_3DCTL_CMD) + cbCmd);
1274 if (!pCmd)
1275 {
1276 WARN(("vboxCmdVbvaCtlCreate failed"));
1277 return NULL;
1278 }
1279 pCmd->Hdr.u32Type = VBOXCMDVBVACTL_TYPE_3DCTL;
1280 pCmd->Hdr.i32Result = VERR_NOT_SUPPORTED;
1281 pCmd->Cmd.Hdr.u32Type = VBOXCMDVBVA3DCTL_TYPE_CMD;
1282 pCmd->Cmd.Hdr.u32CmdClientId = 0;
1283 pCmd->Cmd.Cmd.u8OpCode = VBOXCMDVBVA_OPTYPE_CRCMD;
1284 pCmd->Cmd.Cmd.u8Flags = 0;
1285 pCmd->Cmd.Cmd.u8State = VBOXCMDVBVA_STATE_SUBMITTED;
1286 pCmd->Cmd.Cmd.u.i8Result = -1;
1287 pCmd->Cmd.Cmd.u32FenceID = 0;
1288
1289 return (VBOXCMDVBVA_CRCMD_CMD*)(pCmd+1);
1290}
1291
1292void vboxCmdVbvaConCmdFree(PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx, VBOXCMDVBVA_CRCMD_CMD *pCmd)
1293{
1294 VBOXCMDVBVA_CTL_3DCTL_CMD *pHdr = ((VBOXCMDVBVA_CTL_3DCTL_CMD*)pCmd)-1;
1295 vboxCmdVbvaCtlFree(pHGSMICtx, &pHdr->Hdr);
1296}
1297
1298int vboxCmdVbvaConSubmitAsync(PHGSMIGUESTCOMMANDCONTEXT pHGSMICtx, VBOXCMDVBVA_CRCMD_CMD* pCmd, FNVBOXSHGSMICMDCOMPLETION pfnCompletion, void *pvCompletion)
1299{
1300 VBOXCMDVBVA_CTL_3DCTL_CMD *pHdr = ((VBOXCMDVBVA_CTL_3DCTL_CMD*)pCmd)-1;
1301 return vboxCmdVbvaCtlSubmitAsync(pHGSMICtx, &pHdr->Hdr, pfnCompletion, pvCompletion);
1302}
1303
1304VBOXCMDVBVA_CRCMD_CMD* VBoxCmdVbvaConCmdAlloc(PVBOXMP_DEVEXT pDevExt, uint32_t cbCmd)
1305{
1306 return vboxCmdVbvaConCmdAlloc(&VBoxCommonFromDeviceExt(pDevExt)->guestCtx, cbCmd);
1307}
1308
1309void VBoxCmdVbvaConCmdFree(PVBOXMP_DEVEXT pDevExt, VBOXCMDVBVA_CRCMD_CMD *pCmd)
1310{
1311 vboxCmdVbvaConCmdFree(&VBoxCommonFromDeviceExt(pDevExt)->guestCtx, pCmd);
1312}
1313
1314int VBoxCmdVbvaConCmdSubmitAsync(PVBOXMP_DEVEXT pDevExt, VBOXCMDVBVA_CRCMD_CMD* pCmd, FNVBOXSHGSMICMDCOMPLETION pfnCompletion, void *pvCompletion)
1315{
1316 return vboxCmdVbvaConSubmitAsync(&VBoxCommonFromDeviceExt(pDevExt)->guestCtx, pCmd, pfnCompletion, pvCompletion);
1317}
1318
1319int VBoxCmdVbvaConCmdCompletionData(void *pvCmd, VBOXCMDVBVA_CRCMD_CMD **ppCmd)
1320{
1321 VBOXCMDVBVA_CTL_3DCTL_CMD *pCmd = (VBOXCMDVBVA_CTL_3DCTL_CMD*)pvCmd;
1322 if (ppCmd)
1323 *ppCmd = (VBOXCMDVBVA_CRCMD_CMD*)(pCmd+1);
1324 return pCmd->Hdr.i32Result;
1325}
1326#endif
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