VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/Graphics/Video/disp/xpdm/VBoxDispDriverDDraw.cpp

Last change on this file was 106061, checked in by vboxsync, 4 months ago

Copyright year updates by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 8.7 KB
Line 
1/* $Id: VBoxDispDriverDDraw.cpp 106061 2024-09-16 14:03:52Z vboxsync $ */
2/** @file
3 * VBox XPDM Display driver interface functions related to DirectDraw
4 */
5
6/*
7 * Copyright (C) 2011-2024 Oracle and/or its affiliates.
8 *
9 * This file is part of VirtualBox base platform packages, as
10 * available from https://www.virtualbox.org.
11 *
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation, in version 3 of the
15 * License.
16 *
17 * This program is distributed in the hope that it will be useful, but
18 * WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, see <https://www.gnu.org/licenses>.
24 *
25 * SPDX-License-Identifier: GPL-3.0-only
26 */
27
28#include "VBoxDisp.h"
29#include "VBoxDispDDraw.h"
30
31static void VBoxDispGetDDHalInfo(PVBOXDISPDEV pDev, DD_HALINFO *pHalInfo)
32{
33 memset(pHalInfo, 0, sizeof(DD_HALINFO));
34 pHalInfo->dwSize = sizeof(DD_HALINFO);
35
36 /* memory info */
37
38 pHalInfo->vmiData.fpPrimary = pDev->layout.offFramebuffer;
39 /*pHalInfo->vmiData.dwFlags - unused*/
40 pHalInfo->vmiData.dwDisplayWidth = pDev->mode.ulWidth;
41 pHalInfo->vmiData.dwDisplayHeight = pDev->mode.ulHeight;
42 pHalInfo->vmiData.lDisplayPitch = pDev->mode.lScanlineStride;
43
44 pHalInfo->vmiData.ddpfDisplay.dwSize = sizeof(DDPIXELFORMAT);
45 pHalInfo->vmiData.ddpfDisplay.dwFlags = DDPF_RGB;
46 if (pDev->surface.ulFormat == BMF_8BPP)
47 {
48 pHalInfo->vmiData.ddpfDisplay.dwFlags |= DDPF_PALETTEINDEXED8;
49 }
50 pHalInfo->vmiData.ddpfDisplay.dwRGBBitCount = pDev->mode.ulBitsPerPel;
51 pHalInfo->vmiData.ddpfDisplay.dwRBitMask = pDev->mode.flMaskR;
52 pHalInfo->vmiData.ddpfDisplay.dwGBitMask = pDev->mode.flMaskG;
53 pHalInfo->vmiData.ddpfDisplay.dwBBitMask = pDev->mode.flMaskB;
54
55 pHalInfo->vmiData.dwOffscreenAlign = 4;
56 pHalInfo->vmiData.dwTextureAlign = 4;
57 pHalInfo->vmiData.dwZBufferAlign = 4;
58 pHalInfo->vmiData.dwOverlayAlign = 4;
59
60 pHalInfo->vmiData.pvPrimary = pDev->memInfo.FrameBufferBase;
61
62 /* caps */
63
64 pHalInfo->ddCaps.dwSize = sizeof(DDNTCORECAPS);
65 pHalInfo->ddCaps.dwCaps2 = DDCAPS2_WIDESURFACES;
66 pHalInfo->ddCaps.dwVidMemTotal = pDev->layout.cbDDrawHeap;
67 pHalInfo->ddCaps.dwVidMemFree = pDev->layout.cbDDrawHeap;
68 pHalInfo->ddCaps.ddsCaps.dwCaps = DDSCAPS_PRIMARYSURFACE;
69
70}
71
72/* Called to get supported DirectDraw caps */
73BOOL APIENTRY
74VBoxDispDrvGetDirectDrawInfo(DHPDEV dhpdev, DD_HALINFO *pHalInfo, DWORD *pdwNumHeaps,
75 VIDEOMEMORY *pvmList, DWORD *pdwNumFourCCCodes, DWORD *pdwFourCC)
76{
77 PVBOXDISPDEV pDev = (PVBOXDISPDEV)dhpdev;
78 LOGF_ENTER();
79
80 VBoxDispGetDDHalInfo(pDev, pHalInfo);
81
82#ifdef VBOX_WITH_VIDEOHWACCEL
83 int rc;
84
85 if (!pvmList && !pdwFourCC) /* first call */
86 {
87 rc = VBoxDispVHWAInitHostInfo1(pDev);
88 VBOX_WARNRC_NOBP(rc);
89 }
90
91 if (pDev->vhwa.bEnabled)
92 {
93 rc = VBoxDispVHWAUpdateDDHalInfo(pDev, pHalInfo);
94 VBOX_WARNRC(rc);
95
96 pDev->vhwa.bEnabled = RT_SUCCESS(rc);
97 }
98#else
99 RT_NOREF(pdwFourCC);
100#endif
101
102 /* we could only have 1 heap, so it's not really a list */
103 if (pvmList && pDev->layout.cbDDrawHeap>0)
104 {
105 pvmList->dwFlags = VIDMEM_ISLINEAR;
106 pvmList->fpStart = pDev->layout.offDDrawHeap;
107 pvmList->fpEnd = pDev->layout.offDDrawHeap + pDev->layout.cbDDrawHeap - 1;
108#ifdef VBOX_WITH_VIDEOHWACCEL
109 if (pDev->vhwa.bEnabled)
110 {
111 pvmList->ddsCaps.dwCaps = 0;
112 }
113 else
114#endif
115 {
116 pvmList->ddsCaps.dwCaps = DDSCAPS_OFFSCREENPLAIN;
117 }
118 pvmList->ddsCapsAlt.dwCaps = 0;
119
120 }
121
122 /* Always report number of heaps and supported FourCC's*/
123 *pdwNumHeaps = (pDev->layout.cbDDrawHeap>0) ? 1:0;
124
125#ifndef VBOX_WITH_VIDEOHWACCEL
126 *pdwNumFourCCCodes = 0;
127#else
128 if (pDev->vhwa.bEnabled)
129 {
130 *pdwNumFourCCCodes = pDev->vhwa.numFourCC;
131 if (pdwFourCC && pDev->vhwa.numFourCC)
132 {
133 rc = VBoxDispVHWAInitHostInfo2(pDev, pdwFourCC);
134 VBOX_WARNRC(rc);
135
136 if (RT_FAILURE(rc))
137 {
138 *pdwNumFourCCCodes = 0;
139 pDev->vhwa.numFourCC = 0;
140 }
141 }
142
143 pHalInfo->GetDriverInfo = VBoxDispDDGetDriverInfo;
144 pHalInfo->dwFlags |= DDHALINFO_GETDRIVERINFOSET;
145 }
146#endif
147
148 LOGF_LEAVE();
149 return TRUE;
150}
151
152BOOL APIENTRY
153VBoxDispDrvEnableDirectDraw(DHPDEV dhpdev, DD_CALLBACKS *pCallBacks, DD_SURFACECALLBACKS *pSurfaceCallBacks,
154 DD_PALETTECALLBACKS *pPaletteCallBacks)
155{
156 LOGF_ENTER();
157
158 pCallBacks->dwSize = sizeof(DD_CALLBACKS);
159 pCallBacks->CreateSurface = VBoxDispDDCreateSurface;
160 pCallBacks->CanCreateSurface = VBoxDispDDCanCreateSurface;
161 pCallBacks->MapMemory = VBoxDispDDMapMemory;
162 pCallBacks->dwFlags = DDHAL_CB32_CREATESURFACE|DDHAL_CB32_CANCREATESURFACE|DDHAL_CB32_MAPMEMORY;
163
164 pSurfaceCallBacks->dwSize = sizeof(DD_SURFACECALLBACKS);
165 pSurfaceCallBacks->Lock = VBoxDispDDLock;
166 pSurfaceCallBacks->Unlock = VBoxDispDDUnlock;
167 pSurfaceCallBacks->dwFlags = DDHAL_SURFCB32_LOCK|DDHAL_SURFCB32_UNLOCK;
168
169 pPaletteCallBacks->dwSize = sizeof(DD_PALETTECALLBACKS);
170 pPaletteCallBacks->dwFlags = 0;
171
172#ifdef VBOX_WITH_VIDEOHWACCEL
173 PVBOXDISPDEV pDev = (PVBOXDISPDEV)dhpdev;
174
175 if (pDev->vhwa.bEnabled)
176 {
177 pSurfaceCallBacks->DestroySurface = VBoxDispDDDestroySurface;
178 pSurfaceCallBacks->Flip = VBoxDispDDFlip;
179 pSurfaceCallBacks->GetFlipStatus = VBoxDispDDGetFlipStatus;
180 pSurfaceCallBacks->Blt = VBoxDispDDBlt;
181 pSurfaceCallBacks->GetBltStatus = VBoxDispDDGetBltStatus;
182 pSurfaceCallBacks->SetColorKey = VBoxDispDDSetColorKey;
183 pSurfaceCallBacks->dwFlags |= DDHAL_SURFCB32_DESTROYSURFACE|
184 DDHAL_SURFCB32_FLIP|DDHAL_SURFCB32_GETFLIPSTATUS|
185 DDHAL_SURFCB32_BLT|DDHAL_SURFCB32_GETBLTSTATUS|
186 DDHAL_SURFCB32_SETCOLORKEY;
187
188 if(pDev->vhwa.caps & VBOXVHWA_CAPS_OVERLAY)
189 {
190 pSurfaceCallBacks->UpdateOverlay = VBoxDispDDUpdateOverlay;
191 pSurfaceCallBacks->SetOverlayPosition = VBoxDispDDSetOverlayPosition;
192 pSurfaceCallBacks->dwFlags |= DDHAL_SURFCB32_UPDATEOVERLAY|DDHAL_SURFCB32_SETOVERLAYPOSITION;
193 }
194 }
195#else
196 RT_NOREF(dhpdev);
197#endif
198
199 LOGF_LEAVE();
200 return TRUE;
201}
202
203VOID APIENTRY VBoxDispDrvDisableDirectDraw(DHPDEV dhpdev)
204{
205 RT_NOREF(dhpdev);
206 LOGF_ENTER();
207 LOGF_LEAVE();
208 return;
209}
210
211HBITMAP APIENTRY VBoxDispDrvDeriveSurface(DD_DIRECTDRAW_GLOBAL *pDirectDraw, DD_SURFACE_LOCAL *pSurface)
212{
213 PVBOXDISPDEV pDev = (PVBOXDISPDEV)pDirectDraw->dhpdev;
214 LOGF_ENTER();
215
216 if (pSurface->ddsCaps.dwCaps & DDSCAPS_NONLOCALVIDMEM)
217 {
218 WARN(("Can't derive surface DDSCAPS_NONLOCALVIDMEM"));
219 return NULL;
220 }
221
222 if (pSurface->lpSurfMore->ddsCapsEx.dwCaps2 & DDSCAPS2_TEXTUREMANAGE)
223 {
224 WARN(("Can't derive surface DDSCAPS2_TEXTUREMANAGE"));
225 return NULL;
226 }
227
228 if (pSurface->lpGbl->ddpfSurface.dwRGBBitCount != pDev->mode.ulBitsPerPel)
229 {
230 WARN(("Can't derive surface with different bpp"));
231 return NULL;
232 }
233
234 Assert(pDev->surface.hSurface);
235
236 /* Create GDI managed bitmap, which resides in our DDraw heap memory */
237 HBITMAP hBitmap;
238 SIZEL size;
239
240 size.cx = pDev->mode.ulWidth;
241 size.cy = pDev->mode.ulHeight;
242
243 hBitmap = EngCreateBitmap(size, pSurface->lpGbl->lPitch, pDev->surface.ulFormat,
244 pDev->mode.lScanlineStride>0 ? BMF_TOPDOWN:0,
245 (PBYTE)pDev->memInfo.VideoRamBase + pSurface->lpGbl->fpVidMem);
246
247 if (!hBitmap)
248 {
249 WARN(("EngCreateBitmap failed"));
250 return 0;
251 }
252
253 if (pSurface->lpGbl->fpVidMem == 0)
254 {
255 /* Screen surface, mark it so it will be recognized by the driver.
256 * so the driver will be called on any operations on the surface
257 * (required for VBVA and VRDP).
258 */
259 SURFOBJ *pso;
260
261 if (!EngAssociateSurface((HSURF)hBitmap, pDev->hDevGDI, pDev->flDrawingHooks))
262 {
263 WARN(("EngAssociateSurface failed"));
264 EngDeleteSurface((HSURF)hBitmap);
265 return NULL;
266 }
267
268 pso = EngLockSurface((HSURF)hBitmap);
269 if (!pso)
270 {
271 WARN(("EngLockSurface failed"));
272 EngDeleteSurface((HSURF)hBitmap);
273 return NULL;
274 }
275
276 pso->dhpdev = (DHPDEV)pDev;
277 EngUnlockSurface(pso);
278 }
279
280 LOGF_LEAVE();
281 return hBitmap;
282}
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