VirtualBox

source: vbox/trunk/src/VBox/GuestHost/OpenGL/state_tracker/state_snapshot.c@ 24968

Last change on this file since 24968 was 24891, checked in by vboxsync, 15 years ago

crOpenGL: fix black screen in compiz after snapshot load (public #4868)

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 59.6 KB
Line 
1/* $Id: state_snapshot.c 24891 2009-11-24 11:36:06Z vboxsync $ */
2
3/** @file
4 * VBox Context state saving/loading used by VM snapshot
5 */
6
7/*
8 * Copyright (C) 2008 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
19 * Clara, CA 95054 USA or visit http://www.sun.com if you need
20 * additional information or have any questions.
21 */
22
23#include "state.h"
24#include "state/cr_statetypes.h"
25#include "state/cr_texture.h"
26#include "cr_mem.h"
27#include "cr_string.h"
28#include "cr_pixeldata.h"
29#include <stdio.h>
30
31#include <iprt/assert.h>
32#include <iprt/types.h>
33#include <iprt/err.h>
34#include <VBox/err.h>
35
36/* @todo
37 * We have two ways of saving/loading states.
38 *
39 * First which is being used atm, just pure saving/loading of structures.
40 * The drawback is we have to deal with all the pointers around those structures,
41 * we'd have to update this code if we'd change state tracking.
42 * On the bright side it's fast, though it's not really needed as it's not that often operation.
43 * It could also worth to split those functions into appropriate parts,
44 * similar to the way context creation is being done.
45 *
46 * Second way would be to implement full dispatch api table and substitute diff_api during saving/loading.
47 * Then if we implement that api in a similar way to packer/unpacker with a change to store/load
48 * via provided pSSM handle instead of pack buffer,
49 * saving state could be done by simple diffing against empty "dummy" context.
50 * Restoring state in such case would look like unpacking commands from pSSM instead of network buffer.
51 * This would be slower (who cares) but most likely will not require any code changes to support in future.
52 * We will reduce amount of saved data as we'd save only changed state parts, but I doubt it'd be that much.
53 * It could be done for the first way as well, but requires tons of bit checks.
54 */
55
56/*@todo move with the one from state_texture.c to some header*/
57#define MAX_MIPMAP_LEVELS 20
58
59static int32_t crStateAllocAndSSMR3GetMem(PSSMHANDLE pSSM, void **pBuffer, size_t cbBuffer)
60{
61 CRASSERT(pSSM && pBuffer && cbBuffer>0);
62
63 *pBuffer = crAlloc(cbBuffer);
64 if (!*pBuffer)
65 return VERR_NO_MEMORY;
66
67 return SSMR3GetMem(pSSM, *pBuffer, cbBuffer);
68}
69
70static int32_t crStateSaveTextureObjData(CRTextureObj *pTexture, PSSMHANDLE pSSM)
71{
72 int32_t rc, face, i;
73
74 CRASSERT(pTexture && pSSM);
75
76 for (face = 0; face < 6; face++) {
77 CRASSERT(pTexture->level[face]);
78
79 /*@todo, check if safe to go till MAX_MIPMAP_LEVELS intead of TextureState->maxLevel*/
80 for (i = 0; i < MAX_MIPMAP_LEVELS; i++) {
81 CRTextureLevel *ptl = &(pTexture->level[face][i]);
82 rc = SSMR3PutMem(pSSM, ptl, sizeof(*ptl));
83 AssertRCReturn(rc, rc);
84 if (ptl->img)
85 {
86 CRASSERT(ptl->bytes);
87 rc = SSMR3PutMem(pSSM, ptl->img, ptl->bytes);
88 AssertRCReturn(rc, rc);
89 }
90#ifdef CR_STATE_NO_TEXTURE_IMAGE_STORE
91 /* Note, this is not a bug.
92 * Even with CR_STATE_NO_TEXTURE_IMAGE_STORE defined, it's possible that ptl->img!=NULL.
93 * For ex. we're saving snapshot right after it was loaded
94 * and some context hasn't been used by the guest application yet
95 * (pContext->texture.bResyncNeeded==GL_TRUE).
96 */
97 else if (ptl->bytes)
98 {
99 char *pImg;
100
101 pImg = crAlloc(ptl->bytes);
102 if (!pImg) return VERR_NO_MEMORY;
103
104 diff_api.BindTexture(pTexture->target, pTexture->name);
105 diff_api.GetTexImage(pTexture->target, i, ptl->format, ptl->type, pImg);
106
107 rc = SSMR3PutMem(pSSM, pImg, ptl->bytes);
108 crFree(pImg);
109 AssertRCReturn(rc, rc);
110 }
111#endif
112 }
113 }
114
115 return VINF_SUCCESS;
116}
117
118static int32_t crStateLoadTextureObjData(CRTextureObj *pTexture, PSSMHANDLE pSSM)
119{
120 int32_t rc, face, i;
121
122 CRASSERT(pTexture && pSSM);
123
124 for (face = 0; face < 6; face++) {
125 CRASSERT(pTexture->level[face]);
126
127 for (i = 0; i < MAX_MIPMAP_LEVELS; i++) {
128 CRTextureLevel *ptl = &(pTexture->level[face][i]);
129 CRASSERT(!ptl->img);
130
131 rc = SSMR3GetMem(pSSM, ptl, sizeof(*ptl));
132 AssertRCReturn(rc, rc);
133 if (ptl->img)
134 {
135 CRASSERT(ptl->bytes);
136
137 ptl->img = crAlloc(ptl->bytes);
138 if (!ptl->img) return VERR_NO_MEMORY;
139
140 rc = SSMR3GetMem(pSSM, ptl->img, ptl->bytes);
141 AssertRCReturn(rc, rc);
142 }
143#ifdef CR_STATE_NO_TEXTURE_IMAGE_STORE
144 /* Same story as in crStateSaveTextureObjData */
145 else if (ptl->bytes)
146 {
147 ptl->img = crAlloc(ptl->bytes);
148 if (!ptl->img) return VERR_NO_MEMORY;
149
150 rc = SSMR3GetMem(pSSM, ptl->img, ptl->bytes);
151 AssertRCReturn(rc, rc);
152 }
153#endif
154 crStateTextureInitTextureFormat(ptl, ptl->internalFormat);
155
156 //FILLDIRTY(ptl->dirty);
157 }
158 }
159
160 return VINF_SUCCESS;
161}
162
163static void crStateSaveSharedTextureCB(unsigned long key, void *data1, void *data2)
164{
165 CRTextureObj *pTexture = (CRTextureObj *) data1;
166 PSSMHANDLE pSSM = (PSSMHANDLE) data2;
167 int32_t rc;
168
169 CRASSERT(pTexture && pSSM);
170
171 rc = SSMR3PutMem(pSSM, &key, sizeof(key));
172 CRASSERT(rc == VINF_SUCCESS);
173 rc = SSMR3PutMem(pSSM, pTexture, sizeof(*pTexture));
174 CRASSERT(rc == VINF_SUCCESS);
175 rc = crStateSaveTextureObjData(pTexture, pSSM);
176 CRASSERT(rc == VINF_SUCCESS);
177}
178
179static int32_t crStateSaveMatrixStack(CRMatrixStack *pStack, PSSMHANDLE pSSM)
180{
181 return SSMR3PutMem(pSSM, pStack->stack, sizeof(CRmatrix) * pStack->maxDepth);
182}
183
184static int32_t crStateLoadMatrixStack(CRMatrixStack *pStack, PSSMHANDLE pSSM)
185{
186 int32_t rc;
187
188 CRASSERT(pStack && pSSM);
189
190 rc = SSMR3GetMem(pSSM, pStack->stack, sizeof(CRmatrix) * pStack->maxDepth);
191 /* fixup stack top pointer */
192 pStack->top = &pStack->stack[pStack->depth];
193 return rc;
194}
195
196static int32_t crStateSaveTextureObjPtr(CRTextureObj *pTexture, PSSMHANDLE pSSM)
197{
198 /* Current texture pointer can't be NULL for real texture unit states,
199 * but it could be NULL for unused attribute stack depths.
200 */
201 if (pTexture)
202 return SSMR3PutU32(pSSM, pTexture->name);
203 else
204 return VINF_SUCCESS;
205}
206
207static int32_t crStateLoadTextureObjPtr(CRTextureObj **pTexture, CRContext *pContext, GLenum target, PSSMHANDLE pSSM)
208{
209 uint32_t texName;
210 int32_t rc;
211
212 /* We're loading attrib stack with unused state */
213 if (!*pTexture)
214 return VINF_SUCCESS;
215
216 rc = SSMR3GetU32(pSSM, &texName);
217 AssertRCReturn(rc, rc);
218
219 if (texName)
220 {
221 *pTexture = (CRTextureObj *) crHashtableSearch(pContext->shared->textureTable, texName);
222 }
223 else
224 {
225 switch (target)
226 {
227 case GL_TEXTURE_1D:
228 *pTexture = &(pContext->texture.base1D);
229 break;
230 case GL_TEXTURE_2D:
231 *pTexture = &(pContext->texture.base2D);
232 break;
233#ifdef CR_OPENGL_VERSION_1_2
234 case GL_TEXTURE_3D:
235 *pTexture = &(pContext->texture.base3D);
236 break;
237#endif
238#ifdef CR_ARB_texture_cube_map
239 case GL_TEXTURE_CUBE_MAP_ARB:
240 *pTexture = &(pContext->texture.baseCubeMap);
241 break;
242#endif
243#ifdef CR_NV_texture_rectangle
244 case GL_TEXTURE_RECTANGLE_NV:
245 *pTexture = &(pContext->texture.baseRect);
246 break;
247#endif
248 default:
249 crError("LoadTextureObjPtr: Unknown texture target %d", target);
250 }
251 }
252
253 return rc;
254}
255
256static int32_t crStateSaveTexUnitCurrentTexturePtrs(CRTextureUnit *pTexUnit, PSSMHANDLE pSSM)
257{
258 int32_t rc;
259
260 rc = crStateSaveTextureObjPtr(pTexUnit->currentTexture1D, pSSM);
261 AssertRCReturn(rc, rc);
262 rc = crStateSaveTextureObjPtr(pTexUnit->currentTexture2D, pSSM);
263 AssertRCReturn(rc, rc);
264 rc = crStateSaveTextureObjPtr(pTexUnit->currentTexture3D, pSSM);
265 AssertRCReturn(rc, rc);
266#ifdef CR_ARB_texture_cube_map
267 rc = crStateSaveTextureObjPtr(pTexUnit->currentTextureCubeMap, pSSM);
268 AssertRCReturn(rc, rc);
269#endif
270#ifdef CR_NV_texture_rectangle
271 rc = crStateSaveTextureObjPtr(pTexUnit->currentTextureRect, pSSM);
272 AssertRCReturn(rc, rc);
273#endif
274
275 return rc;
276}
277
278static int32_t crStateLoadTexUnitCurrentTexturePtrs(CRTextureUnit *pTexUnit, CRContext *pContext, PSSMHANDLE pSSM)
279{
280 int32_t rc;
281
282 rc = crStateLoadTextureObjPtr(&pTexUnit->currentTexture1D, pContext, GL_TEXTURE_1D, pSSM);
283 AssertRCReturn(rc, rc);
284 rc = crStateLoadTextureObjPtr(&pTexUnit->currentTexture2D, pContext, GL_TEXTURE_1D, pSSM);
285 AssertRCReturn(rc, rc);
286 rc = crStateLoadTextureObjPtr(&pTexUnit->currentTexture3D, pContext, GL_TEXTURE_2D, pSSM);
287 AssertRCReturn(rc, rc);
288#ifdef CR_ARB_texture_cube_map
289 rc = crStateLoadTextureObjPtr(&pTexUnit->currentTextureCubeMap, pContext, GL_TEXTURE_CUBE_MAP_ARB, pSSM);
290 AssertRCReturn(rc, rc);
291#endif
292#ifdef CR_NV_texture_rectangle
293 rc = crStateLoadTextureObjPtr(&pTexUnit->currentTextureRect, pContext, GL_TEXTURE_RECTANGLE_NV, pSSM);
294 AssertRCReturn(rc, rc);
295#endif
296
297 return rc;
298}
299
300static int32_t crSateSaveEvalCoeffs1D(CREvaluator1D *pEval, PSSMHANDLE pSSM)
301{
302 int32_t rc, i;
303
304 for (i=0; i<GLEVAL_TOT; ++i)
305 {
306 if (pEval[i].coeff)
307 {
308 rc = SSMR3PutMem(pSSM, pEval[i].coeff, pEval[i].order * gleval_sizes[i] * sizeof(GLfloat));
309 AssertRCReturn(rc, rc);
310 }
311 }
312
313 return VINF_SUCCESS;
314}
315
316static int32_t crSateSaveEvalCoeffs2D(CREvaluator2D *pEval, PSSMHANDLE pSSM)
317{
318 int32_t rc, i;
319
320 for (i=0; i<GLEVAL_TOT; ++i)
321 {
322 if (pEval[i].coeff)
323 {
324 rc = SSMR3PutMem(pSSM, pEval[i].coeff, pEval[i].uorder * pEval[i].vorder * gleval_sizes[i] * sizeof(GLfloat));
325 AssertRCReturn(rc, rc);
326 }
327 }
328
329 return VINF_SUCCESS;
330}
331
332static int32_t crSateLoadEvalCoeffs1D(CREvaluator1D *pEval, GLboolean bReallocMem, PSSMHANDLE pSSM)
333{
334 int32_t rc, i;
335 size_t size;
336
337 for (i=0; i<GLEVAL_TOT; ++i)
338 {
339 if (pEval[i].coeff)
340 {
341 size = pEval[i].order * gleval_sizes[i] * sizeof(GLfloat);
342 if (bReallocMem)
343 {
344 pEval[i].coeff = (GLfloat*) crAlloc(size);
345 if (!pEval[i].coeff) return VERR_NO_MEMORY;
346 }
347 rc = SSMR3GetMem(pSSM, pEval[i].coeff, size);
348 AssertRCReturn(rc, rc);
349 }
350 }
351
352 return VINF_SUCCESS;
353}
354
355static int32_t crSateLoadEvalCoeffs2D(CREvaluator2D *pEval, GLboolean bReallocMem, PSSMHANDLE pSSM)
356{
357 int32_t rc, i;
358 size_t size;
359
360 for (i=0; i<GLEVAL_TOT; ++i)
361 {
362 if (pEval[i].coeff)
363 {
364 size = pEval[i].uorder * pEval[i].vorder * gleval_sizes[i] * sizeof(GLfloat);
365 if (bReallocMem)
366 {
367 pEval[i].coeff = (GLfloat*) crAlloc(size);
368 if (!pEval[i].coeff) return VERR_NO_MEMORY;
369 }
370 rc = SSMR3GetMem(pSSM, pEval[i].coeff, size);
371 AssertRCReturn(rc, rc);
372 }
373 }
374
375 return VINF_SUCCESS;
376}
377
378static void crStateCopyEvalPtrs1D(CREvaluator1D *pDst, CREvaluator1D *pSrc)
379{
380 int32_t i;
381
382 for (i=0; i<GLEVAL_TOT; ++i)
383 pDst[i].coeff = pSrc[i].coeff;
384
385 /*
386 pDst[GL_MAP1_VERTEX_3-GL_MAP1_COLOR_4].coeff = pSrc[GL_MAP1_VERTEX_3-GL_MAP1_COLOR_4].coeff;
387 pDst[GL_MAP1_VERTEX_4-GL_MAP1_COLOR_4].coeff = pSrc[GL_MAP1_VERTEX_4-GL_MAP1_COLOR_4].coeff;
388 pDst[GL_MAP1_INDEX-GL_MAP1_COLOR_4].coeff = pSrc[GL_MAP1_INDEX-GL_MAP1_COLOR_4].coeff;
389 pDst[GL_MAP1_COLOR_4-GL_MAP1_COLOR_4].coeff = pSrc[GL_MAP1_COLOR_4-GL_MAP1_COLOR_4].coeff;
390 pDst[GL_MAP1_NORMAL-GL_MAP1_COLOR_4].coeff = pSrc[GL_MAP1_NORMAL-GL_MAP1_COLOR_4].coeff;
391 pDst[GL_MAP1_TEXTURE_COORD_1-GL_MAP1_COLOR_4].coeff = pSrc[GL_MAP1_TEXTURE_COORD_1-GL_MAP1_COLOR_4].coeff;
392 pDst[GL_MAP1_TEXTURE_COORD_2-GL_MAP1_COLOR_4].coeff = pSrc[GL_MAP1_TEXTURE_COORD_2-GL_MAP1_COLOR_4].coeff;
393 pDst[GL_MAP1_TEXTURE_COORD_3-GL_MAP1_COLOR_4].coeff = pSrc[GL_MAP1_TEXTURE_COORD_3-GL_MAP1_COLOR_4].coeff;
394 pDst[GL_MAP1_TEXTURE_COORD_4-GL_MAP1_COLOR_4].coeff = pSrc[GL_MAP1_TEXTURE_COORD_4-GL_MAP1_COLOR_4].coeff;
395 */
396}
397
398static void crStateCopyEvalPtrs2D(CREvaluator2D *pDst, CREvaluator2D *pSrc)
399{
400 int32_t i;
401
402 for (i=0; i<GLEVAL_TOT; ++i)
403 pDst[i].coeff = pSrc[i].coeff;
404
405 /*
406 pDst[GL_MAP2_VERTEX_3-GL_MAP2_COLOR_4].coeff = pSrc[GL_MAP2_VERTEX_3-GL_MAP2_COLOR_4].coeff;
407 pDst[GL_MAP2_VERTEX_4-GL_MAP2_COLOR_4].coeff = pSrc[GL_MAP2_VERTEX_4-GL_MAP2_COLOR_4].coeff;
408 pDst[GL_MAP2_INDEX-GL_MAP2_COLOR_4].coeff = pSrc[GL_MAP2_INDEX-GL_MAP2_COLOR_4].coeff;
409 pDst[GL_MAP2_COLOR_4-GL_MAP2_COLOR_4].coeff = pSrc[GL_MAP2_COLOR_4-GL_MAP2_COLOR_4].coeff;
410 pDst[GL_MAP2_NORMAL-GL_MAP2_COLOR_4].coeff = pSrc[GL_MAP2_NORMAL-GL_MAP2_COLOR_4].coeff;
411 pDst[GL_MAP2_TEXTURE_COORD_1-GL_MAP2_COLOR_4].coeff = pSrc[GL_MAP2_TEXTURE_COORD_1-GL_MAP2_COLOR_4].coeff;
412 pDst[GL_MAP2_TEXTURE_COORD_2-GL_MAP2_COLOR_4].coeff = pSrc[GL_MAP2_TEXTURE_COORD_2-GL_MAP2_COLOR_4].coeff;
413 pDst[GL_MAP2_TEXTURE_COORD_3-GL_MAP2_COLOR_4].coeff = pSrc[GL_MAP2_TEXTURE_COORD_3-GL_MAP2_COLOR_4].coeff;
414 pDst[GL_MAP2_TEXTURE_COORD_4-GL_MAP2_COLOR_4].coeff = pSrc[GL_MAP2_TEXTURE_COORD_4-GL_MAP2_COLOR_4].coeff;
415 */
416}
417
418static void crStateSaveBufferObjectCB(unsigned long key, void *data1, void *data2)
419{
420 CRBufferObject *pBufferObj = (CRBufferObject *) data1;
421 PSSMHANDLE pSSM = (PSSMHANDLE) data2;
422 int32_t rc;
423
424 CRASSERT(pBufferObj && pSSM);
425
426 rc = SSMR3PutMem(pSSM, &key, sizeof(key));
427 CRASSERT(rc == VINF_SUCCESS);
428 rc = SSMR3PutMem(pSSM, pBufferObj, sizeof(*pBufferObj));
429 CRASSERT(rc == VINF_SUCCESS);
430
431 if (pBufferObj->data)
432 {
433 /*We could get here even though retainBufferData is false on host side, in case when we're taking snapshot
434 after state load and before this context was ever made current*/
435 CRASSERT(pBufferObj->size>0);
436 rc = SSMR3PutMem(pSSM, pBufferObj->data, pBufferObj->size);
437 CRASSERT(rc == VINF_SUCCESS);
438 }
439 else if (pBufferObj->name!=0 && pBufferObj->size>0)
440 {
441 diff_api.BindBufferARB(GL_ARRAY_BUFFER_ARB, pBufferObj->name);
442 pBufferObj->pointer = diff_api.MapBufferARB(GL_ARRAY_BUFFER_ARB, GL_READ_ONLY_ARB);
443 rc = SSMR3PutMem(pSSM, &pBufferObj->pointer, sizeof(pBufferObj->pointer));
444 CRASSERT(rc == VINF_SUCCESS);
445 if (pBufferObj->pointer)
446 {
447 rc = SSMR3PutMem(pSSM, pBufferObj->pointer, pBufferObj->size);
448 CRASSERT(rc == VINF_SUCCESS);
449 }
450 diff_api.UnmapBufferARB(GL_ARRAY_BUFFER_ARB);
451 pBufferObj->pointer = NULL;
452 }
453}
454
455static void crStateSaveProgramCB(unsigned long key, void *data1, void *data2)
456{
457 CRProgram *pProgram = (CRProgram *) data1;
458 PSSMHANDLE pSSM = (PSSMHANDLE) data2;
459 CRProgramSymbol *pSymbol;
460 int32_t rc;
461
462 CRASSERT(pProgram && pSSM);
463
464 rc = SSMR3PutMem(pSSM, &key, sizeof(key));
465 CRASSERT(rc == VINF_SUCCESS);
466 rc = SSMR3PutMem(pSSM, pProgram, sizeof(*pProgram));
467 CRASSERT(rc == VINF_SUCCESS);
468 if (pProgram->string)
469 {
470 CRASSERT(pProgram->length);
471 rc = SSMR3PutMem(pSSM, pProgram->string, pProgram->length);
472 CRASSERT(rc == VINF_SUCCESS);
473 }
474
475 for (pSymbol = pProgram->symbolTable; pSymbol; pSymbol=pSymbol->next)
476 {
477 rc = SSMR3PutMem(pSSM, pSymbol, sizeof(*pSymbol));
478 CRASSERT(rc == VINF_SUCCESS);
479 if (pSymbol->name)
480 {
481 CRASSERT(pSymbol->cbName>0);
482 rc = SSMR3PutMem(pSSM, pSymbol->name, pSymbol->cbName);
483 CRASSERT(rc == VINF_SUCCESS);
484 }
485 }
486}
487
488static void crStateSaveFramebuffersCB(unsigned long key, void *data1, void *data2)
489{
490 CRFramebufferObject *pFBO = (CRFramebufferObject*) data1;
491 PSSMHANDLE pSSM = (PSSMHANDLE) data2;
492 int32_t rc;
493
494 rc = SSMR3PutMem(pSSM, &key, sizeof(key));
495 CRASSERT(rc == VINF_SUCCESS);
496
497 rc = SSMR3PutMem(pSSM, pFBO, sizeof(*pFBO));
498 CRASSERT(rc == VINF_SUCCESS);
499}
500
501static void crStateSaveRenderbuffersCB(unsigned long key, void *data1, void *data2)
502{
503 CRRenderbufferObject *pRBO = (CRRenderbufferObject*) data1;
504 PSSMHANDLE pSSM = (PSSMHANDLE) data2;
505 int32_t rc;
506
507 rc = SSMR3PutMem(pSSM, &key, sizeof(key));
508 CRASSERT(rc == VINF_SUCCESS);
509
510 rc = SSMR3PutMem(pSSM, pRBO, sizeof(*pRBO));
511 CRASSERT(rc == VINF_SUCCESS);
512}
513
514static int32_t crStateLoadProgram(CRProgram **ppProgram, PSSMHANDLE pSSM)
515{
516 CRProgramSymbol **ppSymbol;
517 int32_t rc;
518 unsigned long key;
519
520 rc = SSMR3GetMem(pSSM, &key, sizeof(key));
521 AssertRCReturn(rc, rc);
522
523 /* we're loading default vertex or pixel program*/
524 if (*ppProgram)
525 {
526 if (key!=0) return VERR_SSM_UNEXPECTED_DATA;
527 }
528 else
529 {
530 *ppProgram = (CRProgram*) crAlloc(sizeof(CRProgram));
531 if (!ppProgram) return VERR_NO_MEMORY;
532 if (key==0) return VERR_SSM_UNEXPECTED_DATA;
533 }
534
535 rc = SSMR3GetMem(pSSM, *ppProgram, sizeof(**ppProgram));
536 AssertRCReturn(rc, rc);
537
538 if ((*ppProgram)->string)
539 {
540 CRASSERT((*ppProgram)->length);
541 (*ppProgram)->string = crAlloc((*ppProgram)->length);
542 if (!(*ppProgram)->string) return VERR_NO_MEMORY;
543 rc = SSMR3GetMem(pSSM, (void*) (*ppProgram)->string, (*ppProgram)->length);
544 AssertRCReturn(rc, rc);
545 }
546
547 for (ppSymbol = &(*ppProgram)->symbolTable; *ppSymbol; ppSymbol=&(*ppSymbol)->next)
548 {
549 *ppSymbol = crAlloc(sizeof(CRProgramSymbol));
550 if (!ppSymbol) return VERR_NO_MEMORY;
551
552 rc = SSMR3GetMem(pSSM, *ppSymbol, sizeof(**ppSymbol));
553 AssertRCReturn(rc, rc);
554
555 if ((*ppSymbol)->name)
556 {
557 CRASSERT((*ppSymbol)->cbName>0);
558 (*ppSymbol)->name = crAlloc((*ppSymbol)->cbName);
559 if (!(*ppSymbol)->name) return VERR_NO_MEMORY;
560
561 rc = SSMR3GetMem(pSSM, (void*) (*ppSymbol)->name, (*ppSymbol)->cbName);
562 AssertRCReturn(rc, rc);
563 }
564 }
565
566 return VINF_SUCCESS;
567}
568
569static void crStateSaveString(const char *pStr, PSSMHANDLE pSSM)
570{
571 int32_t len;
572 int32_t rc;
573
574 if (pStr)
575 {
576 len = crStrlen(pStr)+1;
577
578 rc = SSMR3PutS32(pSSM, len);
579 CRASSERT(rc == VINF_SUCCESS);
580
581 rc = SSMR3PutMem(pSSM, pStr, len*sizeof(*pStr));
582 CRASSERT(rc == VINF_SUCCESS);
583 }
584 else
585 {
586 rc = SSMR3PutS32(pSSM, 0);
587 CRASSERT(rc == VINF_SUCCESS);
588 }
589}
590
591static char* crStateLoadString(PSSMHANDLE pSSM)
592{
593 int32_t len, rc;
594 char* pStr = NULL;
595
596 rc = SSMR3GetS32(pSSM, &len);
597 CRASSERT(rc == VINF_SUCCESS);
598
599 if (len!=0)
600 {
601 pStr = crAlloc(len*sizeof(*pStr));
602
603 rc = SSMR3GetMem(pSSM, pStr, len*sizeof(*pStr));
604 CRASSERT(rc == VINF_SUCCESS);
605 }
606
607 return pStr;
608}
609
610static void crStateSaveGLSLShaderCB(unsigned long key, void *data1, void *data2)
611{
612 CRGLSLShader *pShader = (CRGLSLShader*) data1;
613 PSSMHANDLE pSSM = (PSSMHANDLE) data2;
614 int32_t rc;
615
616 rc = SSMR3PutMem(pSSM, &key, sizeof(key));
617 CRASSERT(rc == VINF_SUCCESS);
618
619 rc = SSMR3PutMem(pSSM, pShader, sizeof(*pShader));
620 CRASSERT(rc == VINF_SUCCESS);
621
622 if (pShader->source)
623 {
624 crStateSaveString(pShader->source, pSSM);
625 }
626 else
627 {
628 GLint sLen=0;
629 GLchar *source=NULL;
630
631 diff_api.GetShaderiv(pShader->hwid, GL_SHADER_SOURCE_LENGTH, &sLen);
632 if (sLen>0)
633 {
634 source = (GLchar*) crAlloc(sLen);
635 diff_api.GetShaderSource(pShader->hwid, sLen, NULL, source);
636 }
637
638 crStateSaveString(source, pSSM);
639 if (source) crFree(source);
640 }
641}
642
643static CRGLSLShader* crStateLoadGLSLShader(PSSMHANDLE pSSM)
644{
645 CRGLSLShader *pShader;
646 int32_t rc;
647 unsigned long key;
648
649 pShader = crAlloc(sizeof(*pShader));
650 if (!pShader) return NULL;
651
652 rc = SSMR3GetMem(pSSM, &key, sizeof(key));
653 CRASSERT(rc == VINF_SUCCESS);
654
655 rc = SSMR3GetMem(pSSM, pShader, sizeof(*pShader));
656 CRASSERT(rc == VINF_SUCCESS);
657
658 pShader->source = crStateLoadString(pSSM);
659
660 return pShader;
661}
662
663
664static void crStateSaveGLSLShaderKeyCB(unsigned long key, void *data1, void *data2)
665{
666 CRGLSLShader *pShader = (CRGLSLShader*) data1;
667 PSSMHANDLE pSSM = (PSSMHANDLE) data2;
668 int32_t rc;
669
670 rc = SSMR3PutMem(pSSM, &key, sizeof(key));
671 CRASSERT(rc == VINF_SUCCESS);
672}
673
674static void crStateSaveGLSLProgramAttribs(CRGLSLProgramState *pState, PSSMHANDLE pSSM)
675{
676 GLuint i;
677 int32_t rc;
678
679 for (i=0; i<pState->cAttribs; ++i)
680 {
681 rc = SSMR3PutMem(pSSM, &pState->pAttribs[i].index, sizeof(pState->pAttribs[i].index));
682 CRASSERT(rc == VINF_SUCCESS);
683 crStateSaveString(pState->pAttribs[i].name, pSSM);
684 }
685}
686
687static void crStateSaveGLSLProgramCB(unsigned long key, void *data1, void *data2)
688{
689 CRGLSLProgram *pProgram = (CRGLSLProgram*) data1;
690 PSSMHANDLE pSSM = (PSSMHANDLE) data2;
691 int32_t rc;
692 uint32_t ui32;
693 GLint maxUniformLen, activeUniforms=0, uniformsCount=0, i, j;
694 GLchar *name = NULL;
695 GLenum type;
696 GLint size, location;
697
698 rc = SSMR3PutMem(pSSM, &key, sizeof(key));
699 CRASSERT(rc == VINF_SUCCESS);
700
701 rc = SSMR3PutMem(pSSM, pProgram, sizeof(*pProgram));
702 CRASSERT(rc == VINF_SUCCESS);
703
704 ui32 = crHashtableNumElements(pProgram->currentState.attachedShaders);
705 rc = SSMR3PutU32(pSSM, ui32);
706 CRASSERT(rc == VINF_SUCCESS);
707
708 crHashtableWalk(pProgram->currentState.attachedShaders, crStateSaveGLSLShaderKeyCB, pSSM);
709
710 if (pProgram->activeState.attachedShaders)
711 {
712 ui32 = crHashtableNumElements(pProgram->activeState.attachedShaders);
713 rc = SSMR3PutU32(pSSM, ui32);
714 CRASSERT(rc == VINF_SUCCESS);
715 crHashtableWalk(pProgram->currentState.attachedShaders, crStateSaveGLSLShaderCB, pSSM);
716 }
717
718 crStateSaveGLSLProgramAttribs(&pProgram->currentState, pSSM);
719 crStateSaveGLSLProgramAttribs(&pProgram->activeState, pSSM);
720
721 diff_api.GetProgramiv(pProgram->hwid, GL_ACTIVE_UNIFORM_MAX_LENGTH, &maxUniformLen);
722 diff_api.GetProgramiv(pProgram->hwid, GL_ACTIVE_UNIFORMS, &activeUniforms);
723
724 if (activeUniforms>0)
725 {
726 name = (GLchar *) crAlloc((maxUniformLen+8)*sizeof(GLchar));
727
728 if (!name)
729 {
730 crWarning("crStateSaveGLSLProgramCB: out of memory");
731 return;
732 }
733 }
734
735 for (i=0; i<activeUniforms; ++i)
736 {
737 diff_api.GetActiveUniform(pProgram->hwid, i, maxUniformLen, NULL, &size, &type, name);
738 uniformsCount += size;
739 }
740 CRASSERT(uniformsCount>=activeUniforms);
741
742 rc = SSMR3PutS32(pSSM, uniformsCount);
743 CRASSERT(rc == VINF_SUCCESS);
744
745 if (activeUniforms>0)
746 {
747 GLfloat fdata[16];
748 GLint idata[16];
749 char *pIndexStr=NULL;
750
751 for (i=0; i<activeUniforms; ++i)
752 {
753 diff_api.GetActiveUniform(pProgram->hwid, i, maxUniformLen, NULL, &size, &type, name);
754
755 if (size>1)
756 {
757 pIndexStr = crStrchr(name, '[');
758 if (!pIndexStr)
759 {
760 pIndexStr = name+crStrlen(name);
761 }
762 }
763
764 for (j=0; j<size; ++j)
765 {
766 if (size>1)
767 {
768 sprintf(pIndexStr, "[%i]", j);
769 location = diff_api.GetUniformLocation(pProgram->hwid, name);
770 }
771 else
772 {
773 location = i;
774 }
775
776 rc = SSMR3PutMem(pSSM, &type, sizeof(type));
777 CRASSERT(rc == VINF_SUCCESS);
778
779 crStateSaveString(name, pSSM);
780
781 if (crStateIsIntUniform(type))
782 {
783 diff_api.GetUniformiv(pProgram->hwid, location, &idata[0]);
784 rc = SSMR3PutMem(pSSM, &idata[0], crStateGetUniformSize(type)*sizeof(idata[0]));
785 CRASSERT(rc == VINF_SUCCESS);
786 }
787 else
788 {
789 diff_api.GetUniformfv(pProgram->hwid, location, &fdata[0]);
790 rc = SSMR3PutMem(pSSM, &fdata[0], crStateGetUniformSize(type)*sizeof(fdata[0]));
791 CRASSERT(rc == VINF_SUCCESS);
792 }
793 }
794 }
795
796 crFree(name);
797 }
798}
799
800int32_t crStateSaveContext(CRContext *pContext, PSSMHANDLE pSSM)
801{
802 int32_t rc, i;
803 uint32_t ui32, j;
804
805 CRASSERT(pContext && pSSM);
806
807 rc = SSMR3PutMem(pSSM, pContext, sizeof(*pContext));
808 AssertRCReturn(rc, rc);
809
810 if (crHashtableNumElements(pContext->shared->dlistTable)>0)
811 crWarning("Saving state with %d display lists, unsupported", crHashtableNumElements(pContext->shared->dlistTable));
812
813 if (crHashtableNumElements(pContext->program.programHash)>0)
814 crDebug("Saving state with %d programs", crHashtableNumElements(pContext->program.programHash));
815
816 /* Save transform state */
817 rc = SSMR3PutMem(pSSM, pContext->transform.clipPlane, sizeof(GLvectord)*CR_MAX_CLIP_PLANES);
818 AssertRCReturn(rc, rc);
819 rc = SSMR3PutMem(pSSM, pContext->transform.clip, sizeof(GLboolean)*CR_MAX_CLIP_PLANES);
820 AssertRCReturn(rc, rc);
821 rc = crStateSaveMatrixStack(&pContext->transform.modelViewStack, pSSM);
822 AssertRCReturn(rc, rc);
823 rc = crStateSaveMatrixStack(&pContext->transform.projectionStack, pSSM);
824 AssertRCReturn(rc, rc);
825 rc = crStateSaveMatrixStack(&pContext->transform.colorStack, pSSM);
826 AssertRCReturn(rc, rc);
827 for (i = 0 ; i < CR_MAX_TEXTURE_UNITS ; i++)
828 {
829 rc = crStateSaveMatrixStack(&pContext->transform.textureStack[i], pSSM);
830 AssertRCReturn(rc, rc);
831 }
832 for (i = 0 ; i < CR_MAX_PROGRAM_MATRICES ; i++)
833 {
834 rc = crStateSaveMatrixStack(&pContext->transform.programStack[i], pSSM);
835 AssertRCReturn(rc, rc);
836 }
837
838 /* Save textures */
839 rc = crStateSaveTextureObjData(&pContext->texture.base1D, pSSM);
840 AssertRCReturn(rc, rc);
841 rc = crStateSaveTextureObjData(&pContext->texture.base2D, pSSM);
842 AssertRCReturn(rc, rc);
843 rc = crStateSaveTextureObjData(&pContext->texture.base3D, pSSM);
844 AssertRCReturn(rc, rc);
845 rc = crStateSaveTextureObjData(&pContext->texture.proxy1D, pSSM);
846 AssertRCReturn(rc, rc);
847 rc = crStateSaveTextureObjData(&pContext->texture.proxy2D, pSSM);
848 AssertRCReturn(rc, rc);
849 rc = crStateSaveTextureObjData(&pContext->texture.proxy3D, pSSM);
850#ifdef CR_ARB_texture_cube_map
851 rc = crStateSaveTextureObjData(&pContext->texture.baseCubeMap, pSSM);
852 AssertRCReturn(rc, rc);
853 rc = crStateSaveTextureObjData(&pContext->texture.proxyCubeMap, pSSM);
854 AssertRCReturn(rc, rc);
855#endif
856#ifdef CR_NV_texture_rectangle
857 rc = crStateSaveTextureObjData(&pContext->texture.baseRect, pSSM);
858 AssertRCReturn(rc, rc);
859 rc = crStateSaveTextureObjData(&pContext->texture.proxyRect, pSSM);
860 AssertRCReturn(rc, rc);
861#endif
862
863 /* Save shared textures */
864 CRASSERT(pContext->shared && pContext->shared->textureTable);
865 ui32 = crHashtableNumElements(pContext->shared->textureTable);
866 rc = SSMR3PutU32(pSSM, ui32);
867 AssertRCReturn(rc, rc);
868 crHashtableWalk(pContext->shared->textureTable, crStateSaveSharedTextureCB, pSSM);
869
870#ifdef CR_STATE_NO_TEXTURE_IMAGE_STORE
871 /* Restore previous texture bindings via diff_api */
872 if (ui32)
873 {
874 CRTextureUnit *pTexUnit;
875
876 pTexUnit = &pContext->texture.unit[pContext->texture.curTextureUnit];
877
878 diff_api.BindTexture(GL_TEXTURE_1D, pTexUnit->currentTexture1D->name);
879 diff_api.BindTexture(GL_TEXTURE_2D, pTexUnit->currentTexture2D->name);
880 diff_api.BindTexture(GL_TEXTURE_3D, pTexUnit->currentTexture3D->name);
881#ifdef CR_ARB_texture_cube_map
882 diff_api.BindTexture(GL_TEXTURE_CUBE_MAP_ARB, pTexUnit->currentTextureCubeMap->name);
883#endif
884#ifdef CR_NV_texture_rectangle
885 diff_api.BindTexture(GL_TEXTURE_RECTANGLE_NV, pTexUnit->currentTextureRect->name);
886#endif
887 }
888#endif
889
890 /* Save current texture pointers */
891 for (i=0; i<CR_MAX_TEXTURE_UNITS; ++i)
892 {
893 rc = crStateSaveTexUnitCurrentTexturePtrs(&pContext->texture.unit[i], pSSM);
894 AssertRCReturn(rc, rc);
895 }
896
897 /* Save lights */
898 CRASSERT(pContext->lighting.light);
899 rc = SSMR3PutMem(pSSM, pContext->lighting.light, CR_MAX_LIGHTS * sizeof(*pContext->lighting.light));
900 AssertRCReturn(rc, rc);
901
902 /* Save attrib stack*/
903 /*@todo could go up to used stack depth here?*/
904 for ( i = 0 ; i < CR_MAX_ATTRIB_STACK_DEPTH ; i++)
905 {
906 if (pContext->attrib.enableStack[i].clip)
907 {
908 rc = SSMR3PutMem(pSSM, pContext->attrib.enableStack[i].clip,
909 pContext->limits.maxClipPlanes*sizeof(GLboolean));
910 AssertRCReturn(rc, rc);
911 }
912
913 if (pContext->attrib.enableStack[i].light)
914 {
915 rc = SSMR3PutMem(pSSM, pContext->attrib.enableStack[i].light,
916 pContext->limits.maxLights*sizeof(GLboolean));
917 AssertRCReturn(rc, rc);
918 }
919
920 if (pContext->attrib.lightingStack[i].light)
921 {
922 rc = SSMR3PutMem(pSSM, pContext->attrib.lightingStack[i].light,
923 pContext->limits.maxLights*sizeof(CRLight));
924 AssertRCReturn(rc, rc);
925 }
926
927 for (j=0; j<pContext->limits.maxTextureUnits; ++j)
928 {
929 rc = crStateSaveTexUnitCurrentTexturePtrs(&pContext->attrib.textureStack[i].unit[j], pSSM);
930 AssertRCReturn(rc, rc);
931 }
932
933 if (pContext->attrib.transformStack[i].clip)
934 {
935 rc = SSMR3PutMem(pSSM, pContext->attrib.transformStack[i].clip,
936 pContext->limits.maxClipPlanes*sizeof(GLboolean));
937 AssertRCReturn(rc, rc);
938 }
939
940 if (pContext->attrib.transformStack[i].clipPlane)
941 {
942 rc = SSMR3PutMem(pSSM, pContext->attrib.transformStack[i].clipPlane,
943 pContext->limits.maxClipPlanes*sizeof(GLvectord));
944 AssertRCReturn(rc, rc);
945 }
946
947 rc = crSateSaveEvalCoeffs1D(pContext->attrib.evalStack[i].eval1D, pSSM);
948 AssertRCReturn(rc, rc);
949 rc = crSateSaveEvalCoeffs2D(pContext->attrib.evalStack[i].eval2D, pSSM);
950 AssertRCReturn(rc, rc);
951 }
952
953 /* Save evaluator coeffs */
954 rc = crSateSaveEvalCoeffs1D(pContext->eval.eval1D, pSSM);
955 AssertRCReturn(rc, rc);
956 rc = crSateSaveEvalCoeffs2D(pContext->eval.eval2D, pSSM);
957 AssertRCReturn(rc, rc);
958
959#ifdef CR_ARB_vertex_buffer_object
960 /* Save buffer objects */
961 ui32 = crHashtableNumElements(pContext->bufferobject.buffers);
962 rc = SSMR3PutU32(pSSM, ui32);
963 AssertRCReturn(rc, rc);
964 /* Save default one*/
965 crStateSaveBufferObjectCB(0, pContext->bufferobject.nullBuffer, pSSM);
966 /* Save all the rest */
967 crHashtableWalk(pContext->bufferobject.buffers, crStateSaveBufferObjectCB, pSSM);
968 /* Restore binding */
969 diff_api.BindBufferARB(GL_ARRAY_BUFFER_ARB, pContext->bufferobject.arrayBuffer->name);
970 /* Save pointers */
971 rc = SSMR3PutU32(pSSM, pContext->bufferobject.arrayBuffer->name);
972 AssertRCReturn(rc, rc);
973 rc = SSMR3PutU32(pSSM, pContext->bufferobject.elementsBuffer->name);
974 AssertRCReturn(rc, rc);
975 /* Save bound array buffers*/ /*@todo vertexArrayStack*/
976 rc = SSMR3PutU32(pSSM, pContext->client.array.v.buffer->name);
977 AssertRCReturn(rc, rc);
978 rc = SSMR3PutU32(pSSM, pContext->client.array.c.buffer->name);
979 AssertRCReturn(rc, rc);
980 rc = SSMR3PutU32(pSSM, pContext->client.array.f.buffer->name);
981 AssertRCReturn(rc, rc);
982 rc = SSMR3PutU32(pSSM, pContext->client.array.s.buffer->name);
983 AssertRCReturn(rc, rc);
984 rc = SSMR3PutU32(pSSM, pContext->client.array.e.buffer->name);
985 AssertRCReturn(rc, rc);
986 rc = SSMR3PutU32(pSSM, pContext->client.array.i.buffer->name);
987 AssertRCReturn(rc, rc);
988 rc = SSMR3PutU32(pSSM, pContext->client.array.n.buffer->name);
989 AssertRCReturn(rc, rc);
990 for (i = 0; i < CR_MAX_TEXTURE_UNITS; i++)
991 {
992 rc = SSMR3PutU32(pSSM, pContext->client.array.t[i].buffer->name);
993 AssertRCReturn(rc, rc);
994 }
995# ifdef CR_NV_vertex_program
996 for (i = 0; i < CR_MAX_VERTEX_ATTRIBS; i++)
997 {
998 rc = SSMR3PutU32(pSSM, pContext->client.array.a[i].buffer->name);
999 AssertRCReturn(rc, rc);
1000 }
1001# endif
1002#endif /*CR_ARB_vertex_buffer_object*/
1003
1004 /* Save pixel/vertex programs */
1005 ui32 = crHashtableNumElements(pContext->program.programHash);
1006 rc = SSMR3PutU32(pSSM, ui32);
1007 AssertRCReturn(rc, rc);
1008 /* Save defauls programs */
1009 crStateSaveProgramCB(0, pContext->program.defaultVertexProgram, pSSM);
1010 crStateSaveProgramCB(0, pContext->program.defaultFragmentProgram, pSSM);
1011 /* Save all the rest */
1012 crHashtableWalk(pContext->program.programHash, crStateSaveProgramCB, pSSM);
1013 /* Save Pointers */
1014 rc = SSMR3PutU32(pSSM, pContext->program.currentVertexProgram->id);
1015 AssertRCReturn(rc, rc);
1016 rc = SSMR3PutU32(pSSM, pContext->program.currentFragmentProgram->id);
1017 AssertRCReturn(rc, rc);
1018 /* This one is unused it seems*/
1019 CRASSERT(!pContext->program.errorString);
1020
1021#ifdef CR_EXT_framebuffer_object
1022 /* Save FBOs */
1023 ui32 = crHashtableNumElements(pContext->framebufferobject.framebuffers);
1024 rc = SSMR3PutU32(pSSM, ui32);
1025 AssertRCReturn(rc, rc);
1026 crHashtableWalk(pContext->framebufferobject.framebuffers, crStateSaveFramebuffersCB, pSSM);
1027 ui32 = crHashtableNumElements(pContext->framebufferobject.renderbuffers);
1028 rc = SSMR3PutU32(pSSM, ui32);
1029 AssertRCReturn(rc, rc);
1030 crHashtableWalk(pContext->framebufferobject.renderbuffers, crStateSaveRenderbuffersCB, pSSM);
1031 rc = SSMR3PutU32(pSSM, pContext->framebufferobject.drawFB?pContext->framebufferobject.drawFB->id:0);
1032 AssertRCReturn(rc, rc);
1033 rc = SSMR3PutU32(pSSM, pContext->framebufferobject.readFB?pContext->framebufferobject.readFB->id:0);
1034 AssertRCReturn(rc, rc);
1035 rc = SSMR3PutU32(pSSM, pContext->framebufferobject.renderbuffer?pContext->framebufferobject.renderbuffer->id:0);
1036 AssertRCReturn(rc, rc);
1037#endif
1038
1039#ifdef CR_OPENGL_VERSION_2_0
1040 /* Save GLSL related info */
1041 ui32 = crHashtableNumElements(pContext->glsl.shaders);
1042 rc = SSMR3PutU32(pSSM, ui32);
1043 AssertRCReturn(rc, rc);
1044 crHashtableWalk(pContext->glsl.shaders, crStateSaveGLSLShaderCB, pSSM);
1045 ui32 = crHashtableNumElements(pContext->glsl.programs);
1046 rc = SSMR3PutU32(pSSM, ui32);
1047 AssertRCReturn(rc, rc);
1048 crHashtableWalk(pContext->glsl.programs, crStateSaveGLSLProgramCB, pSSM);
1049 rc = SSMR3PutU32(pSSM, pContext->glsl.activeProgram?pContext->glsl.activeProgram->id:0);
1050 AssertRCReturn(rc, rc);
1051#endif
1052
1053 {
1054 CRViewportState *pVP = &pContext->viewport;
1055 CRPixelPackState packing = pContext->client.pack;
1056 GLint cbData = crPixelSize(GL_RGBA, GL_UNSIGNED_BYTE) * pVP->viewportH * pVP->viewportW;
1057 void *pData = crAlloc(cbData);
1058
1059 if (!pData)
1060 {
1061 return VERR_NO_MEMORY;
1062 }
1063
1064 diff_api.PixelStorei(GL_PACK_SKIP_ROWS, 0);
1065 diff_api.PixelStorei(GL_PACK_SKIP_PIXELS, 0);
1066 diff_api.PixelStorei(GL_PACK_ALIGNMENT, 1);
1067 diff_api.PixelStorei(GL_PACK_ROW_LENGTH, 0);
1068 diff_api.PixelStorei(GL_PACK_IMAGE_HEIGHT, 0);
1069 diff_api.PixelStorei(GL_PACK_SKIP_IMAGES, 0);
1070 diff_api.PixelStorei(GL_PACK_SWAP_BYTES, 0);
1071 diff_api.PixelStorei(GL_PACK_LSB_FIRST, 0);
1072
1073 diff_api.ReadPixels(0, 0, pVP->viewportW, pVP->viewportH, GL_RGBA, GL_UNSIGNED_BYTE, pData);
1074
1075 diff_api.PixelStorei(GL_PACK_SKIP_ROWS, packing.skipRows);
1076 diff_api.PixelStorei(GL_PACK_SKIP_PIXELS, packing.skipPixels);
1077 diff_api.PixelStorei(GL_PACK_ALIGNMENT, packing.alignment);
1078 diff_api.PixelStorei(GL_PACK_ROW_LENGTH, packing.rowLength);
1079 diff_api.PixelStorei(GL_PACK_IMAGE_HEIGHT, packing.imageHeight);
1080 diff_api.PixelStorei(GL_PACK_SKIP_IMAGES, packing.skipImages);
1081 diff_api.PixelStorei(GL_PACK_SWAP_BYTES, packing.swapBytes);
1082 diff_api.PixelStorei(GL_PACK_LSB_FIRST, packing.psLSBFirst);
1083
1084 rc = SSMR3PutMem(pSSM, pData, cbData);
1085 AssertRCReturn(rc, rc);
1086
1087 crFree(pData);
1088 }
1089
1090 return VINF_SUCCESS;
1091}
1092
1093#define SLC_COPYPTR(ptr) pTmpContext->ptr = pContext->ptr
1094#define SLC_ASSSERT_NULL_PTR(ptr) CRASSERT(!pContext->ptr)
1095
1096int32_t crStateLoadContext(CRContext *pContext, PSSMHANDLE pSSM)
1097{
1098 CRContext* pTmpContext;
1099 int32_t rc, i, j;
1100 uint32_t uiNumElems, ui, k;
1101 unsigned long key;
1102
1103 CRASSERT(pContext && pSSM);
1104
1105 /* This one is rather big for stack allocation and causes macs to crash */
1106 pTmpContext = (CRContext*)crAlloc(sizeof(*pTmpContext));
1107 if (!pTmpContext)
1108 return VERR_NO_MEMORY;
1109
1110 rc = SSMR3GetMem(pSSM, pTmpContext, sizeof(*pTmpContext));
1111 AssertRCReturn(rc, rc);
1112
1113 SLC_COPYPTR(shared);
1114 SLC_COPYPTR(flush_func);
1115 SLC_COPYPTR(flush_arg);
1116
1117 /* We're supposed to be loading into an empty context, so those pointers should be NULL */
1118 for ( i = 0 ; i < CR_MAX_ATTRIB_STACK_DEPTH ; i++)
1119 {
1120 SLC_ASSSERT_NULL_PTR(attrib.enableStack[i].clip);
1121 SLC_ASSSERT_NULL_PTR(attrib.enableStack[i].light);
1122
1123 SLC_ASSSERT_NULL_PTR(attrib.lightingStack[i].light);
1124 SLC_ASSSERT_NULL_PTR(attrib.transformStack[i].clip);
1125 SLC_ASSSERT_NULL_PTR(attrib.transformStack[i].clipPlane);
1126
1127 for (j=0; j<GLEVAL_TOT; ++j)
1128 {
1129 SLC_ASSSERT_NULL_PTR(attrib.evalStack[i].eval1D[j].coeff);
1130 SLC_ASSSERT_NULL_PTR(attrib.evalStack[i].eval2D[j].coeff);
1131 }
1132 }
1133
1134#ifdef CR_ARB_vertex_buffer_object
1135 SLC_COPYPTR(bufferobject.buffers);
1136 SLC_COPYPTR(bufferobject.nullBuffer);
1137#endif
1138
1139/*@todo, that should be removed probably as those should hold the offset values, so loading should be fine
1140 but better check*/
1141#if 0
1142#ifdef CR_EXT_compiled_vertex_array
1143 SLC_COPYPTR(client.array.v.prevPtr);
1144 SLC_COPYPTR(client.array.c.prevPtr);
1145 SLC_COPYPTR(client.array.f.prevPtr);
1146 SLC_COPYPTR(client.array.s.prevPtr);
1147 SLC_COPYPTR(client.array.e.prevPtr);
1148 SLC_COPYPTR(client.array.i.prevPtr);
1149 SLC_COPYPTR(client.array.n.prevPtr);
1150 for (i = 0 ; i < CR_MAX_TEXTURE_UNITS ; i++)
1151 {
1152 SLC_COPYPTR(client.array.t[i].prevPtr);
1153 }
1154
1155# ifdef CR_NV_vertex_program
1156 for (i = 0; i < CR_MAX_VERTEX_ATTRIBS; i++)
1157 {
1158 SLC_COPYPTR(client.array.a[i].prevPtr);
1159 }
1160# endif
1161#endif
1162#endif
1163
1164#ifdef CR_ARB_vertex_buffer_object
1165 /*That just sets those pointers to NULL*/
1166 SLC_COPYPTR(client.array.v.buffer);
1167 SLC_COPYPTR(client.array.c.buffer);
1168 SLC_COPYPTR(client.array.f.buffer);
1169 SLC_COPYPTR(client.array.s.buffer);
1170 SLC_COPYPTR(client.array.e.buffer);
1171 SLC_COPYPTR(client.array.i.buffer);
1172 SLC_COPYPTR(client.array.n.buffer);
1173 for (i = 0 ; i < CR_MAX_TEXTURE_UNITS ; i++)
1174 {
1175 SLC_COPYPTR(client.array.t[i].buffer);
1176 }
1177# ifdef CR_NV_vertex_program
1178 for (i = 0; i < CR_MAX_VERTEX_ATTRIBS; i++)
1179 {
1180 SLC_COPYPTR(client.array.a[i].buffer);
1181 }
1182# endif
1183#endif /*CR_ARB_vertex_buffer_object*/
1184
1185 /*@todo CR_NV_vertex_program*/
1186 crStateCopyEvalPtrs1D(pTmpContext->eval.eval1D, pContext->eval.eval1D);
1187 crStateCopyEvalPtrs2D(pTmpContext->eval.eval2D, pContext->eval.eval2D);
1188
1189 SLC_COPYPTR(feedback.buffer); /*@todo*/
1190 SLC_COPYPTR(selection.buffer); /*@todo*/
1191
1192 SLC_COPYPTR(lighting.light);
1193
1194 /*This one could be tricky if we're loading snapshot on host with different GPU*/
1195 SLC_COPYPTR(limits.extensions);
1196
1197#if CR_ARB_occlusion_query
1198 SLC_COPYPTR(occlusion.objects); /*@todo*/
1199#endif
1200
1201 SLC_COPYPTR(program.errorString);
1202 SLC_COPYPTR(program.programHash);
1203 SLC_COPYPTR(program.defaultVertexProgram);
1204 SLC_COPYPTR(program.defaultFragmentProgram);
1205
1206 /* Texture pointers */
1207 for (i=0; i<6; ++i)
1208 {
1209 SLC_COPYPTR(texture.base1D.level[i]);
1210 SLC_COPYPTR(texture.base2D.level[i]);
1211 SLC_COPYPTR(texture.base3D.level[i]);
1212 SLC_COPYPTR(texture.proxy1D.level[i]);
1213 SLC_COPYPTR(texture.proxy2D.level[i]);
1214 SLC_COPYPTR(texture.proxy3D.level[i]);
1215#ifdef CR_ARB_texture_cube_map
1216 SLC_COPYPTR(texture.baseCubeMap.level[i]);
1217 SLC_COPYPTR(texture.proxyCubeMap.level[i]);
1218#endif
1219#ifdef CR_NV_texture_rectangle
1220 SLC_COPYPTR(texture.baseRect.level[i]);
1221 SLC_COPYPTR(texture.proxyRect.level[i]);
1222#endif
1223 }
1224
1225 /* Load transform state */
1226 SLC_COPYPTR(transform.clipPlane);
1227 SLC_COPYPTR(transform.clip);
1228 /* Don't have to worry about pContext->transform.current as it'd be set in crStateSetCurrent call */
1229 /*SLC_COPYPTR(transform.currentStack);*/
1230 SLC_COPYPTR(transform.modelViewStack.stack);
1231 SLC_COPYPTR(transform.projectionStack.stack);
1232 SLC_COPYPTR(transform.colorStack.stack);
1233
1234 for (i = 0 ; i < CR_MAX_TEXTURE_UNITS ; i++)
1235 {
1236 SLC_COPYPTR(transform.textureStack[i].stack);
1237 }
1238 for (i = 0 ; i < CR_MAX_PROGRAM_MATRICES ; i++)
1239 {
1240 SLC_COPYPTR(transform.programStack[i].stack);
1241 }
1242
1243#ifdef CR_EXT_framebuffer_object
1244 SLC_COPYPTR(framebufferobject.framebuffers);
1245 SLC_COPYPTR(framebufferobject.renderbuffers);
1246#endif
1247
1248#ifdef CR_OPENGL_VERSION_2_0
1249 SLC_COPYPTR(glsl.shaders);
1250 SLC_COPYPTR(glsl.programs);
1251#endif
1252
1253 /* Have to preserve original context id */
1254 CRASSERT(pTmpContext->id == pContext->id);
1255 /* Copy ordinary state to real context */
1256 crMemcpy(pContext, pTmpContext, sizeof(*pTmpContext));
1257 crFree(pTmpContext);
1258 pTmpContext = NULL;
1259
1260 /* Now deal with pointers */
1261
1262 /* Load transform state */
1263 rc = SSMR3GetMem(pSSM, pContext->transform.clipPlane, sizeof(GLvectord)*CR_MAX_CLIP_PLANES);
1264 AssertRCReturn(rc, rc);
1265 rc = SSMR3GetMem(pSSM, pContext->transform.clip, sizeof(GLboolean)*CR_MAX_CLIP_PLANES);
1266 AssertRCReturn(rc, rc);
1267 rc = crStateLoadMatrixStack(&pContext->transform.modelViewStack, pSSM);
1268 AssertRCReturn(rc, rc);
1269 rc = crStateLoadMatrixStack(&pContext->transform.projectionStack, pSSM);
1270 AssertRCReturn(rc, rc);
1271 rc = crStateLoadMatrixStack(&pContext->transform.colorStack, pSSM);
1272 AssertRCReturn(rc, rc);
1273 for (i = 0 ; i < CR_MAX_TEXTURE_UNITS ; i++)
1274 {
1275 rc = crStateLoadMatrixStack(&pContext->transform.textureStack[i], pSSM);
1276 AssertRCReturn(rc, rc);
1277 }
1278 for (i = 0 ; i < CR_MAX_PROGRAM_MATRICES ; i++)
1279 {
1280 rc = crStateLoadMatrixStack(&pContext->transform.programStack[i], pSSM);
1281 AssertRCReturn(rc, rc);
1282 }
1283
1284 /* Load Textures */
1285 rc = crStateLoadTextureObjData(&pContext->texture.base1D, pSSM);
1286 AssertRCReturn(rc, rc);
1287 rc = crStateLoadTextureObjData(&pContext->texture.base2D, pSSM);
1288 AssertRCReturn(rc, rc);
1289 rc = crStateLoadTextureObjData(&pContext->texture.base3D, pSSM);
1290 AssertRCReturn(rc, rc);
1291 rc = crStateLoadTextureObjData(&pContext->texture.proxy1D, pSSM);
1292 AssertRCReturn(rc, rc);
1293 rc = crStateLoadTextureObjData(&pContext->texture.proxy2D, pSSM);
1294 AssertRCReturn(rc, rc);
1295 rc = crStateLoadTextureObjData(&pContext->texture.proxy3D, pSSM);
1296 AssertRCReturn(rc, rc);
1297#ifdef CR_ARB_texture_cube_map
1298 rc = crStateLoadTextureObjData(&pContext->texture.baseCubeMap, pSSM);
1299 AssertRCReturn(rc, rc);
1300 rc = crStateLoadTextureObjData(&pContext->texture.proxyCubeMap, pSSM);
1301 AssertRCReturn(rc, rc);
1302#endif
1303#ifdef CR_NV_texture_rectangle
1304 rc = crStateLoadTextureObjData(&pContext->texture.baseRect, pSSM);
1305 AssertRCReturn(rc, rc);
1306 rc = crStateLoadTextureObjData(&pContext->texture.proxyRect, pSSM);
1307 AssertRCReturn(rc, rc);
1308#endif
1309
1310 /* Load shared textures */
1311 CRASSERT(pContext->shared && pContext->shared->textureTable);
1312 rc = SSMR3GetU32(pSSM, &uiNumElems);
1313 AssertRCReturn(rc, rc);
1314 for (ui=0; ui<uiNumElems; ++ui)
1315 {
1316 CRTextureObj *pTexture;
1317
1318 rc = SSMR3GetMem(pSSM, &key, sizeof(key));
1319 AssertRCReturn(rc, rc);
1320
1321 pTexture = (CRTextureObj *) crCalloc(sizeof(CRTextureObj));
1322 if (!pTexture) return VERR_NO_MEMORY;
1323
1324 rc = SSMR3GetMem(pSSM, pTexture, sizeof(*pTexture));
1325 AssertRCReturn(rc, rc);
1326
1327 //DIRTY(pTexture->dirty, pContext->neg_bitid);
1328 //DIRTY(pTexture->imageBit, pContext->neg_bitid);
1329
1330 /*allocate actual memory*/
1331 for (i=0; i<6; ++i) {
1332 pTexture->level[i] = (CRTextureLevel *) crCalloc(sizeof(CRTextureLevel) * MAX_MIPMAP_LEVELS);
1333 if (!pTexture->level[i]) return VERR_NO_MEMORY;
1334 }
1335
1336 rc = crStateLoadTextureObjData(pTexture, pSSM);
1337 AssertRCReturn(rc, rc);
1338
1339 crHashtableAdd(pContext->shared->textureTable, key, pTexture);
1340 }
1341
1342 /* Load current texture pointers */
1343 for (i=0; i<CR_MAX_TEXTURE_UNITS; ++i)
1344 {
1345 rc = crStateLoadTexUnitCurrentTexturePtrs(&pContext->texture.unit[i], pContext, pSSM);
1346 AssertRCReturn(rc, rc);
1347 }
1348 //FILLDIRTY(GetCurrentBits()->texture.dirty);
1349
1350 /* Mark textures for resending to GPU */
1351 pContext->texture.bResyncNeeded = GL_TRUE;
1352
1353 /* Load lights */
1354 CRASSERT(pContext->lighting.light);
1355 rc = SSMR3GetMem(pSSM, pContext->lighting.light, CR_MAX_LIGHTS * sizeof(*pContext->lighting.light));
1356 AssertRCReturn(rc, rc);
1357
1358 /* Load attrib stack*/
1359 for ( i = 0 ; i < CR_MAX_ATTRIB_STACK_DEPTH ; i++)
1360 {
1361 if (pContext->attrib.enableStack[i].clip)
1362 {
1363 rc = crStateAllocAndSSMR3GetMem(pSSM, (void**)&pContext->attrib.enableStack[i].clip,
1364 pContext->limits.maxClipPlanes*sizeof(GLboolean));
1365 AssertRCReturn(rc, rc);
1366 }
1367
1368 if (pContext->attrib.enableStack[i].light)
1369 {
1370 rc = crStateAllocAndSSMR3GetMem(pSSM, (void**)&pContext->attrib.enableStack[i].light,
1371 pContext->limits.maxLights*sizeof(GLboolean));
1372 AssertRCReturn(rc, rc);
1373 }
1374
1375 if (pContext->attrib.lightingStack[i].light)
1376 {
1377 rc = crStateAllocAndSSMR3GetMem(pSSM, (void**)&pContext->attrib.lightingStack[i].light,
1378 pContext->limits.maxLights*sizeof(CRLight));
1379 AssertRCReturn(rc, rc);
1380 }
1381
1382 for (k=0; k<pContext->limits.maxTextureUnits; ++k)
1383 {
1384 rc = crStateLoadTexUnitCurrentTexturePtrs(&pContext->attrib.textureStack[i].unit[k], pContext, pSSM);
1385 AssertRCReturn(rc, rc);
1386 }
1387
1388 if (pContext->attrib.transformStack[i].clip)
1389 {
1390 rc = crStateAllocAndSSMR3GetMem(pSSM, (void*)&pContext->attrib.transformStack[i].clip,
1391 pContext->limits.maxClipPlanes*sizeof(GLboolean));
1392 AssertRCReturn(rc, rc);
1393 }
1394
1395 if (pContext->attrib.transformStack[i].clipPlane)
1396 {
1397 rc = crStateAllocAndSSMR3GetMem(pSSM, (void**)&pContext->attrib.transformStack[i].clipPlane,
1398 pContext->limits.maxClipPlanes*sizeof(GLvectord));
1399 AssertRCReturn(rc, rc);
1400 }
1401 rc = crSateLoadEvalCoeffs1D(pContext->attrib.evalStack[i].eval1D, GL_TRUE, pSSM);
1402 AssertRCReturn(rc, rc);
1403 rc = crSateLoadEvalCoeffs2D(pContext->attrib.evalStack[i].eval2D, GL_TRUE, pSSM);
1404 AssertRCReturn(rc, rc);
1405 }
1406
1407 /* Load evaluator coeffs */
1408 rc = crSateLoadEvalCoeffs1D(pContext->eval.eval1D, GL_FALSE, pSSM);
1409 AssertRCReturn(rc, rc);
1410 rc = crSateLoadEvalCoeffs2D(pContext->eval.eval2D, GL_FALSE, pSSM);
1411 AssertRCReturn(rc, rc);
1412
1413 /* Load buffer objects */
1414#ifdef CR_ARB_vertex_buffer_object
1415 rc = SSMR3GetU32(pSSM, &uiNumElems);
1416 AssertRCReturn(rc, rc);
1417 for (ui=0; ui<=uiNumElems; ++ui) /*ui<=uiNumElems to load nullBuffer in same loop*/
1418 {
1419 CRBufferObject *pBufferObj;
1420
1421 rc = SSMR3GetMem(pSSM, &key, sizeof(key));
1422 AssertRCReturn(rc, rc);
1423
1424 /* default one should be already allocated */
1425 if (key==0)
1426 {
1427 pBufferObj = pContext->bufferobject.nullBuffer;
1428 if (!pBufferObj) return VERR_SSM_UNEXPECTED_DATA;
1429 }
1430 else
1431 {
1432 pBufferObj = (CRBufferObject *) crCalloc(sizeof(*pBufferObj));
1433 if (!pBufferObj) return VERR_NO_MEMORY;
1434 }
1435
1436 rc = SSMR3GetMem(pSSM, pBufferObj, sizeof(*pBufferObj));
1437 AssertRCReturn(rc, rc);
1438
1439 if (pBufferObj->data)
1440 {
1441 CRASSERT(pBufferObj->size>0);
1442 pBufferObj->data = crAlloc(pBufferObj->size);
1443 rc = SSMR3GetMem(pSSM, pBufferObj->data, pBufferObj->size);
1444 AssertRCReturn(rc, rc);
1445
1446 //DIRTY(pBufferObj->dirty, pContext->neg_bitid);
1447 //pBufferObj->dirtyStart = 0;
1448 //pBufferObj->dirtyLength = pBufferObj->size;
1449 }
1450 else if (pBufferObj->name!=0 && pBufferObj->size>0)
1451 {
1452 rc = SSMR3GetMem(pSSM, &pBufferObj->data, sizeof(pBufferObj->data));
1453 AssertRCReturn(rc, rc);
1454
1455 if (pBufferObj->data)
1456 {
1457 pBufferObj->data = crAlloc(pBufferObj->size);
1458 rc = SSMR3GetMem(pSSM, pBufferObj->data, pBufferObj->size);
1459 AssertRCReturn(rc, rc);
1460 }
1461 }
1462
1463
1464 if (key!=0)
1465 crHashtableAdd(pContext->bufferobject.buffers, key, pBufferObj);
1466 }
1467 //FILLDIRTY(GetCurrentBits()->bufferobject.dirty);
1468 /* Load pointers */
1469#define CRS_GET_BO(name) (((name)==0) ? (pContext->bufferobject.nullBuffer) : crHashtableSearch(pContext->bufferobject.buffers, name))
1470 rc = SSMR3GetU32(pSSM, &ui);
1471 AssertRCReturn(rc, rc);
1472 pContext->bufferobject.arrayBuffer = CRS_GET_BO(ui);
1473 rc = SSMR3GetU32(pSSM, &ui);
1474 AssertRCReturn(rc, rc);
1475 pContext->bufferobject.elementsBuffer = CRS_GET_BO(ui);
1476
1477 /* Load bound array buffers*/
1478 rc = SSMR3GetU32(pSSM, &ui);
1479 AssertRCReturn(rc, rc);
1480 pContext->client.array.v.buffer = CRS_GET_BO(ui);
1481
1482 rc = SSMR3GetU32(pSSM, &ui);
1483 AssertRCReturn(rc, rc);
1484 pContext->client.array.c.buffer = CRS_GET_BO(ui);
1485
1486 rc = SSMR3GetU32(pSSM, &ui);
1487 AssertRCReturn(rc, rc);
1488 pContext->client.array.f.buffer = CRS_GET_BO(ui);
1489
1490 rc = SSMR3GetU32(pSSM, &ui);
1491 AssertRCReturn(rc, rc);
1492 pContext->client.array.s.buffer = CRS_GET_BO(ui);
1493
1494 rc = SSMR3GetU32(pSSM, &ui);
1495 AssertRCReturn(rc, rc);
1496 pContext->client.array.e.buffer = CRS_GET_BO(ui);
1497
1498 rc = SSMR3GetU32(pSSM, &ui);
1499 AssertRCReturn(rc, rc);
1500 pContext->client.array.i.buffer = CRS_GET_BO(ui);
1501
1502 rc = SSMR3GetU32(pSSM, &ui);
1503 AssertRCReturn(rc, rc);
1504 pContext->client.array.n.buffer = CRS_GET_BO(ui);
1505
1506 for (i = 0; i < CR_MAX_TEXTURE_UNITS; i++)
1507 {
1508 rc = SSMR3GetU32(pSSM, &ui);
1509 AssertRCReturn(rc, rc);
1510 pContext->client.array.t[i].buffer = CRS_GET_BO(ui);
1511 }
1512# ifdef CR_NV_vertex_program
1513 for (i = 0; i < CR_MAX_VERTEX_ATTRIBS; i++)
1514 {
1515 rc = SSMR3GetU32(pSSM, &ui);
1516 AssertRCReturn(rc, rc);
1517 pContext->client.array.a[i].buffer = CRS_GET_BO(ui);
1518 }
1519# endif
1520#undef CRS_GET_BO
1521
1522 pContext->bufferobject.bResyncNeeded = GL_TRUE;
1523#endif
1524
1525 /* Load pixel/vertex programs */
1526 rc = SSMR3GetU32(pSSM, &uiNumElems);
1527 AssertRCReturn(rc, rc);
1528 /* Load defauls programs */
1529 rc = crStateLoadProgram(&pContext->program.defaultVertexProgram, pSSM);
1530 AssertRCReturn(rc, rc);
1531 //FILLDIRTY(pContext->program.defaultVertexProgram->dirtyProgram);
1532 rc = crStateLoadProgram(&pContext->program.defaultFragmentProgram, pSSM);
1533 AssertRCReturn(rc, rc);
1534 //FILLDIRTY(pContext->program.defaultFragmentProgram->dirtyProgram);
1535 /* Load all the rest */
1536 for (ui=0; ui<uiNumElems; ++ui)
1537 {
1538 CRProgram *pProgram = NULL;
1539 rc = crStateLoadProgram(&pProgram, pSSM);
1540 AssertRCReturn(rc, rc);
1541 crHashtableAdd(pContext->program.programHash, pProgram->id, pProgram);
1542 //DIRTY(pProgram->dirtyProgram, pContext->neg_bitid);
1543
1544 }
1545 //FILLDIRTY(GetCurrentBits()->program.dirty);
1546 /* Load Pointers */
1547 rc = SSMR3GetU32(pSSM, &ui);
1548 AssertRCReturn(rc, rc);
1549 pContext->program.currentVertexProgram = ui==0 ? pContext->program.defaultVertexProgram
1550 : crHashtableSearch(pContext->program.programHash, ui);
1551 rc = SSMR3GetU32(pSSM, &ui);
1552 AssertRCReturn(rc, rc);
1553 pContext->program.currentFragmentProgram = ui==0 ? pContext->program.defaultFragmentProgram
1554 : crHashtableSearch(pContext->program.programHash, ui);
1555
1556 /* Mark programs for resending to GPU */
1557 pContext->program.bResyncNeeded = GL_TRUE;
1558
1559#ifdef CR_EXT_framebuffer_object
1560 /* Load FBOs */
1561 rc = SSMR3GetU32(pSSM, &uiNumElems);
1562 AssertRCReturn(rc, rc);
1563 for (ui=0; ui<uiNumElems; ++ui)
1564 {
1565 CRFramebufferObject *pFBO;
1566 pFBO = crAlloc(sizeof(*pFBO));
1567 if (!pFBO) return VERR_NO_MEMORY;
1568
1569 rc = SSMR3GetMem(pSSM, &key, sizeof(key));
1570 AssertRCReturn(rc, rc);
1571
1572 rc = SSMR3GetMem(pSSM, pFBO, sizeof(*pFBO));
1573 AssertRCReturn(rc, rc);
1574
1575 crHashtableAdd(pContext->framebufferobject.framebuffers, key, pFBO);
1576 }
1577
1578 rc = SSMR3GetU32(pSSM, &uiNumElems);
1579 AssertRCReturn(rc, rc);
1580 for (ui=0; ui<uiNumElems; ++ui)
1581 {
1582 CRRenderbufferObject *pRBO;
1583 pRBO = crAlloc(sizeof(*pRBO));
1584 if (!pRBO) return VERR_NO_MEMORY;
1585
1586 rc = SSMR3GetMem(pSSM, &key, sizeof(key));
1587 AssertRCReturn(rc, rc);
1588
1589 rc = SSMR3GetMem(pSSM, pRBO, sizeof(*pRBO));
1590 AssertRCReturn(rc, rc);
1591
1592 crHashtableAdd(pContext->framebufferobject.renderbuffers, key, pRBO);
1593 }
1594
1595 rc = SSMR3GetU32(pSSM, &ui);
1596 AssertRCReturn(rc, rc);
1597 pContext->framebufferobject.drawFB = ui==0 ? NULL
1598 : crHashtableSearch(pContext->framebufferobject.framebuffers, ui);
1599
1600 rc = SSMR3GetU32(pSSM, &ui);
1601 AssertRCReturn(rc, rc);
1602 pContext->framebufferobject.readFB = ui==0 ? NULL
1603 : crHashtableSearch(pContext->framebufferobject.framebuffers, ui);
1604
1605 rc = SSMR3GetU32(pSSM, &ui);
1606 AssertRCReturn(rc, rc);
1607 pContext->framebufferobject.renderbuffer = ui==0 ? NULL
1608 : crHashtableSearch(pContext->framebufferobject.renderbuffers, ui);
1609
1610 /* Mark FBOs/RBOs for resending to GPU */
1611 pContext->framebufferobject.bResyncNeeded = GL_TRUE;
1612#endif
1613
1614#ifdef CR_OPENGL_VERSION_2_0
1615 /* Load GLSL related info */
1616 rc = SSMR3GetU32(pSSM, &uiNumElems);
1617 AssertRCReturn(rc, rc);
1618
1619 for (ui=0; ui<uiNumElems; ++ui)
1620 {
1621 CRGLSLShader *pShader = crStateLoadGLSLShader(pSSM);
1622 if (!pShader) return VERR_SSM_UNEXPECTED_DATA;
1623 crHashtableAdd(pContext->glsl.shaders, pShader->id, pShader);
1624 }
1625
1626 rc = SSMR3GetU32(pSSM, &uiNumElems);
1627 AssertRCReturn(rc, rc);
1628
1629 for (ui=0; ui<uiNumElems; ++ui)
1630 {
1631 CRGLSLProgram *pProgram;
1632 uint32_t numShaders;
1633
1634 pProgram = crAlloc(sizeof(*pProgram));
1635 if (!pProgram) return VERR_NO_MEMORY;
1636
1637 rc = SSMR3GetMem(pSSM, &key, sizeof(key));
1638 AssertRCReturn(rc, rc);
1639
1640 rc = SSMR3GetMem(pSSM, pProgram, sizeof(*pProgram));
1641 AssertRCReturn(rc, rc);
1642
1643 crHashtableAdd(pContext->glsl.programs, key, pProgram);
1644
1645 pProgram->currentState.attachedShaders = crAllocHashtable();
1646
1647 rc = SSMR3GetU32(pSSM, &numShaders);
1648 AssertRCReturn(rc, rc);
1649
1650 for (k=0; k<numShaders; ++k)
1651 {
1652 rc = SSMR3GetMem(pSSM, &key, sizeof(key));
1653 AssertRCReturn(rc, rc);
1654 crHashtableAdd(pProgram->currentState.attachedShaders, key, crHashtableSearch(pContext->glsl.shaders, key));
1655 }
1656
1657 if (pProgram->activeState.attachedShaders)
1658 {
1659 pProgram->activeState.attachedShaders = crAllocHashtable();
1660
1661 rc = SSMR3GetU32(pSSM, &numShaders);
1662 AssertRCReturn(rc, rc);
1663
1664 for (k=0; k<numShaders; ++k)
1665 {
1666 CRGLSLShader *pShader = crStateLoadGLSLShader(pSSM);
1667 if (!pShader) return VERR_SSM_UNEXPECTED_DATA;
1668 crHashtableAdd(pProgram->activeState.attachedShaders, pShader->id, pShader);
1669 }
1670 }
1671
1672 if (pProgram->currentState.cAttribs)
1673 pProgram->currentState.pAttribs = (CRGLSLAttrib*) crAlloc(pProgram->currentState.cAttribs*sizeof(CRGLSLAttrib));
1674 for (k=0; k<pProgram->currentState.cAttribs; ++k)
1675 {
1676 rc = SSMR3GetMem(pSSM, &pProgram->currentState.pAttribs[k].index, sizeof(pProgram->currentState.pAttribs[k].index));
1677 AssertRCReturn(rc, rc);
1678 pProgram->currentState.pAttribs[k].name = crStateLoadString(pSSM);
1679 }
1680
1681 if (pProgram->activeState.cAttribs)
1682 pProgram->activeState.pAttribs = (CRGLSLAttrib*) crAlloc(pProgram->activeState.cAttribs*sizeof(CRGLSLAttrib));
1683 for (k=0; k<pProgram->activeState.cAttribs; ++k)
1684 {
1685 rc = SSMR3GetMem(pSSM, &pProgram->activeState.pAttribs[k].index, sizeof(pProgram->activeState.pAttribs[k].index));
1686 AssertRCReturn(rc, rc);
1687 pProgram->activeState.pAttribs[k].name = crStateLoadString(pSSM);
1688 }
1689
1690 {
1691 int32_t cUniforms;
1692 rc = SSMR3GetS32(pSSM, &cUniforms);
1693 pProgram->cUniforms = cUniforms;
1694 AssertRCReturn(rc, rc);
1695 }
1696
1697 if (pProgram->cUniforms)
1698 {
1699 pProgram->pUniforms = crAlloc(pProgram->cUniforms*sizeof(CRGLSLUniform));
1700 if (!pProgram) return VERR_NO_MEMORY;
1701
1702 for (k=0; k<pProgram->cUniforms; ++k)
1703 {
1704 size_t itemsize, datasize;
1705
1706 rc = SSMR3GetMem(pSSM, &pProgram->pUniforms[k].type, sizeof(GLenum));
1707 pProgram->pUniforms[k].name = crStateLoadString(pSSM);
1708
1709 if (crStateIsIntUniform(pProgram->pUniforms[k].type))
1710 {
1711 itemsize = sizeof(GLint);
1712 } else itemsize = sizeof(GLfloat);
1713
1714 datasize = crStateGetUniformSize(pProgram->pUniforms[k].type)*itemsize;
1715 pProgram->pUniforms[k].data = crAlloc(datasize);
1716 if (!pProgram->pUniforms[k].data) return VERR_NO_MEMORY;
1717
1718 rc = SSMR3GetMem(pSSM, pProgram->pUniforms[k].data, datasize);
1719 }
1720 }
1721 }
1722
1723 rc = SSMR3GetU32(pSSM, &ui);
1724 AssertRCReturn(rc, rc);
1725 pContext->glsl.activeProgram = ui==0 ? NULL
1726 : crHashtableSearch(pContext->glsl.programs, ui);
1727
1728 /*Mark for resending to GPU*/
1729 pContext->glsl.bResyncNeeded = GL_TRUE;
1730#endif
1731
1732 {
1733 CRViewportState *pVP = &pContext->viewport;
1734 CRPixelPackState unpack = pContext->client.unpack;
1735 GLint cbData = crPixelSize(GL_RGBA, GL_UNSIGNED_BYTE) * pVP->viewportH * pVP->viewportW;
1736 void *pData = crAlloc(cbData);
1737
1738 if (!pData)
1739 {
1740 return VERR_NO_MEMORY;
1741 }
1742
1743 rc = SSMR3GetMem(pSSM, pData, cbData);
1744 AssertRCReturn(rc, rc);
1745
1746 diff_api.PixelStorei(GL_UNPACK_SKIP_ROWS, 0);
1747 diff_api.PixelStorei(GL_UNPACK_SKIP_PIXELS, 0);
1748 diff_api.PixelStorei(GL_UNPACK_ALIGNMENT, 1);
1749 diff_api.PixelStorei(GL_UNPACK_ROW_LENGTH, 0);
1750 diff_api.PixelStorei(GL_UNPACK_IMAGE_HEIGHT, 0);
1751 diff_api.PixelStorei(GL_UNPACK_SKIP_IMAGES, 0);
1752 diff_api.PixelStorei(GL_UNPACK_SWAP_BYTES, 0);
1753 diff_api.PixelStorei(GL_UNPACK_LSB_FIRST, 0);
1754
1755 diff_api.WindowPos2iARB(0, 0);
1756 diff_api.DrawPixels(pVP->viewportW, pVP->viewportH, GL_RGBA, GL_UNSIGNED_BYTE, pData);
1757
1758 diff_api.PixelStorei(GL_UNPACK_SKIP_ROWS, unpack.skipRows);
1759 diff_api.PixelStorei(GL_UNPACK_SKIP_PIXELS, unpack.skipPixels);
1760 diff_api.PixelStorei(GL_UNPACK_ALIGNMENT, unpack.alignment);
1761 diff_api.PixelStorei(GL_UNPACK_ROW_LENGTH, unpack.rowLength);
1762 diff_api.PixelStorei(GL_UNPACK_IMAGE_HEIGHT, unpack.imageHeight);
1763 diff_api.PixelStorei(GL_UNPACK_SKIP_IMAGES, unpack.skipImages);
1764 diff_api.PixelStorei(GL_UNPACK_SWAP_BYTES, unpack.swapBytes);
1765 diff_api.PixelStorei(GL_UNPACK_LSB_FIRST, unpack.psLSBFirst);
1766
1767 crFree(pData);
1768 }
1769
1770 return VINF_SUCCESS;
1771}
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