VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/Graphics/Video/disp/xpdm/VBoxDispVBVA.cpp@ 37273

Last change on this file since 37273 was 36867, checked in by vboxsync, 14 years ago

Additions/Video: display/miniport drivers

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 18.8 KB
Line 
1/* $Id: VBoxDispVBVA.cpp 36867 2011-04-28 07:27:03Z vboxsync $ */
2
3/** @file
4 * VBox XPDM Display driver
5 */
6
7/*
8 * Copyright (C) 2011 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 "VBoxDisp.h"
20#include "VBoxDispMini.h"
21#include <VBox/HGSMI/HGSMIChSetup.h>
22
23#ifdef VBOX_VBVA_ADJUST_RECT
24static ULONG vbvaConvertPixel(BYTE *pu8PixelFrom, int cbPixelFrom, int cbPixelTo)
25{
26 BYTE r, g, b;
27 ULONG ulConvertedPixel = 0;
28
29 switch (cbPixelFrom)
30 {
31 case 4:
32 {
33 switch (cbPixelTo)
34 {
35 case 3:
36 {
37 memcpy (&ulConvertedPixel, pu8PixelFrom, 3);
38 } break;
39
40 case 2:
41 {
42 ulConvertedPixel = *(ULONG *)pu8PixelFrom;
43
44 r = (BYTE)(ulConvertedPixel >> 16);
45 g = (BYTE)(ulConvertedPixel >> 8);
46 b = (BYTE)(ulConvertedPixel);
47
48 ulConvertedPixel = ((r >> 3) << 11) + ((g >> 2) << 5) + (b >> 3);
49 } break;
50 }
51 } break;
52
53 case 3:
54 {
55 switch (cbPixelTo)
56 {
57 case 2:
58 {
59 memcpy (&ulConvertedPixel, pu8PixelFrom, 3);
60
61 r = (BYTE)(ulConvertedPixel >> 16);
62 g = (BYTE)(ulConvertedPixel >> 8);
63 b = (BYTE)(ulConvertedPixel);
64
65 ulConvertedPixel = ((r >> 3) << 11) + ((g >> 2) << 5) + (b >> 3);
66 } break;
67 }
68 } break;
69 }
70
71 return ulConvertedPixel;
72}
73
74BOOL vbvaFindChangedRect(SURFOBJ *psoDest, SURFOBJ *psoSrc, RECTL *prclDest, POINTL *pptlSrc)
75{
76 int x, y;
77 int fTopNonEqualFound;
78 int yTopmost;
79 int yBottommost;
80 int cbPixelSrc;
81 int cbPixelDest;
82 RECTL rclDest;
83 RECTL rclSrc;
84 BYTE *pu8Src;
85 BYTE *pu8Dest;
86
87 if (!prclDest || !pptlSrc)
88 {
89 return TRUE;
90 }
91
92 LOGF(("dest %d,%d %dx%d from %d,%d",
93 prclDest->left, prclDest->top, prclDest->right - prclDest->left, prclDest->bottom - prclDest->top,
94 pptlSrc->x, pptlSrc->y));
95
96 switch (psoDest->iBitmapFormat)
97 {
98 case BMF_16BPP: cbPixelDest = 2; break;
99 case BMF_24BPP: cbPixelDest = 3; break;
100 case BMF_32BPP: cbPixelDest = 4; break;
101 default: cbPixelDest = 0;
102 }
103
104 switch (psoSrc->iBitmapFormat)
105 {
106 case BMF_16BPP: cbPixelSrc = 2; break;
107 case BMF_24BPP: cbPixelSrc = 3; break;
108 case BMF_32BPP: cbPixelSrc = 4; break;
109 default: cbPixelSrc = 0;
110 }
111
112 if (cbPixelDest == 0 || cbPixelSrc == 0)
113 {
114 WARN(("unsupported pixel format src %d dst %d", psoDest->iBitmapFormat, psoSrc->iBitmapFormat));
115 return TRUE;
116 }
117
118 rclDest = *prclDest;
119
120 vrdpAdjustRect(psoDest, &rclDest);
121
122 pptlSrc->x += rclDest.left - prclDest->left;
123 pptlSrc->y += rclDest.top - prclDest->top;
124
125 *prclDest = rclDest;
126
127 if ( rclDest.right == rclDest.left
128 || rclDest.bottom == rclDest.top)
129 {
130 WARN(("empty dest rect: %d-%d, %d-%d", rclDest.left, rclDest.right, rclDest.top, rclDest.bottom));
131 return FALSE;
132 }
133
134 rclSrc.left = pptlSrc->x;
135 rclSrc.top = pptlSrc->y;
136 rclSrc.right = pptlSrc->x + (rclDest.right - rclDest.left);
137 rclSrc.bottom = pptlSrc->y + (rclDest.bottom - rclDest.top);
138 vrdpAdjustRect (psoSrc, &rclSrc);
139
140 if ( rclSrc.right == rclSrc.left
141 || rclSrc.bottom == rclSrc.top)
142 {
143 prclDest->right = prclDest->left;
144 prclDest->bottom = prclDest->top;
145
146 WARN(("empty src rect: %d-%d, %d-%d", rclSrc.left, rclSrc.right, rclSrc.top, rclSrc.bottom));
147 return FALSE;
148 }
149
150 Assert(pptlSrc->x == rclSrc.left);
151 Assert(pptlSrc->y == rclSrc.top);
152
153 /*
154 * Compare the content of the screen surface (psoDest) with the source surface (psoSrc).
155 * Update the prclDest with the rectangle that will be actually changed after
156 * copying the source bits to the screen.
157 */
158 pu8Src = (BYTE *)psoSrc->pvScan0 + psoSrc->lDelta * pptlSrc->y + cbPixelSrc * pptlSrc->x;
159 pu8Dest = (BYTE *)psoDest->pvScan0 + psoDest->lDelta * prclDest->top + cbPixelDest * prclDest->left;
160
161 /* Use the rclDest as the bounding rectangle for the changed area. */
162 rclDest.left = prclDest->right; /* +inf */
163 rclDest.right = prclDest->left; /* -inf */
164 rclDest.top = prclDest->bottom; /* +inf */
165 rclDest.bottom = prclDest->top; /* -inf */
166
167 fTopNonEqualFound = 0;
168 yTopmost = prclDest->top; /* inclusive */
169 yBottommost = prclDest->top - 1; /* inclusive */
170
171 for (y = prclDest->top; y < prclDest->bottom; y++)
172 {
173 int fLeftNonEqualFound = 0;
174
175 /* Init to an empty line. */
176 int xLeftmost = prclDest->left; /* inclusive */
177 int xRightmost = prclDest->left - 1; /* inclusive */
178
179 BYTE *pu8SrcLine = pu8Src;
180 BYTE *pu8DestLine = pu8Dest;
181
182 for (x = prclDest->left; x < prclDest->right; x++)
183 {
184 int fEqualPixels;
185
186 if (cbPixelSrc == cbPixelDest)
187 {
188 fEqualPixels = (memcmp (pu8SrcLine, pu8DestLine, cbPixelDest) == 0);
189 }
190 else
191 {
192 /* Convert larger pixel to the smaller pixel format. */
193 ULONG ulConvertedPixel;
194 if (cbPixelSrc > cbPixelDest)
195 {
196 /* Convert the source pixel to the destination pixel format. */
197 ulConvertedPixel = vbvaConvertPixel ( /*from*/ pu8SrcLine, cbPixelSrc,
198 /*to*/ cbPixelDest);
199 fEqualPixels = (memcmp (&ulConvertedPixel, pu8DestLine, cbPixelDest) == 0);
200 }
201 else
202 {
203 /* Convert the destination pixel to the source pixel format. */
204 ulConvertedPixel = vbvaConvertPixel ( /*from*/ pu8DestLine, cbPixelDest,
205 /*to*/ cbPixelSrc);
206 fEqualPixels = (memcmp (&ulConvertedPixel, pu8SrcLine, cbPixelSrc) == 0);
207 }
208 }
209
210 if (fEqualPixels)
211 {
212 /* Equal pixels. */
213 if (!fLeftNonEqualFound)
214 {
215 xLeftmost = x;
216 }
217 }
218 else
219 {
220 fLeftNonEqualFound = 1;
221 xRightmost = x;
222 }
223
224 pu8SrcLine += cbPixelSrc;
225 pu8DestLine += cbPixelDest;
226 }
227
228 /* min */
229 if (rclDest.left > xLeftmost)
230 {
231 rclDest.left = xLeftmost;
232 }
233
234 /* max */
235 if (rclDest.right < xRightmost)
236 {
237 rclDest.right = xRightmost;
238 }
239
240 if (xLeftmost > xRightmost) /* xRightmost is inclusive, so '>', not '>='. */
241 {
242 /* Empty line. */
243 if (!fTopNonEqualFound)
244 {
245 yTopmost = y;
246 }
247 }
248 else
249 {
250 fTopNonEqualFound = 1;
251 yBottommost = y;
252 }
253
254 pu8Src += psoSrc->lDelta;
255 pu8Dest += psoDest->lDelta;
256 }
257
258 /* min */
259 if (rclDest.top > yTopmost)
260 {
261 rclDest.top = yTopmost;
262 }
263
264 /* max */
265 if (rclDest.bottom < yBottommost)
266 {
267 rclDest.bottom = yBottommost;
268 }
269
270 /* rclDest was calculated with right-bottom inclusive.
271 * The following checks and the caller require exclusive coords.
272 */
273 rclDest.right++;
274 rclDest.bottom++;
275
276 LOG(("new dest %d,%d %dx%d from %d,%d",
277 rclDest.left, rclDest.top, rclDest.right - rclDest.left, rclDest.bottom - rclDest.top,
278 pptlSrc->x, pptlSrc->y));
279
280 /* Update the rectangle with the changed area. */
281 if ( rclDest.left >= rclDest.right
282 || rclDest.top >= rclDest.bottom)
283 {
284 /* Empty rect. */
285 LOG(("empty"));
286 prclDest->right = prclDest->left;
287 prclDest->bottom = prclDest->top;
288 return FALSE;
289 }
290
291 LOG(("not empty"));
292
293 pptlSrc->x += rclDest.left - prclDest->left;
294 pptlSrc->y += rclDest.top - prclDest->top;
295
296 *prclDest = rclDest;
297
298 return TRUE;
299}
300#endif /* VBOX_VBVA_ADJUST_RECT */
301
302int VBoxDispVBVAInit(PVBOXDISPDEV pDev)
303{
304 int rc;
305 DWORD dwrc;
306 ULONG cbReturned;
307 QUERYHGSMIRESULT info;
308 HGSMIQUERYCALLBACKS callbacks;
309 HGSMIQUERYCPORTPROCS portProcs;
310 LOGF_ENTER();
311
312 /* Check if HGSMI is supported and obtain necessary info */
313 rc = VBoxDispMPQueryHGSMIInfo(pDev->hDriver, &info);
314 if (RT_SUCCESS(rc))
315 {
316 rc = VBoxDispMPQueryHGSMICallbacks(pDev->hDriver, &callbacks);
317 if (RT_SUCCESS(rc))
318 {
319 rc = VBoxDispMPHGSMIQueryPortProcs(pDev->hDriver, &portProcs);
320 }
321 }
322 if (RT_SUCCESS(rc))
323 {
324 pDev->hgsmi.bSupported = TRUE;
325
326 pDev->hgsmi.mp = callbacks;
327 pDev->vpAPI = portProcs;
328 }
329
330 if (pDev->hgsmi.bSupported)
331 {
332 HGSMIHANDLERENABLE HandlerReg;
333
334 memset(&HandlerReg, 0, sizeof(HandlerReg));
335 HandlerReg.u8Channel = HGSMI_CH_VBVA;
336 dwrc = EngDeviceIoControl(pDev->hDriver, IOCTL_VIDEO_HGSMI_HANDLER_ENABLE, &HandlerReg, sizeof(HandlerReg),
337 0, NULL, &cbReturned);
338 VBOX_WARN_WINERR(dwrc);
339
340#ifdef VBOX_WITH_VIDEOHWACCEL
341 if (NO_ERROR == dwrc)
342 {
343 VBoxDispVHWAInit(pDev);
344 }
345#endif
346 }
347
348 /* Check if we have enough VRAM and update layout info.
349 * 0=Framebuffer(fixed)->DDrawHeap(all left vram)->VBVABuffer(64k..cbFramebuffer)->DisplayInfo(fixed)->=EndOfVRAM
350 */
351 if (pDev->hgsmi.bSupported)
352 {
353 ULONG cbAvailable;
354 VBOXDISPVRAMLAYOUT *vram = &pDev->layout;
355
356 pDev->iDevice = info.iDevice;
357
358 vram->cbVRAM = pDev->memInfo.VideoRamLength;
359
360 vram->offFramebuffer = 0;
361 vram->cbFramebuffer = RT_ALIGN_32(pDev->memInfo.FrameBufferLength, 0x1000);
362 cbAvailable = vram->cbVRAM - vram->cbFramebuffer;
363
364 if (cbAvailable <= info.u32DisplayInfoSize)
365 {
366 pDev->hgsmi.bSupported = FALSE;
367 }
368 else
369 {
370 vram->offDisplayInfo = vram->cbVRAM - info.u32DisplayInfoSize;
371 vram->cbDisplayInfo = info.u32DisplayInfoSize;
372 cbAvailable -= vram->cbDisplayInfo;
373
374 for (vram->cbVBVABuffer = vram->cbFramebuffer;
375 vram->cbVBVABuffer >= info.u32MinVBVABufferSize;
376 vram->cbVBVABuffer /= 2)
377 {
378 if (vram->cbVBVABuffer < cbAvailable)
379 {
380 break;
381 }
382 }
383
384 if (vram->cbVBVABuffer >= cbAvailable)
385 {
386 pDev->hgsmi.bSupported = FALSE;
387 }
388 else
389 {
390 vram->offDDrawHeap = vram->offFramebuffer + vram->cbFramebuffer;
391
392 cbAvailable -= vram->cbVBVABuffer;
393 vram->cbDDrawHeap = cbAvailable;
394
395 vram->offVBVABuffer = vram->offDDrawHeap + vram->cbDDrawHeap;
396 }
397 }
398 }
399
400 /* Setup HGSMI heap in the display information area.
401 * The area has some space reserved for HGSMI event flags in the beginning.
402 */
403 if (pDev->hgsmi.bSupported)
404 {
405 LOG(("offBase=%#x", info.areaDisplay.offBase));
406
407 rc = HGSMIHeapSetup(&pDev->hgsmi.ctx.heapCtx,
408 (uint8_t *)pDev->memInfo.VideoRamBase+pDev->layout.offDisplayInfo+sizeof(HGSMIHOSTFLAGS),
409 pDev->layout.cbDisplayInfo-sizeof(HGSMIHOSTFLAGS),
410 info.areaDisplay.offBase+pDev->layout.offDisplayInfo+sizeof(HGSMIHOSTFLAGS),
411 false /*fOffsetBased*/);
412
413 if (RT_SUCCESS(rc))
414 {
415 pDev->hgsmi.ctx.port = info.IOPortGuestCommand;
416 }
417 else
418 {
419 VBOX_WARNRC(rc);
420 pDev->hgsmi.bSupported = FALSE;
421 }
422 }
423
424 /* If we don't have HGSMI or doesn't have enough VRAM, setup layout without VBVA buffer and display info */
425 if (!pDev->hgsmi.bSupported)
426 {
427 VBOXDISPVRAMLAYOUT *vram = &pDev->layout;
428
429 pDev->iDevice = 0;
430
431 /* Setup a layout without both the VBVA buffer and the display information. */
432 vram->cbVRAM = pDev->memInfo.VideoRamLength;
433
434 vram->offFramebuffer = 0;
435 vram->cbFramebuffer = RT_ALIGN_32(pDev->memInfo.FrameBufferLength, 0x1000);
436
437 vram->offDDrawHeap = vram->offFramebuffer + vram->cbFramebuffer;
438 vram->cbDDrawHeap = vram->cbVRAM - vram->offDDrawHeap;
439
440 vram->offVBVABuffer = vram->offDDrawHeap + vram->cbDDrawHeap;
441 vram->cbVBVABuffer = 0;
442
443 vram->offDisplayInfo = vram->offVBVABuffer + vram->cbVBVABuffer;
444 vram->cbDisplayInfo = 0;
445 }
446
447 /* Update buffer layout in VBVA context info */
448 VBoxVBVASetupBufferContext(&pDev->vbvaCtx, pDev->layout.offVBVABuffer, pDev->layout.cbVBVABuffer);
449
450 LOG(("\n"
451 " cbVRAM=%#X\n"
452 " offFramebuffer=%#X cbFramebuffer=%#X\n"
453 " offDDrawHeap=%#X cbDDrawHeap=%#X\n"
454 " offVBVABuffer=%#X cbVBVABuffer=%#X\n"
455 " offDisplayInfo=%#X cbDisplayInfo=%#X\n",
456 pDev->layout.cbVRAM,
457 pDev->layout.offFramebuffer, pDev->layout.cbFramebuffer,
458 pDev->layout.offDDrawHeap, pDev->layout.cbDDrawHeap,
459 pDev->layout.offVBVABuffer, pDev->layout.cbVBVABuffer,
460 pDev->layout.offDisplayInfo, pDev->layout.cbDisplayInfo
461 ));
462
463 LOGF_LEAVE();
464 return VINF_SUCCESS;
465}
466
467void VBoxDispVBVAHostCommandComplete(PVBOXDISPDEV pDev, VBVAHOSTCMD *pCmd)
468{
469 pDev->hgsmi.mp.pfnCompletionHandler(pDev->hgsmi.mp.hContext, pCmd);
470}
471
472void vbvaReportDirtyRect(PVBOXDISPDEV pDev, RECTL *pRectOrig)
473{
474 if (pDev && pRectOrig)
475 {
476
477 LOG(("dirty rect: left %d, top: %d, width: %d, height: %d",
478 pRectOrig->left, pRectOrig->top,
479 pRectOrig->right - pRectOrig->left, pRectOrig->bottom - pRectOrig->top));
480
481 VBVACMDHDR hdr;
482 RECTL rect;
483
484 /* Ensure correct order. */
485 if (pRectOrig->left <= pRectOrig->right)
486 {
487 rect.left = pRectOrig->left;
488 rect.right = pRectOrig->right;
489 }
490 else
491 {
492 rect.left = pRectOrig->right;
493 rect.right = pRectOrig->left;
494 }
495
496 if (pRectOrig->top <= pRectOrig->bottom)
497 {
498 rect.top = pRectOrig->top;
499 rect.bottom = pRectOrig->bottom;
500 }
501 else
502 {
503 rect.top = pRectOrig->bottom;
504 rect.bottom = pRectOrig->top;
505 }
506
507 /* Clip the rectangle. */
508 rect.left = RT_CLAMP(rect.left, 0, (LONG)pDev->mode.ulWidth);
509 rect.top = RT_CLAMP(rect.top, 0, (LONG)pDev->mode.ulHeight);
510 rect.right = RT_CLAMP(rect.right, 0, (LONG)pDev->mode.ulWidth);
511 rect.bottom = RT_CLAMP(rect.bottom, 0, (LONG)pDev->mode.ulHeight);
512
513 /* If the rectangle is empty, still report it. */
514 if (rect.right < rect.left)
515 {
516 rect.right = rect.left;
517 }
518 if (rect.bottom < rect.top)
519 {
520 rect.bottom = rect.top;
521 }
522
523 hdr.x = (int16_t)(rect.left + pDev->orgDev.x);
524 hdr.y = (int16_t)(rect.top + pDev->orgDev.y);
525 hdr.w = (uint16_t)(rect.right - rect.left);
526 hdr.h = (uint16_t)(rect.bottom - rect.top);
527
528 VBoxVBVAWrite(&pDev->vbvaCtx, &pDev->hgsmi.ctx, &hdr, sizeof(hdr));
529 }
530}
531
532static void vbvaReportDirtyPath(PVBOXDISPDEV pDev, PATHOBJ *ppo)
533{
534 RECTFX rcfxBounds;
535 RECTL rclBounds;
536
537 PATHOBJ_vGetBounds(ppo, &rcfxBounds);
538
539 rclBounds.left = FXTOLFLOOR(rcfxBounds.xLeft);
540 rclBounds.right = FXTOLCEILING(rcfxBounds.xRight);
541 rclBounds.top = FXTOLFLOOR(rcfxBounds.yTop);
542 rclBounds.bottom = FXTOLCEILING(rcfxBounds.yBottom);
543
544 vbvaReportDirtyRect(pDev, &rclBounds);
545}
546
547static void vbvaReportDirtyClip(PVBOXDISPDEV pDev, CLIPOBJ *pco, RECTL *prcl)
548{
549 if (prcl)
550 {
551 vbvaReportDirtyRect(pDev, prcl);
552 }
553 else if (pco)
554 {
555 vbvaReportDirtyRect(pDev, &pco->rclBounds);
556 }
557}
558
559/*
560 * VBVA driver functions.
561 */
562
563void vbvaDrvLineTo(SURFOBJ *pso, CLIPOBJ *pco, BRUSHOBJ *pbo,
564 LONG x1, LONG y1, LONG x2, LONG y2, RECTL *prclBounds, MIX mix)
565{
566 PVBOXDISPDEV pDev = (PVBOXDISPDEV)pso->dhpdev;
567 vbvaReportDirtyClip(pDev, pco, prclBounds);
568}
569
570void vbvaDrvStrokePath(SURFOBJ *pso, PATHOBJ *ppo, CLIPOBJ *pco, XFORMOBJ *pxo,
571 BRUSHOBJ *pbo, POINTL *pptlBrushOrg, LINEATTRS *plineattrs, MIX mix)
572{
573 PVBOXDISPDEV pDev = (PVBOXDISPDEV)pso->dhpdev;
574 vbvaReportDirtyPath(pDev, ppo);
575}
576
577void vbvaDrvFillPath(SURFOBJ *pso, PATHOBJ *ppo, CLIPOBJ *pco, BRUSHOBJ *pbo, POINTL *pptlBrushOrg,
578 MIX mix, FLONG flOptions)
579{
580 PVBOXDISPDEV pDev = (PVBOXDISPDEV)pso->dhpdev;
581 vbvaReportDirtyPath(pDev, ppo);
582}
583
584void vbvaDrvPaint(SURFOBJ *pso, CLIPOBJ *pco, BRUSHOBJ *pbo, POINTL *pptlBrushOrg, MIX mix)
585{
586 PVBOXDISPDEV pDev = (PVBOXDISPDEV)pso->dhpdev;
587 vbvaReportDirtyClip(pDev, pco, NULL);
588}
589
590void vbvaDrvTextOut(SURFOBJ *pso, STROBJ *pstro, FONTOBJ *pfo, CLIPOBJ *pco,
591 RECTL *prclExtra, RECTL *prclOpaque, BRUSHOBJ *pboFore,
592 BRUSHOBJ *pboOpaque, POINTL *pptlOrg, MIX mix)
593{
594 PVBOXDISPDEV pDev = (PVBOXDISPDEV)pso->dhpdev;
595 vbvaReportDirtyClip(pDev, pco, prclOpaque? prclOpaque: &pstro->rclBkGround);
596}
597
598void vbvaDrvSaveScreenBits(SURFOBJ *pso, ULONG iMode, ULONG_PTR ident, RECTL *prcl)
599{
600 PVBOXDISPDEV pDev = (PVBOXDISPDEV)pso->dhpdev;
601
602 Assert(iMode == SS_RESTORE || iMode == SS_SAVE);
603 vbvaReportDirtyRect(pDev, prcl);
604}
605
606void vbvaDrvBitBlt(SURFOBJ *psoTrg, SURFOBJ *psoSrc, SURFOBJ *psoMask, CLIPOBJ *pco, XLATEOBJ *pxlo,
607 RECTL *prclTrg, POINTL *pptlSrc, POINTL *pptlMask, BRUSHOBJ *pbo, POINTL *pptlBrush,
608 ROP4 rop4)
609{
610 PVBOXDISPDEV pDev = (PVBOXDISPDEV)psoTrg->dhpdev;
611 vbvaReportDirtyClip(pDev, pco, prclTrg);
612}
613
614void vbvaDrvStretchBlt(SURFOBJ *psoDest, SURFOBJ *psoSrc, SURFOBJ *psoMask, CLIPOBJ *pco, XLATEOBJ *pxlo,
615 COLORADJUSTMENT *pca, POINTL *pptlHTOrg, RECTL *prclDest, RECTL *prclSrc,
616 POINTL *pptlMask, ULONG iMode)
617{
618 PVBOXDISPDEV pDev = (PVBOXDISPDEV)psoDest->dhpdev;
619 vbvaReportDirtyClip(pDev, pco, prclDest);
620}
621
622void vbvaDrvCopyBits(SURFOBJ *psoDest, SURFOBJ *psoSrc, CLIPOBJ *pco, XLATEOBJ *pxlo,
623 RECTL *prclDest, POINTL *pptlSrc)
624{
625 PVBOXDISPDEV pDev = (PVBOXDISPDEV)psoDest->dhpdev;
626 vbvaReportDirtyClip(pDev, pco, prclDest);
627}
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