VirtualBox

source: vbox/trunk/src/VBox/Devices/Graphics/DevVGA-SVGA3d-shared.h@ 54532

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

VMSVGA3d: Cleaning up some of the dual-profile mess, switched back to using the 2.1 profile by default.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 35.3 KB
Line 
1/** @file
2 * VMware SVGA device -- 3D part
3 */
4/*
5 * Copyright (C) 2013 Oracle Corporation
6 *
7 * This file is part of VirtualBox Open Source Edition (OSE), as
8 * available from http://www.virtualbox.org. This file is free software;
9 * you can redistribute it and/or modify it under the terms of the GNU
10 * General Public License (GPL) as published by the Free Software
11 * Foundation, in version 2 as it comes in the "COPYING" file of the
12 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
13 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
14 */
15#ifndef __DEVVMWARE3D_STATE_H__
16#define __DEVVMWARE3D_STATE_H__
17
18int vmsvga3dLoadExec(PVGASTATE pThis, PSSMHANDLE pSSM, uint32_t uVersion, uint32_t uPass)
19{
20 PVMSVGA3DSTATE pState = (PVMSVGA3DSTATE)pThis->svga.p3dState;
21 AssertReturn(pState, VERR_NO_MEMORY);
22 int rc;
23 uint32_t cContexts, cSurfaces;
24
25 /* Must initialize now as the recreation calls below rely on an initialized 3d subsystem. */
26 vmsvga3dPowerOn(pThis);
27
28 /* Get the generic 3d state first. */
29 rc = SSMR3GetStructEx(pSSM, pState, sizeof(*pState), 0, g_aVMSVGA3DSTATEFields, NULL);
30 AssertRCReturn(rc, rc);
31
32 cContexts = pState->cContexts;
33 cSurfaces = pState->cSurfaces;
34 pState->cContexts = 0;
35 pState->cSurfaces = 0;
36
37 /* Fetch all active contexts. */
38 for (uint32_t i = 0; i < cContexts; i++)
39 {
40 PVMSVGA3DCONTEXT pContext;
41 uint32_t cid;
42
43 /* Get the context id */
44 rc = SSMR3GetU32(pSSM, &cid);
45 AssertRCReturn(rc, rc);
46
47 if (cid != SVGA3D_INVALID_ID)
48 {
49 uint32_t cPixelShaderConst, cVertexShaderConst, cPixelShaders, cVertexShaders;
50
51 rc = vmsvga3dContextDefine(pThis, cid, false /*fOtherProfile*/);
52 AssertRCReturn(rc, rc);
53
54 pContext = &pState->paContext[i];
55 AssertReturn(pContext->id == cid, VERR_INTERNAL_ERROR);
56
57 rc = SSMR3GetStructEx(pSSM, pContext, sizeof(*pContext), 0, g_aVMSVGA3DCONTEXTFields, NULL);
58 AssertRCReturn(rc, rc);
59
60 cPixelShaders = pContext->cPixelShaders;
61 cVertexShaders = pContext->cVertexShaders;
62 cPixelShaderConst = pContext->state.cPixelShaderConst;
63 cVertexShaderConst = pContext->state.cVertexShaderConst;
64 pContext->cPixelShaders = 0;
65 pContext->cVertexShaders = 0;
66 pContext->state.cPixelShaderConst = 0;
67 pContext->state.cVertexShaderConst = 0;
68
69 /* Fetch all pixel shaders. */
70 for (uint32_t j = 0; j < cPixelShaders; j++)
71 {
72 VMSVGA3DSHADER shader;
73 uint32_t shid;
74
75 /* Fetch the id first. */
76 rc = SSMR3GetU32(pSSM, &shid);
77 AssertRCReturn(rc, rc);
78
79 if (shid != SVGA3D_INVALID_ID)
80 {
81 uint32_t *pData;
82
83 /* Fetch a copy of the shader struct. */
84 rc = SSMR3GetStructEx(pSSM, &shader, sizeof(shader), 0, g_aVMSVGA3DSHADERFields, NULL);
85 AssertRCReturn(rc, rc);
86
87 pData = (uint32_t *)RTMemAlloc(shader.cbData);
88 AssertReturn(pData, VERR_NO_MEMORY);
89
90 rc = SSMR3GetMem(pSSM, pData, shader.cbData);
91 AssertRCReturn(rc, rc);
92
93 rc = vmsvga3dShaderDefine(pThis, cid, shid, shader.type, shader.cbData, pData);
94 AssertRCReturn(rc, rc);
95
96 RTMemFree(pData);
97 }
98 }
99
100 /* Fetch all vertex shaders. */
101 for (uint32_t j = 0; j < cVertexShaders; j++)
102 {
103 VMSVGA3DSHADER shader;
104 uint32_t shid;
105
106 /* Fetch the id first. */
107 rc = SSMR3GetU32(pSSM, &shid);
108 AssertRCReturn(rc, rc);
109
110 if (shid != SVGA3D_INVALID_ID)
111 {
112 uint32_t *pData;
113
114 /* Fetch a copy of the shader struct. */
115 rc = SSMR3GetStructEx(pSSM, &shader, sizeof(shader), 0, g_aVMSVGA3DSHADERFields, NULL);
116 AssertRCReturn(rc, rc);
117
118 pData = (uint32_t *)RTMemAlloc(shader.cbData);
119 AssertReturn(pData, VERR_NO_MEMORY);
120
121 rc = SSMR3GetMem(pSSM, pData, shader.cbData);
122 AssertRCReturn(rc, rc);
123
124 rc = vmsvga3dShaderDefine(pThis, cid, shid, shader.type, shader.cbData, pData);
125 AssertRCReturn(rc, rc);
126
127 RTMemFree(pData);
128 }
129 }
130
131 /* Fetch pixel shader constants. */
132 for (uint32_t j = 0; j < cPixelShaderConst; j++)
133 {
134 VMSVGASHADERCONST ShaderConst;
135
136 rc = SSMR3GetStructEx(pSSM, &ShaderConst, sizeof(ShaderConst), 0, g_aVMSVGASHADERCONSTFields, NULL);
137 AssertRCReturn(rc, rc);
138
139 if (ShaderConst.fValid)
140 {
141 rc = vmsvga3dShaderSetConst(pThis, cid, j, SVGA3D_SHADERTYPE_PS, ShaderConst.ctype, 1, ShaderConst.value);
142 AssertRCReturn(rc, rc);
143 }
144 }
145
146 /* Fetch vertex shader constants. */
147 for (uint32_t j = 0; j < cVertexShaderConst; j++)
148 {
149 VMSVGASHADERCONST ShaderConst;
150
151 rc = SSMR3GetStructEx(pSSM, &ShaderConst, sizeof(ShaderConst), 0, g_aVMSVGASHADERCONSTFields, NULL);
152 AssertRCReturn(rc, rc);
153
154 if (ShaderConst.fValid)
155 {
156 rc = vmsvga3dShaderSetConst(pThis, cid, j, SVGA3D_SHADERTYPE_VS, ShaderConst.ctype, 1, ShaderConst.value);
157 AssertRCReturn(rc, rc);
158 }
159 }
160
161 }
162 }
163
164 /* Fetch all surfaces. */
165 for (uint32_t i = 0; i < cSurfaces; i++)
166 {
167 uint32_t sid;
168
169 /* Fetch the id first. */
170 rc = SSMR3GetU32(pSSM, &sid);
171 AssertRCReturn(rc, rc);
172
173 if (sid != SVGA3D_INVALID_ID)
174 {
175 VMSVGA3DSURFACE surface;
176
177 /* Fetch the surface structure first. */
178 rc = SSMR3GetStructEx(pSSM, &surface, sizeof(surface), 0, g_aVMSVGA3DSURFACEFields, NULL);
179 AssertRCReturn(rc, rc);
180
181 {
182 uint32_t cMipLevels = surface.faces[0].numMipLevels * surface.cFaces;
183 PVMSVGA3DMIPMAPLEVEL pMipmapLevel = (PVMSVGA3DMIPMAPLEVEL)RTMemAlloc(cMipLevels * sizeof(VMSVGA3DMIPMAPLEVEL));
184 AssertReturn(pMipmapLevel, VERR_NO_MEMORY);
185 SVGA3dSize *pMipmapLevelSize = (SVGA3dSize *)RTMemAlloc(cMipLevels * sizeof(SVGA3dSize));
186 AssertReturn(pMipmapLevelSize, VERR_NO_MEMORY);
187
188 /* Load the mip map level info. */
189 for (uint32_t face=0; face < surface.cFaces; face++)
190 {
191 for (uint32_t j = 0; j < surface.faces[0].numMipLevels; j++)
192 {
193 uint32_t idx = j + face * surface.faces[0].numMipLevels;
194 /* Load the mip map level struct. */
195 rc = SSMR3GetStructEx(pSSM, &pMipmapLevel[idx], sizeof(pMipmapLevel[idx]), 0, g_aVMSVGA3DMIPMAPLEVELFields, NULL);
196 AssertRCReturn(rc, rc);
197
198 pMipmapLevelSize[idx] = pMipmapLevel[idx].size;
199 }
200 }
201
202 rc = vmsvga3dSurfaceDefine(pThis, sid, surface.flags, surface.format, surface.faces, surface.multiSampleCount, surface.autogenFilter, cMipLevels, pMipmapLevelSize);
203 AssertRCReturn(rc, rc);
204
205 RTMemFree(pMipmapLevelSize);
206 RTMemFree(pMipmapLevel);
207 }
208
209 PVMSVGA3DSURFACE pSurface = &pState->paSurface[sid];
210 Assert(pSurface->id == sid);
211
212 pSurface->fDirty = false;
213
214 /* Load the mip map level data. */
215 for (uint32_t j = 0; j < pSurface->faces[0].numMipLevels * pSurface->cFaces; j++)
216 {
217 PVMSVGA3DMIPMAPLEVEL pMipmapLevel = &pSurface->pMipmapLevels[j];
218 bool fDataPresent = false;
219
220 Assert(pMipmapLevel->cbSurface);
221 pMipmapLevel->pSurfaceData = RTMemAllocZ(pMipmapLevel->cbSurface);
222 AssertReturn(pMipmapLevel->pSurfaceData, VERR_NO_MEMORY);
223
224 /* Fetch the data present boolean first. */
225 rc = SSMR3GetBool(pSSM, &fDataPresent);
226 AssertRCReturn(rc, rc);
227
228 Log(("Surface sid=%x: load mipmap level %d with %x bytes data (present=%d).\n", sid, j, pMipmapLevel->cbSurface, fDataPresent));
229
230 if (fDataPresent)
231 {
232 rc = SSMR3GetMem(pSSM, pMipmapLevel->pSurfaceData, pMipmapLevel->cbSurface);
233 AssertRCReturn(rc, rc);
234 pMipmapLevel->fDirty = true;
235 pSurface->fDirty = true;
236 }
237 else
238 {
239 pMipmapLevel->fDirty = false;
240 }
241 }
242 }
243 }
244
245 /* Reinitialize all active contexts. */
246 for (uint32_t i = 0; i < pState->cContexts; i++)
247 {
248 PVMSVGA3DCONTEXT pContext = &pState->paContext[i];
249 uint32_t cid = pContext->id;
250
251 if (cid != SVGA3D_INVALID_ID)
252 {
253 /* First set the render targets as they change the internal state (reset viewport etc) */
254 Log(("vmsvga3dLoadExec: Recreate render targets BEGIN\n"));
255 for (uint32_t j = 0; j < RT_ELEMENTS(pContext->state.aRenderTargets); j++)
256 {
257 if (pContext->state.aRenderTargets[j] != SVGA3D_INVALID_ID)
258 {
259 SVGA3dSurfaceImageId target;
260
261 target.sid = pContext->state.aRenderTargets[j];
262 target.face = 0;
263 target.mipmap = 0;
264 rc = vmsvga3dSetRenderTarget(pThis, cid, (SVGA3dRenderTargetType)j, target);
265 AssertRCReturn(rc, rc);
266 }
267 }
268 Log(("vmsvga3dLoadExec: Recreate render targets END\n"));
269
270 /* Recreate the render state */
271 Log(("vmsvga3dLoadExec: Recreate render state BEGIN\n"));
272 for (uint32_t j = 0; j < RT_ELEMENTS(pContext->state.aRenderState); j++)
273 {
274 SVGA3dRenderState *pRenderState = &pContext->state.aRenderState[j];
275
276 if (pRenderState->state != SVGA3D_RS_INVALID)
277 vmsvga3dSetRenderState(pThis, pContext->id, 1, pRenderState);
278 }
279 Log(("vmsvga3dLoadExec: Recreate render state END\n"));
280
281 /* Recreate the texture state */
282 Log(("vmsvga3dLoadExec: Recreate texture state BEGIN\n"));
283 for (uint32_t iStage = 0; iStage < SVGA3D_MAX_TEXTURE_STAGE; iStage++)
284 {
285 for (uint32_t j = 0; j < SVGA3D_TS_MAX; j++)
286 {
287 SVGA3dTextureState *pTextureState = &pContext->state.aTextureState[iStage][j];
288
289 if (pTextureState->name != SVGA3D_TS_INVALID)
290 vmsvga3dSetTextureState(pThis, pContext->id, 1, pTextureState);
291 }
292 }
293 Log(("vmsvga3dLoadExec: Recreate texture state END\n"));
294
295 /* Reprogram the clip planes. */
296 for (uint32_t j = 0; j < RT_ELEMENTS(pContext->state.aClipPlane); j++)
297 {
298 if (pContext->state.aClipPlane[j].fValid == true)
299 vmsvga3dSetClipPlane(pThis, cid, j, pContext->state.aClipPlane[j].plane);
300 }
301
302 /* Reprogram the light data. */
303 for (uint32_t j = 0; j < RT_ELEMENTS(pContext->state.aLightData); j++)
304 {
305 if (pContext->state.aLightData[j].fValidData == true)
306 vmsvga3dSetLightData(pThis, cid, j, &pContext->state.aLightData[j].data);
307 if (pContext->state.aLightData[j].fEnabled)
308 vmsvga3dSetLightEnabled(pThis, cid, j, true);
309 }
310
311 /* Recreate the transform state. */
312 if (pContext->state.u32UpdateFlags & VMSVGA3D_UPDATE_TRANSFORM)
313 {
314 for (uint32_t j = 0; j < RT_ELEMENTS(pContext->state.aTransformState); j++)
315 {
316 if (pContext->state.aTransformState[j].fValid == true)
317 vmsvga3dSetTransform(pThis, cid, (SVGA3dTransformType)j, pContext->state.aTransformState[j].matrix);
318 }
319 }
320
321 /* Reprogram the material data. */
322 if (pContext->state.u32UpdateFlags & VMSVGA3D_UPDATE_MATERIAL)
323 {
324 for (uint32_t j = 0; j < RT_ELEMENTS(pContext->state.aMaterial); j++)
325 {
326 if (pContext->state.aMaterial[j].fValid == true)
327 vmsvga3dSetMaterial(pThis, cid, (SVGA3dFace)j, &pContext->state.aMaterial[j].material);
328 }
329 }
330
331 if (pContext->state.u32UpdateFlags & VMSVGA3D_UPDATE_SCISSORRECT)
332 vmsvga3dSetScissorRect(pThis, cid, &pContext->state.RectScissor);
333 if (pContext->state.u32UpdateFlags & VMSVGA3D_UPDATE_ZRANGE)
334 vmsvga3dSetZRange(pThis, cid, pContext->state.zRange);
335 if (pContext->state.u32UpdateFlags & VMSVGA3D_UPDATE_VIEWPORT)
336 vmsvga3dSetViewPort(pThis, cid, &pContext->state.RectViewPort);
337 if (pContext->state.u32UpdateFlags & VMSVGA3D_UPDATE_VERTEXSHADER)
338 vmsvga3dShaderSet(pThis, cid, SVGA3D_SHADERTYPE_VS, pContext->state.shidVertex);
339 if (pContext->state.u32UpdateFlags & VMSVGA3D_UPDATE_PIXELSHADER)
340 vmsvga3dShaderSet(pThis, cid, SVGA3D_SHADERTYPE_PS, pContext->state.shidPixel);
341 }
342 }
343 return VINF_SUCCESS;
344}
345
346int vmsvga3dSaveExec(PVGASTATE pThis, PSSMHANDLE pSSM)
347{
348 PVMSVGA3DSTATE pState = (PVMSVGA3DSTATE)pThis->svga.p3dState;
349 AssertReturn(pState, VERR_NO_MEMORY);
350 int rc;
351
352 /* Save a copy of the generic 3d state first. */
353 rc = SSMR3PutStructEx(pSSM, pState, sizeof(*pState), 0, g_aVMSVGA3DSTATEFields, NULL);
354 AssertRCReturn(rc, rc);
355
356 /* Save all active contexts. */
357 for (uint32_t i = 0; i < pState->cContexts; i++)
358 {
359 PVMSVGA3DCONTEXT pContext = &pState->paContext[i];
360 uint32_t cid = pContext->id;
361
362 /* Save the id first. */
363 rc = SSMR3PutU32(pSSM, cid);
364 AssertRCReturn(rc, rc);
365
366 if (cid != SVGA3D_INVALID_ID)
367 {
368 /* Save a copy of the context structure first. */
369 rc = SSMR3PutStructEx(pSSM, pContext, sizeof(*pContext), 0, g_aVMSVGA3DCONTEXTFields, NULL);
370 AssertRCReturn(rc, rc);
371
372 /* Save all pixel shaders. */
373 for (uint32_t j = 0; j < pContext->cPixelShaders; j++)
374 {
375 PVMSVGA3DSHADER pShader = &pContext->paPixelShader[j];
376
377 /* Save the id first. */
378 rc = SSMR3PutU32(pSSM, pShader->id);
379 AssertRCReturn(rc, rc);
380
381 if (pShader->id != SVGA3D_INVALID_ID)
382 {
383 uint32_t cbData = pShader->cbData;
384
385 /* Save a copy of the shader struct. */
386 rc = SSMR3PutStructEx(pSSM, pShader, sizeof(*pShader), 0, g_aVMSVGA3DSHADERFields, NULL);
387 AssertRCReturn(rc, rc);
388
389 Log(("Save pixelshader shid=%d with %x bytes code.\n", pShader->id, cbData));
390 rc = SSMR3PutMem(pSSM, pShader->pShaderProgram, cbData);
391 AssertRCReturn(rc, rc);
392 }
393 }
394
395 /* Save all vertex shaders. */
396 for (uint32_t j = 0; j < pContext->cVertexShaders; j++)
397 {
398 PVMSVGA3DSHADER pShader = &pContext->paVertexShader[j];
399
400 /* Save the id first. */
401 rc = SSMR3PutU32(pSSM, pShader->id);
402 AssertRCReturn(rc, rc);
403
404 if (pShader->id != SVGA3D_INVALID_ID)
405 {
406 uint32_t cbData = pShader->cbData;
407
408 /* Save a copy of the shader struct. */
409 rc = SSMR3PutStructEx(pSSM, pShader, sizeof(*pShader), 0, g_aVMSVGA3DSHADERFields, NULL);
410 AssertRCReturn(rc, rc);
411
412 Log(("Save vertex shader shid=%d with %x bytes code.\n", pShader->id, cbData));
413 /* Fetch the shader code and save it. */
414 rc = SSMR3PutMem(pSSM, pShader->pShaderProgram, cbData);
415 AssertRCReturn(rc, rc);
416 }
417 }
418
419 /* Save pixel shader constants. */
420 for (uint32_t j = 0; j < pContext->state.cPixelShaderConst; j++)
421 {
422 rc = SSMR3PutStructEx(pSSM, &pContext->state.paPixelShaderConst[j], sizeof(pContext->state.paPixelShaderConst[j]), 0, g_aVMSVGASHADERCONSTFields, NULL);
423 AssertRCReturn(rc, rc);
424 }
425
426 /* Save vertex shader constants. */
427 for (uint32_t j = 0; j < pContext->state.cVertexShaderConst; j++)
428 {
429 rc = SSMR3PutStructEx(pSSM, &pContext->state.paVertexShaderConst[j], sizeof(pContext->state.paVertexShaderConst[j]), 0, g_aVMSVGASHADERCONSTFields, NULL);
430 AssertRCReturn(rc, rc);
431 }
432 }
433 }
434
435 /* Save all active surfaces. */
436 for (uint32_t sid = 0; sid < pState->cSurfaces; sid++)
437 {
438 PVMSVGA3DSURFACE pSurface = &pState->paSurface[sid];
439
440 /* Save the id first. */
441 rc = SSMR3PutU32(pSSM, pSurface->id);
442 AssertRCReturn(rc, rc);
443
444 if (pSurface->id != SVGA3D_INVALID_ID)
445 {
446 /* Save a copy of the surface structure first. */
447 rc = SSMR3PutStructEx(pSSM, pSurface, sizeof(*pSurface), 0, g_aVMSVGA3DSURFACEFields, NULL);
448 AssertRCReturn(rc, rc);
449
450 /* Save the mip map level info. */
451 for (uint32_t face=0; face < pSurface->cFaces; face++)
452 {
453 for (uint32_t i = 0; i < pSurface->faces[0].numMipLevels; i++)
454 {
455 uint32_t idx = i + face * pSurface->faces[0].numMipLevels;
456 PVMSVGA3DMIPMAPLEVEL pMipmapLevel = &pSurface->pMipmapLevels[idx];
457
458 /* Save a copy of the mip map level struct. */
459 rc = SSMR3PutStructEx(pSSM, pMipmapLevel, sizeof(*pMipmapLevel), 0, g_aVMSVGA3DMIPMAPLEVELFields, NULL);
460 AssertRCReturn(rc, rc);
461 }
462 }
463
464 /* Save the mip map level data. */
465 for (uint32_t face=0; face < pSurface->cFaces; face++)
466 {
467 for (uint32_t i = 0; i < pSurface->faces[0].numMipLevels; i++)
468 {
469 uint32_t idx = i + face * pSurface->faces[0].numMipLevels;
470 PVMSVGA3DMIPMAPLEVEL pMipmapLevel = &pSurface->pMipmapLevels[idx];
471
472 Log(("Surface sid=%d: save mipmap level %d with %x bytes data.\n", sid, i, pMipmapLevel->cbSurface));
473
474#ifdef VMSVGA3D_DIRECT3D
475 if (!pSurface->u.pSurface)
476#else
477 if (pSurface->oglId.texture == OPENGL_INVALID_ID)
478#endif
479 {
480 if (pMipmapLevel->fDirty)
481 {
482 /* Data follows */
483 rc = SSMR3PutBool(pSSM, true);
484 AssertRCReturn(rc, rc);
485
486 Assert(pMipmapLevel->cbSurface);
487 rc = SSMR3PutMem(pSSM, pMipmapLevel->pSurfaceData, pMipmapLevel->cbSurface);
488 AssertRCReturn(rc, rc);
489 }
490 else
491 {
492 /* No data follows */
493 rc = SSMR3PutBool(pSSM, false);
494 AssertRCReturn(rc, rc);
495 }
496 }
497 else
498 {
499#ifdef VMSVGA3D_DIRECT3D
500 void *pData;
501 bool fRenderTargetTexture = false;
502 bool fTexture = false;
503 bool fVertex = false;
504 bool fSkipSave = false;
505 HRESULT hr;
506
507 Assert(pMipmapLevel->cbSurface);
508 pData = RTMemAllocZ(pMipmapLevel->cbSurface);
509 AssertReturn(pData, VERR_NO_MEMORY);
510
511 switch (pSurface->flags & (SVGA3D_SURFACE_HINT_INDEXBUFFER | SVGA3D_SURFACE_HINT_VERTEXBUFFER | SVGA3D_SURFACE_HINT_TEXTURE | SVGA3D_SURFACE_HINT_RENDERTARGET | SVGA3D_SURFACE_HINT_DEPTHSTENCIL | SVGA3D_SURFACE_CUBEMAP))
512 {
513 case SVGA3D_SURFACE_HINT_DEPTHSTENCIL:
514 case SVGA3D_SURFACE_HINT_DEPTHSTENCIL | SVGA3D_SURFACE_HINT_TEXTURE:
515 /* @todo unable to easily fetch depth surface data in d3d 9 */
516 fSkipSave = true;
517 break;
518 case SVGA3D_SURFACE_HINT_TEXTURE | SVGA3D_SURFACE_HINT_RENDERTARGET:
519 fRenderTargetTexture = true;
520 /* no break */
521 case SVGA3D_SURFACE_HINT_TEXTURE:
522 fTexture = true;
523 /* no break */
524 case SVGA3D_SURFACE_HINT_RENDERTARGET:
525 {
526 D3DLOCKED_RECT LockedRect;
527
528 if (fTexture)
529 {
530 if (pSurface->bounce.pTexture)
531 {
532 if ( !pSurface->fDirty
533 && fRenderTargetTexture
534 && i == 0 /* only the first time */)
535 {
536 IDirect3DSurface9 *pSrc, *pDest;
537
538 /* @todo stricter checks for associated context */
539 uint32_t cid = pSurface->idAssociatedContext;
540 if ( cid >= pState->cContexts
541 || pState->paContext[cid].id != cid)
542 {
543 Log(("vmsvga3dSaveExec invalid context id (%x - %x)!\n", cid, (cid >= pState->cContexts) ? -1 : pState->paContext[cid].id));
544 AssertFailedReturn(VERR_INVALID_PARAMETER);
545 }
546 PVMSVGA3DCONTEXT pContext = &pState->paContext[cid];
547
548 hr = pSurface->bounce.pTexture->GetSurfaceLevel(i, &pDest);
549 AssertMsgReturn(hr == D3D_OK, ("vmsvga3dSaveExec: GetSurfaceLevel failed with %x\n", hr), VERR_INTERNAL_ERROR);
550
551 hr = pSurface->u.pTexture->GetSurfaceLevel(i, &pSrc);
552 AssertMsgReturn(hr == D3D_OK, ("vmsvga3dSaveExec: GetSurfaceLevel failed with %x\n", hr), VERR_INTERNAL_ERROR);
553
554 hr = pContext->pDevice->GetRenderTargetData(pSrc, pDest);
555 AssertMsgReturn(hr == D3D_OK, ("vmsvga3dSaveExec: GetRenderTargetData failed with %x\n", hr), VERR_INTERNAL_ERROR);
556
557 pSrc->Release();
558 pDest->Release();
559 }
560
561 hr = pSurface->bounce.pTexture->LockRect(i, /* texture level */
562 &LockedRect,
563 NULL,
564 D3DLOCK_READONLY);
565 }
566 else
567 hr = pSurface->u.pTexture->LockRect(i, /* texture level */
568 &LockedRect,
569 NULL,
570 D3DLOCK_READONLY);
571 }
572 else
573 hr = pSurface->u.pSurface->LockRect(&LockedRect,
574 NULL,
575 D3DLOCK_READONLY);
576 AssertMsgReturn(hr == D3D_OK, ("vmsvga3dSaveExec: LockRect failed with %x\n", hr), VERR_INTERNAL_ERROR);
577
578 /* Copy the data one line at a time in case the internal pitch is different. */
579 for (uint32_t j = 0; j < pMipmapLevel->size.height; j++)
580 {
581 memcpy((uint8_t *)pData + j * pMipmapLevel->cbSurfacePitch, (uint8_t *)LockedRect.pBits + j * LockedRect.Pitch, pMipmapLevel->cbSurfacePitch);
582 }
583
584 if (fTexture)
585 {
586 if (pSurface->bounce.pTexture)
587 {
588 hr = pSurface->bounce.pTexture->UnlockRect(i);
589 AssertMsgReturn(hr == D3D_OK, ("vmsvga3dSaveExec: UnlockRect failed with %x\n", hr), VERR_INTERNAL_ERROR);
590 }
591 else
592 hr = pSurface->u.pTexture->UnlockRect(i);
593 }
594 else
595 hr = pSurface->u.pSurface->UnlockRect();
596 AssertMsgReturn(hr == D3D_OK, ("vmsvga3dSaveExec: UnlockRect failed with %x\n", hr), VERR_INTERNAL_ERROR);
597 break;
598 }
599
600 case SVGA3D_SURFACE_HINT_VERTEXBUFFER:
601 fVertex = true;
602 /* no break */
603
604 case SVGA3D_SURFACE_HINT_INDEXBUFFER:
605 {
606 uint8_t *pD3DData;
607
608 if (fVertex)
609 hr = pSurface->u.pVertexBuffer->Lock(0, 0, (void **)&pD3DData, D3DLOCK_READONLY);
610 else
611 hr = pSurface->u.pIndexBuffer->Lock(0, 0, (void **)&pD3DData, D3DLOCK_READONLY);
612 AssertMsg(hr == D3D_OK, ("vmsvga3dSaveExec: Lock %s failed with %x\n", (fVertex) ? "vertex" : "index", hr));
613
614 memcpy(pData, pD3DData, pMipmapLevel->cbSurface);
615
616 if (fVertex)
617 hr = pSurface->u.pVertexBuffer->Unlock();
618 else
619 hr = pSurface->u.pIndexBuffer->Unlock();
620 AssertMsg(hr == D3D_OK, ("vmsvga3dSaveExec: Unlock %s failed with %x\n", (fVertex) ? "vertex" : "index", hr));
621 break;
622 }
623
624 default:
625 AssertFailed();
626 break;
627 }
628
629 if (!fSkipSave)
630 {
631 /* Data follows */
632 rc = SSMR3PutBool(pSSM, true);
633 AssertRCReturn(rc, rc);
634
635 /* And write the surface data. */
636 rc = SSMR3PutMem(pSSM, pData, pMipmapLevel->cbSurface);
637 AssertRCReturn(rc, rc);
638 }
639 else
640 {
641 /* No data follows */
642 rc = SSMR3PutBool(pSSM, false);
643 AssertRCReturn(rc, rc);
644 }
645
646 RTMemFree(pData);
647#elif defined(VMSVGA3D_OPENGL)
648 void *pData = NULL;
649
650 /* @todo stricter checks for associated context */
651 uint32_t cid = pSurface->idAssociatedContext;
652 if ( cid >= pState->cContexts
653 || pState->paContext[cid].id != cid)
654 {
655 Log(("vmsvga3dSaveExec: invalid context id (%x - %x)!\n", cid, (cid >= pState->cContexts) ? -1 : pState->paContext[cid].id));
656 AssertFailedReturn(VERR_INVALID_PARAMETER);
657 }
658 PVMSVGA3DCONTEXT pContext = &pState->paContext[cid];
659 VMSVGA3D_SET_CURRENT_CONTEXT(pState, pContext);
660
661 Assert(pMipmapLevel->cbSurface);
662
663 switch (pSurface->flags & (SVGA3D_SURFACE_HINT_INDEXBUFFER | SVGA3D_SURFACE_HINT_VERTEXBUFFER | SVGA3D_SURFACE_HINT_TEXTURE | SVGA3D_SURFACE_HINT_RENDERTARGET | SVGA3D_SURFACE_HINT_DEPTHSTENCIL | SVGA3D_SURFACE_CUBEMAP))
664 {
665 default:
666 AssertFailed();
667 /* no break */
668 case SVGA3D_SURFACE_HINT_DEPTHSTENCIL:
669 case SVGA3D_SURFACE_HINT_DEPTHSTENCIL | SVGA3D_SURFACE_HINT_TEXTURE:
670 /* @todo fetch data from the renderbuffer */
671 /* No data follows */
672 rc = SSMR3PutBool(pSSM, false);
673 AssertRCReturn(rc, rc);
674 break;
675
676 case SVGA3D_SURFACE_HINT_TEXTURE | SVGA3D_SURFACE_HINT_RENDERTARGET:
677 case SVGA3D_SURFACE_HINT_TEXTURE:
678 case SVGA3D_SURFACE_HINT_RENDERTARGET:
679 {
680 GLint activeTexture;
681
682 pData = RTMemAllocZ(pMipmapLevel->cbSurface);
683 AssertReturn(pData, VERR_NO_MEMORY);
684
685 glGetIntegerv(GL_TEXTURE_BINDING_2D, &activeTexture);
686 VMSVGA3D_CHECK_LAST_ERROR_WARN(pState, pContext);
687
688 glBindTexture(GL_TEXTURE_2D, pSurface->oglId.texture);
689 VMSVGA3D_CHECK_LAST_ERROR_WARN(pState, pContext);
690
691 glGetTexImage(GL_TEXTURE_2D,
692 i,
693 pSurface->formatGL,
694 pSurface->typeGL,
695 pData);
696 VMSVGA3D_CHECK_LAST_ERROR_WARN(pState, pContext);
697
698 /* Data follows */
699 rc = SSMR3PutBool(pSSM, true);
700 AssertRCReturn(rc, rc);
701
702 /* And write the surface data. */
703 rc = SSMR3PutMem(pSSM, pData, pMipmapLevel->cbSurface);
704 AssertRCReturn(rc, rc);
705
706 /* Restore the old active texture. */
707 glBindTexture(GL_TEXTURE_2D, activeTexture);
708 VMSVGA3D_CHECK_LAST_ERROR_WARN(pState, pContext);
709 break;
710 }
711
712 case SVGA3D_SURFACE_HINT_VERTEXBUFFER:
713 case SVGA3D_SURFACE_HINT_INDEXBUFFER:
714 {
715 uint8_t *pBufferData;
716
717 pState->ext.glBindBuffer(GL_ARRAY_BUFFER, pSurface->oglId.buffer);
718 VMSVGA3D_CHECK_LAST_ERROR(pState, pContext);
719
720 pBufferData = (uint8_t *)pState->ext.glMapBuffer(GL_ARRAY_BUFFER, GL_READ_ONLY);
721 VMSVGA3D_CHECK_LAST_ERROR(pState, pContext);
722 Assert(pBufferData);
723
724 /* Data follows */
725 rc = SSMR3PutBool(pSSM, true);
726 AssertRCReturn(rc, rc);
727
728 /* And write the surface data. */
729 rc = SSMR3PutMem(pSSM, pBufferData, pMipmapLevel->cbSurface);
730 AssertRCReturn(rc, rc);
731
732 pState->ext.glUnmapBuffer(GL_ARRAY_BUFFER);
733 VMSVGA3D_CHECK_LAST_ERROR(pState, pContext);
734
735 pState->ext.glBindBuffer(GL_ARRAY_BUFFER, 0);
736 VMSVGA3D_CHECK_LAST_ERROR(pState, pContext);
737
738 }
739 }
740 if (pData)
741 RTMemFree(pData);
742#else
743#error "Unexpected 3d backend"
744#endif
745 }
746 }
747 }
748 }
749 }
750 return VINF_SUCCESS;
751}
752
753static uint32_t vmsvga3dSaveShaderConst(PVMSVGA3DCONTEXT pContext, uint32_t reg, SVGA3dShaderType type, SVGA3dShaderConstType ctype, uint32_t val1, uint32_t val2, uint32_t val3, uint32_t val4)
754{
755 /* Choose a sane upper limit. */
756 AssertReturn(reg < _32K, VERR_INVALID_PARAMETER);
757
758 if (type == SVGA3D_SHADERTYPE_VS)
759 {
760 if (pContext->state.cVertexShaderConst <= reg)
761 {
762 pContext->state.paVertexShaderConst = (PVMSVGASHADERCONST)RTMemRealloc(pContext->state.paVertexShaderConst, sizeof(VMSVGASHADERCONST) * (reg + 1));
763 AssertReturn(pContext->state.paVertexShaderConst, VERR_NO_MEMORY);
764 for (uint32_t i = pContext->state.cVertexShaderConst; i < reg + 1; i++)
765 pContext->state.paVertexShaderConst[i].fValid = false;
766 pContext->state.cVertexShaderConst = reg + 1;
767 }
768
769 pContext->state.paVertexShaderConst[reg].fValid = true;
770 pContext->state.paVertexShaderConst[reg].ctype = ctype;
771 pContext->state.paVertexShaderConst[reg].value[0] = val1;
772 pContext->state.paVertexShaderConst[reg].value[1] = val2;
773 pContext->state.paVertexShaderConst[reg].value[2] = val3;
774 pContext->state.paVertexShaderConst[reg].value[3] = val4;
775 }
776 else
777 {
778 Assert(type == SVGA3D_SHADERTYPE_PS);
779 if (pContext->state.cPixelShaderConst <= reg)
780 {
781 pContext->state.paPixelShaderConst = (PVMSVGASHADERCONST)RTMemRealloc(pContext->state.paPixelShaderConst, sizeof(VMSVGASHADERCONST) * (reg + 1));
782 AssertReturn(pContext->state.paPixelShaderConst, VERR_NO_MEMORY);
783 for (uint32_t i = pContext->state.cPixelShaderConst; i < reg + 1; i++)
784 pContext->state.paPixelShaderConst[i].fValid = false;
785 pContext->state.cPixelShaderConst = reg + 1;
786 }
787
788 pContext->state.paPixelShaderConst[reg].fValid = true;
789 pContext->state.paPixelShaderConst[reg].ctype = ctype;
790 pContext->state.paPixelShaderConst[reg].value[0] = val1;
791 pContext->state.paPixelShaderConst[reg].value[1] = val2;
792 pContext->state.paPixelShaderConst[reg].value[2] = val3;
793 pContext->state.paPixelShaderConst[reg].value[3] = val4;
794 }
795
796 return VINF_SUCCESS;
797}
798
799
800#endif /* __DEVVMWARE3D_STATE_H__ */
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