2 * Copyright 2006 Ivan Gyurdiev
3 * Copyright 2005, 2008 Henri Verbeet
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2.1 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
22 #include "wine/test.h"
24 static DWORD texture_stages
;
26 static HWND
create_window(void)
29 wc
.lpfnWndProc
= DefWindowProc
;
30 wc
.lpszClassName
= "d3d8_test_wc";
33 return CreateWindow("d3d8_test_wc", "d3d8_test", 0, 0, 0, 0, 0, 0, 0, 0, 0);
36 static HRESULT
init_d3d8(HMODULE d3d8_module
, IDirect3DDevice8
**device
, D3DPRESENT_PARAMETERS
*device_pparams
)
38 IDirect3D8
* (WINAPI
*d3d8_create
)(UINT SDKVersion
) = 0;
44 d3d8_create
= (void *)GetProcAddress(d3d8_module
, "Direct3DCreate8");
45 if (!d3d8_create
) return E_FAIL
;
47 d3d8
= d3d8_create(D3D_SDK_VERSION
);
50 skip("Failed to create D3D8 object.\n");
54 window
= create_window();
56 IDirect3D8_GetAdapterDisplayMode(d3d8
, D3DADAPTER_DEFAULT
, &d3ddm
);
57 memset(device_pparams
, 0, sizeof(*device_pparams
));
58 device_pparams
->Windowed
= TRUE
;
59 device_pparams
->SwapEffect
= D3DSWAPEFFECT_DISCARD
;
60 device_pparams
->BackBufferFormat
= d3ddm
.Format
;
62 hr
= IDirect3D8_CreateDevice(d3d8
, D3DADAPTER_DEFAULT
, D3DDEVTYPE_HAL
, window
,
63 D3DCREATE_SOFTWARE_VERTEXPROCESSING
, device_pparams
, device
);
64 ok(SUCCEEDED(hr
) || hr
== D3DERR_NOTAVAILABLE
|| broken(hr
== D3DERR_INVALIDCALL
),
65 "IDirect3D8_CreateDevice failed, hr %#x.\n", hr
);
70 static void test_begin_end_state_block(IDirect3DDevice8
*device
)
76 hr
= IDirect3DDevice8_BeginStateBlock(device
);
77 ok(SUCCEEDED(hr
), "BeginStateBlock failed, hr %#x.\n", hr
);
78 if (FAILED(hr
)) return;
80 /* Calling BeginStateBlock while recording should return D3DERR_INVALIDCALL */
81 hr
= IDirect3DDevice8_BeginStateBlock(device
);
82 ok(hr
== D3DERR_INVALIDCALL
, "BeginStateBlock returned %#x, expected %#x.\n", hr
, D3DERR_INVALIDCALL
);
83 if (hr
!= D3DERR_INVALIDCALL
) return;
86 state_block
= 0xdeadbeef;
87 hr
= IDirect3DDevice8_EndStateBlock(device
, &state_block
);
88 ok(SUCCEEDED(hr
) && state_block
&& state_block
!= 0xdeadbeef,
89 "EndStateBlock returned: hr %#x, state_block %#x. "
90 "Expected hr %#x, state_block != 0, state_block != 0xdeadbeef.\n", hr
, state_block
, D3D_OK
);
91 IDirect3DDevice8_DeleteStateBlock(device
, state_block
);
93 /* Calling EndStateBlock while not recording should return D3DERR_INVALIDCALL.
94 * state_block should not be touched. */
95 state_block
= 0xdeadbeef;
96 hr
= IDirect3DDevice8_EndStateBlock(device
, &state_block
);
97 ok(hr
== D3DERR_INVALIDCALL
&& state_block
== 0xdeadbeef,
98 "EndStateBlock returned: hr %#x, state_block %#x. "
99 "Expected hr %#x, state_block 0xdeadbeef.\n", hr
, state_block
, D3DERR_INVALIDCALL
);
102 /* ============================ State Testing Framework ========================== */
106 const char *test_name
;
108 /* The initial data is usually the same
109 * as the default data, but a write can have side effects.
110 * The initial data is tested first, before any writes take place
111 * The default data can be tested after a write */
112 const void *initial_data
;
114 /* The default data is the standard state to compare
115 * against, and restore to */
116 const void *default_data
;
118 /* The test data is the experiment data to try
119 * in - what we want to write
120 * out - what windows will actually write (not necessarily the same) */
121 const void *test_data_in
;
122 const void *test_data_out_all
;
123 const void *test_data_out_vertex
;
124 const void *test_data_out_pixel
;
126 HRESULT (*init
)(IDirect3DDevice8
*device
, struct state_test
*test
);
127 void (*cleanup
)(IDirect3DDevice8
*device
, struct state_test
*test
);
128 void (*apply_data
)(IDirect3DDevice8
*device
, const struct state_test
*test
,
130 void (*check_data
)(IDirect3DDevice8
*device
, const struct state_test
*test
,
131 const void *data_expected
, unsigned int chain_stage
);
134 const void *test_arg
;
136 /* Test-specific context data */
140 /* See below for explanation of the flags */
142 #define EVENT_ERROR -1
147 IDirect3DSurface8
*original_render_target
;
148 IDirect3DSwapChain8
*new_swap_chain
;
164 int (*event_fn
)(IDirect3DDevice8
*device
, struct event_data
*event_data
);
165 enum stateblock_data check
;
166 enum stateblock_data apply
;
169 static const void *get_event_data(const struct state_test
*test
, enum stateblock_data data
)
173 case SB_DATA_DEFAULT
:
174 return test
->default_data
;
176 case SB_DATA_INITIAL
:
177 return test
->initial_data
;
179 case SB_DATA_TEST_IN
:
180 return test
->test_data_in
;
182 case SB_DATA_TEST_ALL
:
183 return test
->test_data_out_all
;
185 case SB_DATA_TEST_VERTEX
:
186 return test
->test_data_out_vertex
;
188 case SB_DATA_TEST_PIXEL
:
189 return test
->test_data_out_pixel
;
196 /* This is an event-machine, which tests things.
197 * It tests get and set operations for a batch of states, based on
198 * results from the event function, which directs what's to be done */
199 static void execute_test_chain(IDirect3DDevice8
*device
, struct state_test
*test
,
200 unsigned int ntests
, struct event
*event
, unsigned int nevents
, struct event_data
*event_data
)
204 /* For each queued event */
205 for (j
= 0; j
< nevents
; ++j
)
209 /* Execute the next event handler (if available). */
210 if (event
[j
].event_fn
)
212 if (event
[j
].event_fn(device
, event_data
) == EVENT_ERROR
)
214 trace("Stage %u in error state, aborting.\n", j
);
219 if (event
[j
].check
!= SB_DATA_NONE
)
221 for (i
= 0; i
< ntests
; ++i
)
223 data
= get_event_data(&test
[i
], event
[j
].check
);
224 test
[i
].check_data(device
, &test
[i
], data
, j
);
228 if (event
[j
].apply
!= SB_DATA_NONE
)
230 for (i
= 0; i
< ntests
; ++i
)
232 data
= get_event_data(&test
[i
], event
[j
].apply
);
233 test
[i
].apply_data(device
, &test
[i
], data
);
238 /* Attempt to reset any changes made. */
239 for (i
= 0; i
< ntests
; ++i
)
241 test
[i
].apply_data(device
, &test
[i
], test
[i
].default_data
);
245 static int switch_render_target(IDirect3DDevice8
* device
, struct event_data
*event_data
)
247 D3DPRESENT_PARAMETERS present_parameters
;
248 IDirect3DSwapChain8
*swapchain
= NULL
;
249 IDirect3DSurface8
*backbuffer
= NULL
;
250 D3DDISPLAYMODE d3ddm
;
253 /* Parameters for new swapchain */
254 IDirect3DDevice8_GetDisplayMode(device
, &d3ddm
);
255 memset(&present_parameters
, 0, sizeof(present_parameters
));
256 present_parameters
.Windowed
= TRUE
;
257 present_parameters
.SwapEffect
= D3DSWAPEFFECT_DISCARD
;
258 present_parameters
.BackBufferFormat
= d3ddm
.Format
;
260 /* Create new swapchain */
261 hr
= IDirect3DDevice8_CreateAdditionalSwapChain(device
, &present_parameters
, &swapchain
);
262 ok(SUCCEEDED(hr
), "CreateAdditionalSwapChain returned %#x.\n", hr
);
263 if (FAILED(hr
)) goto error
;
265 /* Get its backbuffer */
266 hr
= IDirect3DSwapChain8_GetBackBuffer(swapchain
, 0, D3DBACKBUFFER_TYPE_MONO
, &backbuffer
);
267 ok(SUCCEEDED(hr
), "GetBackBuffer returned %#x.\n", hr
);
268 if (FAILED(hr
)) goto error
;
270 /* Save the current render target */
271 hr
= IDirect3DDevice8_GetRenderTarget(device
, &event_data
->original_render_target
);
272 ok(SUCCEEDED(hr
), "GetRenderTarget returned %#x.\n", hr
);
273 if (FAILED(hr
)) goto error
;
275 /* Set the new swapchain's backbuffer as a render target */
276 hr
= IDirect3DDevice8_SetRenderTarget(device
, backbuffer
, NULL
);
277 ok(SUCCEEDED(hr
), "SetRenderTarget returned %#x.\n", hr
);
278 if (FAILED(hr
)) goto error
;
280 IDirect3DSurface8_Release(backbuffer
);
281 event_data
->new_swap_chain
= swapchain
;
285 if (backbuffer
) IDirect3DSurface8_Release(backbuffer
);
286 if (swapchain
) IDirect3DSwapChain8_Release(swapchain
);
290 static int revert_render_target(IDirect3DDevice8
*device
, struct event_data
*event_data
)
294 /* Reset the old render target */
295 hr
= IDirect3DDevice8_SetRenderTarget(device
, event_data
->original_render_target
, NULL
);
296 ok(SUCCEEDED(hr
), "SetRenderTarget returned %#x.\n", hr
);
299 IDirect3DSurface8_Release(event_data
->original_render_target
);
303 IDirect3DSurface8_Release(event_data
->original_render_target
);
304 IDirect3DSwapChain8_Release(event_data
->new_swap_chain
);
308 static int create_stateblock_all(IDirect3DDevice8
*device
, struct event_data
*event_data
)
312 hr
= IDirect3DDevice8_CreateStateBlock(device
, D3DSBT_ALL
, &event_data
->stateblock
);
313 ok(SUCCEEDED(hr
), "CreateStateBlock returned %#x.\n", hr
);
314 if (FAILED(hr
)) return EVENT_ERROR
;
318 static int create_stateblock_vertex(IDirect3DDevice8
*device
, struct event_data
*event_data
)
322 hr
= IDirect3DDevice8_CreateStateBlock(device
, D3DSBT_VERTEXSTATE
, &event_data
->stateblock
);
323 ok(SUCCEEDED(hr
), "CreateStateBlock returned %#x.\n", hr
);
324 if (FAILED(hr
)) return EVENT_ERROR
;
328 static int create_stateblock_pixel(IDirect3DDevice8
*device
, struct event_data
*event_data
)
332 hr
= IDirect3DDevice8_CreateStateBlock(device
, D3DSBT_PIXELSTATE
, &event_data
->stateblock
);
333 ok(SUCCEEDED(hr
), "CreateStateBlock returned %#x.\n", hr
);
334 if (FAILED(hr
)) return EVENT_ERROR
;
338 static int begin_stateblock(IDirect3DDevice8
*device
, struct event_data
*event_data
)
342 hr
= IDirect3DDevice8_BeginStateBlock(device
);
343 ok(SUCCEEDED(hr
), "BeginStateBlock returned %#x.\n", hr
);
344 if (FAILED(hr
)) return EVENT_ERROR
;
348 static int end_stateblock(IDirect3DDevice8
*device
, struct event_data
*event_data
)
352 hr
= IDirect3DDevice8_EndStateBlock(device
, &event_data
->stateblock
);
353 ok(SUCCEEDED(hr
), "EndStateBlock returned %#x.\n", hr
);
354 if (FAILED(hr
)) return EVENT_ERROR
;
358 static int delete_stateblock(IDirect3DDevice8
*device
, struct event_data
*event_data
)
360 IDirect3DDevice8_DeleteStateBlock(device
, event_data
->stateblock
);
364 static int apply_stateblock(IDirect3DDevice8
*device
, struct event_data
*event_data
)
368 hr
= IDirect3DDevice8_ApplyStateBlock(device
, event_data
->stateblock
);
369 ok(SUCCEEDED(hr
), "Apply returned %#x.\n", hr
);
371 IDirect3DDevice8_DeleteStateBlock(device
, event_data
->stateblock
);
372 if (FAILED(hr
)) return EVENT_ERROR
;
376 static int capture_stateblock(IDirect3DDevice8
*device
, struct event_data
*event_data
)
380 hr
= IDirect3DDevice8_CaptureStateBlock(device
, event_data
->stateblock
);
381 ok(SUCCEEDED(hr
), "Capture returned %#x.\n", hr
);
382 if (FAILED(hr
)) return EVENT_ERROR
;
387 static void execute_test_chain_all(IDirect3DDevice8
*device
, struct state_test
*test
, unsigned int ntests
)
389 struct event_data arg
;
393 struct event read_events
[] =
395 {NULL
, SB_DATA_INITIAL
, SB_DATA_NONE
},
398 struct event write_read_events
[] =
400 {NULL
, SB_DATA_NONE
, SB_DATA_TEST_IN
},
401 {NULL
, SB_DATA_TEST_ALL
, SB_DATA_NONE
},
404 struct event abort_stateblock_events
[] =
406 {begin_stateblock
, SB_DATA_NONE
, SB_DATA_TEST_IN
},
407 {end_stateblock
, SB_DATA_NONE
, SB_DATA_NONE
},
408 {delete_stateblock
, SB_DATA_DEFAULT
, SB_DATA_NONE
},
411 struct event apply_stateblock_events
[] =
413 {begin_stateblock
, SB_DATA_NONE
, SB_DATA_TEST_IN
},
414 {end_stateblock
, SB_DATA_NONE
, SB_DATA_NONE
},
415 {apply_stateblock
, SB_DATA_TEST_ALL
, SB_DATA_NONE
},
418 struct event capture_reapply_stateblock_events
[] =
420 {begin_stateblock
, SB_DATA_NONE
, SB_DATA_TEST_IN
},
421 {end_stateblock
, SB_DATA_NONE
, SB_DATA_NONE
},
422 {capture_stateblock
, SB_DATA_DEFAULT
, SB_DATA_TEST_IN
},
423 {apply_stateblock
, SB_DATA_DEFAULT
, SB_DATA_NONE
},
426 struct event create_stateblock_capture_apply_all_events
[] =
428 {create_stateblock_all
, SB_DATA_DEFAULT
, SB_DATA_TEST_IN
},
429 {capture_stateblock
, SB_DATA_TEST_ALL
, SB_DATA_DEFAULT
},
430 {apply_stateblock
, SB_DATA_TEST_ALL
, SB_DATA_NONE
},
433 struct event create_stateblock_apply_all_events
[] =
435 {NULL
, SB_DATA_DEFAULT
, SB_DATA_TEST_IN
},
436 {create_stateblock_all
, SB_DATA_TEST_ALL
, SB_DATA_DEFAULT
},
437 {apply_stateblock
, SB_DATA_TEST_ALL
, SB_DATA_NONE
},
440 struct event create_stateblock_capture_apply_vertex_events
[] =
442 {create_stateblock_vertex
, SB_DATA_DEFAULT
, SB_DATA_TEST_IN
},
443 {capture_stateblock
, SB_DATA_TEST_ALL
, SB_DATA_DEFAULT
},
444 {apply_stateblock
, SB_DATA_TEST_VERTEX
, SB_DATA_NONE
},
447 struct event create_stateblock_apply_vertex_events
[] =
449 {NULL
, SB_DATA_DEFAULT
, SB_DATA_TEST_IN
},
450 {create_stateblock_vertex
, SB_DATA_TEST_ALL
, SB_DATA_DEFAULT
},
451 {apply_stateblock
, SB_DATA_TEST_VERTEX
, SB_DATA_NONE
},
454 struct event create_stateblock_capture_apply_pixel_events
[] =
456 {create_stateblock_pixel
, SB_DATA_DEFAULT
, SB_DATA_TEST_IN
},
457 {capture_stateblock
, SB_DATA_TEST_ALL
, SB_DATA_DEFAULT
},
458 {apply_stateblock
, SB_DATA_TEST_PIXEL
, SB_DATA_NONE
},
461 struct event create_stateblock_apply_pixel_events
[] =
463 {NULL
, SB_DATA_DEFAULT
, SB_DATA_TEST_IN
},
464 {create_stateblock_pixel
, SB_DATA_TEST_ALL
, SB_DATA_DEFAULT
},
465 {apply_stateblock
, SB_DATA_TEST_PIXEL
, SB_DATA_NONE
},
468 struct event rendertarget_switch_events
[] =
470 {NULL
, SB_DATA_NONE
, SB_DATA_TEST_IN
},
471 {switch_render_target
, SB_DATA_TEST_ALL
, SB_DATA_NONE
},
472 {revert_render_target
, SB_DATA_NONE
, SB_DATA_NONE
},
475 struct event rendertarget_stateblock_events
[] =
477 {begin_stateblock
, SB_DATA_NONE
, SB_DATA_TEST_IN
},
478 {switch_render_target
, SB_DATA_DEFAULT
, SB_DATA_NONE
},
479 {end_stateblock
, SB_DATA_NONE
, SB_DATA_NONE
},
480 {revert_render_target
, SB_DATA_NONE
, SB_DATA_NONE
},
481 {apply_stateblock
, SB_DATA_TEST_ALL
, SB_DATA_NONE
},
484 /* Setup each test for execution */
485 for (i
= 0; i
< ntests
; ++i
)
487 hr
= test
[i
].init(device
, &test
[i
]);
488 ok(SUCCEEDED(hr
), "Test \"%s\" failed setup, aborting\n", test
[i
].test_name
);
489 if (FAILED(hr
)) return;
492 trace("Running initial read state tests\n");
493 execute_test_chain(device
, test
, ntests
, read_events
, 1, NULL
);
495 trace("Running write-read state tests\n");
496 execute_test_chain(device
, test
, ntests
, write_read_events
, 2, NULL
);
498 trace("Running stateblock abort state tests\n");
499 execute_test_chain(device
, test
, ntests
, abort_stateblock_events
, 3, &arg
);
501 trace("Running stateblock apply state tests\n");
502 execute_test_chain(device
, test
, ntests
, apply_stateblock_events
, 3, &arg
);
504 trace("Running stateblock capture/reapply state tests\n");
505 execute_test_chain(device
, test
, ntests
, capture_reapply_stateblock_events
, 4, &arg
);
507 trace("Running create stateblock capture/apply all state tests\n");
508 execute_test_chain(device
, test
, ntests
, create_stateblock_capture_apply_all_events
, 3, &arg
);
510 trace("Running create stateblock apply all state tests\n");
511 execute_test_chain(device
, test
, ntests
, create_stateblock_apply_all_events
, 3, &arg
);
513 trace("Running create stateblock capture/apply vertex state tests\n");
514 execute_test_chain(device
, test
, ntests
, create_stateblock_capture_apply_vertex_events
, 3, &arg
);
516 trace("Running create stateblock apply vertex state tests\n");
517 execute_test_chain(device
, test
, ntests
, create_stateblock_apply_vertex_events
, 3, &arg
);
519 trace("Running create stateblock capture/apply pixel state tests\n");
520 execute_test_chain(device
, test
, ntests
, create_stateblock_capture_apply_pixel_events
, 3, &arg
);
522 trace("Running create stateblock apply pixel state tests\n");
523 execute_test_chain(device
, test
, ntests
, create_stateblock_apply_pixel_events
, 3, &arg
);
525 trace("Running rendertarget switch state tests\n");
526 execute_test_chain(device
, test
, ntests
, rendertarget_switch_events
, 3, &arg
);
528 trace("Running stateblock apply over rendertarget switch interrupt tests\n");
529 execute_test_chain(device
, test
, ntests
, rendertarget_stateblock_events
, 5, &arg
);
531 /* Cleanup resources */
532 for (i
= 0; i
< ntests
; ++i
)
534 if (test
[i
].cleanup
) test
[i
].cleanup(device
, &test
[i
]);
538 /* =================== State test: Pixel and Vertex Shader constants ============ */
540 struct shader_constant_data
542 float float_constant
[4]; /* 1x4 float constant */
545 struct shader_constant_arg
551 static const struct shader_constant_data shader_constant_poison_data
=
553 {1.0f
, 2.0f
, 3.0f
, 4.0f
},
556 static const struct shader_constant_data shader_constant_default_data
=
558 {0.0f
, 0.0f
, 0.0f
, 0.0f
},
561 static const struct shader_constant_data shader_constant_test_data
=
563 {5.0f
, 6.0f
, 7.0f
, 8.0f
},
566 static void shader_constant_apply_data(IDirect3DDevice8
*device
, const struct state_test
*test
, const void *data
)
568 const struct shader_constant_data
*scdata
= data
;
569 const struct shader_constant_arg
*scarg
= test
->test_arg
;
570 unsigned int index
= scarg
->idx
;
575 hr
= IDirect3DDevice8_SetVertexShaderConstant(device
, index
, scdata
->float_constant
, 1);
576 ok(SUCCEEDED(hr
), "SetVertexShaderConstant returned %#x.\n", hr
);
580 hr
= IDirect3DDevice8_SetPixelShaderConstant(device
, index
, scdata
->float_constant
, 1);
581 ok(SUCCEEDED(hr
), "SetPixelShaderConstant returned %#x.\n", hr
);
585 static void shader_constant_check_data(IDirect3DDevice8
*device
, const struct state_test
*test
,
586 const void *expected_data
, unsigned int chain_stage
)
588 const struct shader_constant_data
*scdata
= expected_data
;
589 const struct shader_constant_arg
*scarg
= test
->test_arg
;
590 struct shader_constant_data value
;
593 value
= shader_constant_poison_data
;
597 hr
= IDirect3DDevice8_GetVertexShaderConstant(device
, scarg
->idx
, value
.float_constant
, 1);
598 ok(SUCCEEDED(hr
), "GetVertexShaderConstant returned %#x.\n", hr
);
602 hr
= IDirect3DDevice8_GetPixelShaderConstant(device
, scarg
->idx
, value
.float_constant
, 1);
603 ok(SUCCEEDED(hr
), "GetPixelShaderConstant returned %#x.\n", hr
);
606 ok(!memcmp(value
.float_constant
, scdata
->float_constant
, sizeof(scdata
->float_constant
)),
607 "Chain stage %u, %s constant:\n\t{%.8e, %.8e, %.8e, %.8e} expected\n\t{%.8e, %.8e, %.8e, %.8e} received\n",
608 chain_stage
, scarg
->pshader
? "pixel shader" : "vertex shader",
609 scdata
->float_constant
[0], scdata
->float_constant
[1],
610 scdata
->float_constant
[2], scdata
->float_constant
[3],
611 value
.float_constant
[0], value
.float_constant
[1],
612 value
.float_constant
[2], value
.float_constant
[3]);
615 static HRESULT
shader_constant_test_init(IDirect3DDevice8
*device
, struct state_test
*test
)
617 const struct shader_constant_arg
*test_arg
= test
->test_arg
;
619 test
->test_context
= NULL
;
620 test
->test_data_in
= &shader_constant_test_data
;
621 test
->test_data_out_all
= &shader_constant_test_data
;
622 if (test_arg
->pshader
)
624 test
->test_data_out_vertex
= &shader_constant_default_data
;
625 test
->test_data_out_pixel
= &shader_constant_test_data
;
629 test
->test_data_out_vertex
= &shader_constant_test_data
;
630 test
->test_data_out_pixel
= &shader_constant_default_data
;
632 test
->default_data
= &shader_constant_default_data
;
633 test
->initial_data
= &shader_constant_default_data
;
638 static void shader_constants_queue_test(struct state_test
*test
, const struct shader_constant_arg
*test_arg
)
640 test
->init
= shader_constant_test_init
;
641 test
->cleanup
= NULL
;
642 test
->apply_data
= shader_constant_apply_data
;
643 test
->check_data
= shader_constant_check_data
;
644 test
->test_name
= test_arg
->pshader
? "set_get_pshader_constants" : "set_get_vshader_constants";
645 test
->test_arg
= test_arg
;
648 /* =================== State test: Lights ===================================== */
654 HRESULT get_light_result
;
655 HRESULT get_enabled_result
;
663 static const struct light_data light_poison_data
=
667 {7.0, 4.0, 2.0, 1.0},
668 {7.0, 4.0, 2.0, 1.0},
669 {7.0, 4.0, 2.0, 1.0},
672 12.12f
, 13.13f
, 14.14f
, 15.15f
, 16.16f
, 17.17f
, 18.18f
,
679 static const struct light_data light_default_data
=
682 D3DLIGHT_DIRECTIONAL
,
683 {1.0, 1.0, 1.0, 0.0},
684 {0.0, 0.0, 0.0, 0.0},
685 {0.0, 0.0, 0.0, 0.0},
688 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0,
695 /* This is used for the initial read state (before a write causes side effects)
696 * The proper return status is D3DERR_INVALIDCALL */
697 static const struct light_data light_initial_data
=
701 {7.0, 4.0, 2.0, 1.0},
702 {7.0, 4.0, 2.0, 1.0},
703 {7.0, 4.0, 2.0, 1.0},
706 12.12f
, 13.13f
, 14.14f
, 15.15f
, 16.16f
, 17.17f
, 18.18f
,
713 static const struct light_data light_test_data_in
=
717 {2.0, 2.0, 2.0, 2.0},
718 {3.0, 3.0, 3.0, 3.0},
719 {4.0, 4.0, 4.0, 4.0},
722 7.0, 8.0, 9.0, 10.0, 11.0, 12.0, 13.0,
729 /* SetLight will use 128 as the "enabled" value */
730 static const struct light_data light_test_data_out
=
734 {2.0, 2.0, 2.0, 2.0},
735 {3.0, 3.0, 3.0, 3.0},
736 {4.0, 4.0, 4.0, 4.0},
739 7.0, 8.0, 9.0, 10.0, 11.0, 12.0, 13.0,
746 static void light_apply_data(IDirect3DDevice8
*device
, const struct state_test
*test
, const void *data
)
748 const struct light_data
*ldata
= data
;
749 const struct light_arg
*larg
= test
->test_arg
;
750 unsigned int index
= larg
->idx
;
753 hr
= IDirect3DDevice8_SetLight(device
, index
, &ldata
->light
);
754 ok(SUCCEEDED(hr
), "SetLight returned %#x.\n", hr
);
756 hr
= IDirect3DDevice8_LightEnable(device
, index
, ldata
->enabled
);
757 ok(SUCCEEDED(hr
), "SetLightEnable returned %#x.\n", hr
);
760 static void light_check_data(IDirect3DDevice8
*device
, const struct state_test
*test
,
761 const void *expected_data
, unsigned int chain_stage
)
763 const struct light_arg
*larg
= test
->test_arg
;
764 const struct light_data
*ldata
= expected_data
;
765 struct light_data value
;
767 value
= light_poison_data
;
769 value
.get_enabled_result
= IDirect3DDevice8_GetLightEnable(device
, larg
->idx
, &value
.enabled
);
770 value
.get_light_result
= IDirect3DDevice8_GetLight(device
, larg
->idx
, &value
.light
);
772 ok(value
.get_enabled_result
== ldata
->get_enabled_result
,
773 "Chain stage %u: expected get_enabled_result %#x, got %#x.\n",
774 chain_stage
, ldata
->get_enabled_result
, value
.get_enabled_result
);
775 ok(value
.get_light_result
== ldata
->get_light_result
,
776 "Chain stage %u: expected get_light_result %#x, got %#x.\n",
777 chain_stage
, ldata
->get_light_result
, value
.get_light_result
);
779 ok(value
.enabled
== ldata
->enabled
,
780 "Chain stage %u: expected enabled %#x, got %#x.\n",
781 chain_stage
, ldata
->enabled
, value
.enabled
);
782 ok(value
.light
.Type
== ldata
->light
.Type
,
783 "Chain stage %u: expected light.Type %#x, got %#x.\n",
784 chain_stage
, ldata
->light
.Type
, value
.light
.Type
);
785 ok(!memcmp(&value
.light
.Diffuse
, &ldata
->light
.Diffuse
, sizeof(value
.light
.Diffuse
)),
786 "Chain stage %u, light.Diffuse:\n\t{%.8e, %.8e, %.8e, %.8e} expected\n"
787 "\t{%.8e, %.8e, %.8e, %.8e} received.\n", chain_stage
,
788 ldata
->light
.Diffuse
.r
, ldata
->light
.Diffuse
.g
,
789 ldata
->light
.Diffuse
.b
, ldata
->light
.Diffuse
.a
,
790 value
.light
.Diffuse
.r
, value
.light
.Diffuse
.g
,
791 value
.light
.Diffuse
.b
, value
.light
.Diffuse
.a
);
792 ok(!memcmp(&value
.light
.Specular
, &ldata
->light
.Specular
, sizeof(value
.light
.Specular
)),
793 "Chain stage %u, light.Specular:\n\t{%.8e, %.8e, %.8e, %.8e} expected\n"
794 "\t{%.8e, %.8e, %.8e, %.8e} received.\n", chain_stage
,
795 ldata
->light
.Specular
.r
, ldata
->light
.Specular
.g
,
796 ldata
->light
.Specular
.b
, ldata
->light
.Specular
.a
,
797 value
.light
.Specular
.r
, value
.light
.Specular
.g
,
798 value
.light
.Specular
.b
, value
.light
.Specular
.a
);
799 ok(!memcmp(&value
.light
.Ambient
, &ldata
->light
.Ambient
, sizeof(value
.light
.Ambient
)),
800 "Chain stage %u, light.Ambient:\n\t{%.8e, %.8e, %.8e, %.8e} expected\n"
801 "\t{%.8e, %.8e, %.8e, %.8e} received.\n", chain_stage
,
802 ldata
->light
.Ambient
.r
, ldata
->light
.Ambient
.g
,
803 ldata
->light
.Ambient
.b
, ldata
->light
.Ambient
.a
,
804 value
.light
.Ambient
.r
, value
.light
.Ambient
.g
,
805 value
.light
.Ambient
.b
, value
.light
.Ambient
.a
);
806 ok(!memcmp(&value
.light
.Position
, &ldata
->light
.Position
, sizeof(value
.light
.Position
)),
807 "Chain stage %u, light.Position:\n\t{%.8e, %.8e, %.8e} expected\n\t{%.8e, %.8e, %.8e} received.\n",
808 chain_stage
, ldata
->light
.Position
.x
, ldata
->light
.Position
.y
, ldata
->light
.Position
.z
,
809 value
.light
.Position
.x
, value
.light
.Position
.y
, value
.light
.Position
.z
);
810 ok(!memcmp(&value
.light
.Direction
, &ldata
->light
.Direction
, sizeof(value
.light
.Direction
)),
811 "Chain stage %u, light.Direction:\n\t{%.8e, %.8e, %.8e} expected\n\t{%.8e, %.8e, %.8e} received.\n",
812 chain_stage
, ldata
->light
.Direction
.x
, ldata
->light
.Direction
.y
, ldata
->light
.Direction
.z
,
813 value
.light
.Direction
.x
, value
.light
.Direction
.y
, value
.light
.Direction
.z
);
814 ok(value
.light
.Range
== ldata
->light
.Range
,
815 "Chain stage %u: expected light.Range %.8e, got %.8e.\n",
816 chain_stage
, ldata
->light
.Range
, value
.light
.Range
);
817 ok(value
.light
.Falloff
== ldata
->light
.Falloff
,
818 "Chain stage %u: expected light.Falloff %.8e, got %.8e.\n",
819 chain_stage
, ldata
->light
.Falloff
, value
.light
.Falloff
);
820 ok(value
.light
.Attenuation0
== ldata
->light
.Attenuation0
,
821 "Chain stage %u: expected light.Attenuation0 %.8e, got %.8e.\n",
822 chain_stage
, ldata
->light
.Attenuation0
, value
.light
.Attenuation0
);
823 ok(value
.light
.Attenuation1
== ldata
->light
.Attenuation1
,
824 "Chain stage %u: expected light.Attenuation1 %.8e, got %.8e.\n",
825 chain_stage
, ldata
->light
.Attenuation1
, value
.light
.Attenuation1
);
826 ok(value
.light
.Attenuation2
== ldata
->light
.Attenuation2
,
827 "Chain stage %u: expected light.Attenuation2 %.8e, got %.8e.\n",
828 chain_stage
, ldata
->light
.Attenuation2
, value
.light
.Attenuation2
);
829 ok(value
.light
.Theta
== ldata
->light
.Theta
,
830 "Chain stage %u: expected light.Theta %.8e, got %.8e.\n",
831 chain_stage
, ldata
->light
.Theta
, value
.light
.Theta
);
832 ok(value
.light
.Phi
== ldata
->light
.Phi
,
833 "Chain stage %u: expected light.Phi %.8e, got %.8e.\n",
834 chain_stage
, ldata
->light
.Phi
, value
.light
.Phi
);
837 static HRESULT
light_test_init(IDirect3DDevice8
*device
, struct state_test
*test
)
839 test
->test_context
= NULL
;
840 test
->test_data_in
= &light_test_data_in
;
841 test
->test_data_out_all
= &light_test_data_out
;
842 test
->test_data_out_vertex
= &light_test_data_out
;
843 test
->test_data_out_pixel
= &light_default_data
;
844 test
->default_data
= &light_default_data
;
845 test
->initial_data
= &light_initial_data
;
850 static void lights_queue_test(struct state_test
*test
, const struct light_arg
*test_arg
)
852 test
->init
= light_test_init
;
853 test
->cleanup
= NULL
;
854 test
->apply_data
= light_apply_data
;
855 test
->check_data
= light_check_data
;
856 test
->test_name
= "set_get_light";
857 test
->test_arg
= test_arg
;
860 /* =================== State test: Transforms ===================================== */
862 struct transform_data
865 D3DMATRIX projection
;
872 static const struct transform_data transform_default_data
=
875 1.0f
, 0.0f
, 0.0f
, 0.0f
,
876 0.0f
, 1.0f
, 0.0f
, 0.0f
,
877 0.0f
, 0.0f
, 1.0f
, 0.0f
,
878 0.0f
, 0.0f
, 0.0f
, 1.0f
,
881 1.0f
, 0.0f
, 0.0f
, 0.0f
,
882 0.0f
, 1.0f
, 0.0f
, 0.0f
,
883 0.0f
, 0.0f
, 1.0f
, 0.0f
,
884 0.0f
, 0.0f
, 0.0f
, 1.0f
,
887 1.0f
, 0.0f
, 0.0f
, 0.0f
,
888 0.0f
, 1.0f
, 0.0f
, 0.0f
,
889 0.0f
, 0.0f
, 1.0f
, 0.0f
,
890 0.0f
, 0.0f
, 0.0f
, 1.0f
,
893 1.0f
, 0.0f
, 0.0f
, 0.0f
,
894 0.0f
, 1.0f
, 0.0f
, 0.0f
,
895 0.0f
, 0.0f
, 1.0f
, 0.0f
,
896 0.0f
, 0.0f
, 0.0f
, 1.0f
,
899 1.0f
, 0.0f
, 0.0f
, 0.0f
,
900 0.0f
, 1.0f
, 0.0f
, 0.0f
,
901 0.0f
, 0.0f
, 1.0f
, 0.0f
,
902 0.0f
, 0.0f
, 0.0f
, 1.0f
,
905 1.0f
, 0.0f
, 0.0f
, 0.0f
,
906 0.0f
, 1.0f
, 0.0f
, 0.0f
,
907 0.0f
, 0.0f
, 1.0f
, 0.0f
,
908 0.0f
, 0.0f
, 0.0f
, 1.0f
,
912 static const struct transform_data transform_poison_data
=
915 1.0f
, 2.0f
, 3.0f
, 4.0f
,
916 5.0f
, 6.0f
, 7.0f
, 8.0f
,
917 9.0f
, 10.0f
, 11.0f
, 12.0f
,
918 13.0f
, 14.0f
, 15.0f
, 16.0f
,
921 17.0f
, 18.0f
, 19.0f
, 20.0f
,
922 21.0f
, 22.0f
, 23.0f
, 24.0f
,
923 25.0f
, 26.0f
, 27.0f
, 28.0f
,
924 29.0f
, 30.0f
, 31.0f
, 32.0f
,
927 33.0f
, 34.0f
, 35.0f
, 36.0f
,
928 37.0f
, 38.0f
, 39.0f
, 40.0f
,
929 41.0f
, 42.0f
, 43.0f
, 44.0f
,
930 45.0f
, 46.0f
, 47.0f
, 48.0f
,
933 49.0f
, 50.0f
, 51.0f
, 52.0f
,
934 53.0f
, 54.0f
, 55.0f
, 56.0f
,
935 57.0f
, 58.0f
, 59.0f
, 60.0f
,
936 61.0f
, 62.0f
, 63.0f
, 64.0f
,
939 64.0f
, 66.0f
, 67.0f
, 68.0f
,
940 69.0f
, 70.0f
, 71.0f
, 72.0f
,
941 73.0f
, 74.0f
, 75.0f
, 76.0f
,
942 77.0f
, 78.0f
, 79.0f
, 80.0f
,
945 81.0f
, 82.0f
, 83.0f
, 84.0f
,
946 85.0f
, 86.0f
, 87.0f
, 88.0f
,
947 89.0f
, 90.0f
, 91.0f
, 92.0f
,
948 93.0f
, 94.0f
, 95.0f
, 96.0f
,
952 static const struct transform_data transform_test_data
=
955 1.2f
, 3.4f
, -5.6f
, 7.2f
,
956 10.11f
, -12.13f
, 14.15f
, -1.5f
,
957 23.56f
, 12.89f
, 44.56f
, -1.0f
,
958 2.3f
, 0.0f
, 4.4f
, 5.5f
,
961 9.2f
, 38.7f
, -6.6f
, 7.2f
,
962 10.11f
, -12.13f
, 77.15f
, -1.5f
,
963 23.56f
, 12.89f
, 14.56f
, -1.0f
,
964 12.3f
, 0.0f
, 4.4f
, 5.5f
,
967 10.2f
, 3.4f
, 0.6f
, 7.2f
,
968 10.11f
, -12.13f
, 14.15f
, -1.5f
,
969 23.54f
, 12.9f
, 44.56f
, -1.0f
,
970 2.3f
, 0.0f
, 4.4f
, 5.5f
,
973 1.2f
, 3.4f
, -5.6f
, 7.2f
,
974 10.11f
, -12.13f
, -14.5f
, -1.5f
,
975 2.56f
, 12.89f
, 23.56f
, -1.0f
,
976 112.3f
, 0.0f
, 4.4f
, 2.5f
,
979 1.2f
, 31.41f
, 58.6f
, 7.2f
,
980 10.11f
, -12.13f
, -14.5f
, -1.5f
,
981 2.56f
, 12.89f
, 11.56f
, -1.0f
,
982 112.3f
, 0.0f
, 44.4f
, 2.5f
,
986 1.20f
, 3.4f
, -5.6f
, 7.0f
,
987 10.11f
, -12.156f
, -14.5f
, -1.5f
,
988 2.56f
, 1.829f
, 23.6f
, -1.0f
,
989 112.3f
, 0.0f
, 41.4f
, 2.5f
,
994 static void transform_apply_data(IDirect3DDevice8
*device
, const struct state_test
*test
, const void *data
)
996 const struct transform_data
*tdata
= data
;
999 hr
= IDirect3DDevice8_SetTransform(device
, D3DTS_VIEW
, &tdata
->view
);
1000 ok(SUCCEEDED(hr
), "SetTransform returned %#x.\n", hr
);
1002 hr
= IDirect3DDevice8_SetTransform(device
, D3DTS_PROJECTION
, &tdata
->projection
);
1003 ok(SUCCEEDED(hr
), "SetTransform returned %#x.\n", hr
);
1005 hr
= IDirect3DDevice8_SetTransform(device
, D3DTS_TEXTURE0
, &tdata
->texture0
);
1006 ok(SUCCEEDED(hr
), "SetTransform returned %#x.\n", hr
);
1008 hr
= IDirect3DDevice8_SetTransform(device
, D3DTS_TEXTURE0
+ texture_stages
- 1, &tdata
->texture7
);
1009 ok(SUCCEEDED(hr
), "SetTransform returned %#x.\n", hr
);
1011 hr
= IDirect3DDevice8_SetTransform(device
, D3DTS_WORLD
, &tdata
->world0
);
1012 ok(SUCCEEDED(hr
), "SetTransform returned %#x.\n", hr
);
1014 hr
= IDirect3DDevice8_SetTransform(device
, D3DTS_WORLDMATRIX(255), &tdata
->world255
);
1015 ok(SUCCEEDED(hr
), "SetTransform returned %#x.\n", hr
);
1018 static void compare_matrix(const char *name
, unsigned int chain_stage
,
1019 const D3DMATRIX
*received
, const D3DMATRIX
*expected
)
1021 ok(!memcmp(expected
, received
, sizeof(*expected
)),
1022 "Chain stage %u, matrix %s:\n"
1024 "\t\t%.8e, %.8e, %.8e, %.8e,\n"
1025 "\t\t%.8e, %.8e, %.8e, %.8e,\n"
1026 "\t\t%.8e, %.8e, %.8e, %.8e,\n"
1027 "\t\t%.8e, %.8e, %.8e, %.8e,\n"
1030 "\t\t%.8e, %.8e, %.8e, %.8e,\n"
1031 "\t\t%.8e, %.8e, %.8e, %.8e,\n"
1032 "\t\t%.8e, %.8e, %.8e, %.8e,\n"
1033 "\t\t%.8e, %.8e, %.8e, %.8e,\n"
1036 U(*expected
).m
[0][0], U(*expected
).m
[1][0], U(*expected
).m
[2][0], U(*expected
).m
[3][0],
1037 U(*expected
).m
[0][1], U(*expected
).m
[1][1], U(*expected
).m
[2][1], U(*expected
).m
[3][1],
1038 U(*expected
).m
[0][2], U(*expected
).m
[1][2], U(*expected
).m
[2][2], U(*expected
).m
[3][2],
1039 U(*expected
).m
[0][3], U(*expected
).m
[1][3], U(*expected
).m
[2][3], U(*expected
).m
[3][3],
1040 U(*received
).m
[0][0], U(*received
).m
[1][0], U(*received
).m
[2][0], U(*received
).m
[3][0],
1041 U(*received
).m
[0][1], U(*received
).m
[1][1], U(*received
).m
[2][1], U(*received
).m
[3][1],
1042 U(*received
).m
[0][2], U(*received
).m
[1][2], U(*received
).m
[2][2], U(*received
).m
[3][2],
1043 U(*received
).m
[0][3], U(*received
).m
[1][3], U(*received
).m
[2][3], U(*received
).m
[3][3]);
1046 static void transform_check_data(IDirect3DDevice8
*device
, const struct state_test
*test
,
1047 const void *expected_data
, unsigned int chain_stage
)
1049 const struct transform_data
*tdata
= expected_data
;
1053 value
= transform_poison_data
.view
;
1054 hr
= IDirect3DDevice8_GetTransform(device
, D3DTS_VIEW
, &value
);
1055 ok(SUCCEEDED(hr
), "GetTransform returned %#x.\n", hr
);
1056 compare_matrix("View", chain_stage
, &value
, &tdata
->view
);
1058 value
= transform_poison_data
.projection
;
1059 hr
= IDirect3DDevice8_GetTransform(device
, D3DTS_PROJECTION
, &value
);
1060 ok(SUCCEEDED(hr
), "GetTransform returned %#x.\n", hr
);
1061 compare_matrix("Projection", chain_stage
, &value
, &tdata
->projection
);
1063 value
= transform_poison_data
.texture0
;
1064 hr
= IDirect3DDevice8_GetTransform(device
, D3DTS_TEXTURE0
, &value
);
1065 ok(SUCCEEDED(hr
), "GetTransform returned %#x.\n", hr
);
1066 compare_matrix("Texture0", chain_stage
, &value
, &tdata
->texture0
);
1068 value
= transform_poison_data
.texture7
;
1069 hr
= IDirect3DDevice8_GetTransform(device
, D3DTS_TEXTURE0
+ texture_stages
- 1, &value
);
1070 ok(SUCCEEDED(hr
), "GetTransform returned %#x.\n", hr
);
1071 compare_matrix("Texture7", chain_stage
, &value
, &tdata
->texture7
);
1073 value
= transform_poison_data
.world0
;
1074 hr
= IDirect3DDevice8_GetTransform(device
, D3DTS_WORLD
, &value
);
1075 ok(SUCCEEDED(hr
), "GetTransform returned %#x.\n", hr
);
1076 compare_matrix("World0", chain_stage
, &value
, &tdata
->world0
);
1078 value
= transform_poison_data
.world255
;
1079 hr
= IDirect3DDevice8_GetTransform(device
, D3DTS_WORLDMATRIX(255), &value
);
1080 ok(SUCCEEDED(hr
), "GetTransform returned %#x.\n", hr
);
1081 compare_matrix("World255", chain_stage
, &value
, &tdata
->world255
);
1084 static HRESULT
transform_test_init(IDirect3DDevice8
*device
, struct state_test
*test
)
1086 test
->test_context
= NULL
;
1087 test
->test_data_in
= &transform_test_data
;
1088 test
->test_data_out_all
= &transform_test_data
;
1089 test
->test_data_out_vertex
= &transform_default_data
;
1090 test
->test_data_out_pixel
= &transform_default_data
;
1091 test
->default_data
= &transform_default_data
;
1092 test
->initial_data
= &transform_default_data
;
1097 static void transform_queue_test(struct state_test
*test
)
1099 test
->init
= transform_test_init
;
1100 test
->cleanup
= NULL
;
1101 test
->apply_data
= transform_apply_data
;
1102 test
->check_data
= transform_check_data
;
1103 test
->test_name
= "set_get_transforms";
1104 test
->test_arg
= NULL
;
1107 /* =================== State test: Render States ===================================== */
1109 const D3DRENDERSTATETYPE render_state_indices
[] =
1115 D3DRS_ALPHATESTENABLE
,
1124 D3DRS_ALPHABLENDENABLE
,
1126 D3DRS_SPECULARENABLE
,
1132 D3DRS_RANGEFOGENABLE
,
1133 D3DRS_STENCILENABLE
,
1140 D3DRS_STENCILWRITEMASK
,
1141 D3DRS_TEXTUREFACTOR
,
1153 D3DRS_FOGVERTEXMODE
,
1156 D3DRS_NORMALIZENORMALS
,
1157 D3DRS_DIFFUSEMATERIALSOURCE
,
1158 D3DRS_SPECULARMATERIALSOURCE
,
1159 D3DRS_AMBIENTMATERIALSOURCE
,
1160 D3DRS_EMISSIVEMATERIALSOURCE
,
1162 D3DRS_CLIPPLANEENABLE
,
1163 #if 0 /* Driver dependent */
1166 D3DRS_POINTSIZE_MIN
,
1167 D3DRS_POINTSPRITEENABLE
,
1168 D3DRS_POINTSCALEENABLE
,
1172 D3DRS_MULTISAMPLEANTIALIAS
,
1173 D3DRS_MULTISAMPLEMASK
,
1174 D3DRS_PATCHEDGESTYLE
,
1175 #if 0 /* Apparently not recorded in the stateblock */
1176 D3DRS_DEBUGMONITORTOKEN
,
1178 D3DRS_POINTSIZE_MAX
,
1179 D3DRS_INDEXEDVERTEXBLENDENABLE
,
1180 D3DRS_COLORWRITEENABLE
,
1185 struct render_state_data
1187 DWORD states
[sizeof(render_state_indices
) / sizeof(*render_state_indices
)];
1190 struct render_state_arg
1192 D3DPRESENT_PARAMETERS
*device_pparams
;
1193 float pointsize_max
;
1196 struct render_state_context
1198 struct render_state_data default_data_buffer
;
1199 struct render_state_data test_data_all_buffer
;
1200 struct render_state_data test_data_vertex_buffer
;
1201 struct render_state_data test_data_pixel_buffer
;
1202 struct render_state_data poison_data_buffer
;
1205 static void render_state_apply_data(IDirect3DDevice8
*device
, const struct state_test
*test
, const void *data
)
1207 const struct render_state_data
*rsdata
= data
;
1211 for (i
= 0; i
< sizeof(render_state_indices
) / sizeof(*render_state_indices
); ++i
)
1213 hr
= IDirect3DDevice8_SetRenderState(device
, render_state_indices
[i
], rsdata
->states
[i
]);
1214 ok(SUCCEEDED(hr
), "SetRenderState returned %#x.\n", hr
);
1218 static void render_state_check_data(IDirect3DDevice8
*device
, const struct state_test
*test
,
1219 const void *expected_data
, unsigned int chain_stage
)
1221 const struct render_state_context
*ctx
= test
->test_context
;
1222 const struct render_state_data
*rsdata
= expected_data
;
1226 for (i
= 0; i
< sizeof(render_state_indices
) / sizeof(*render_state_indices
); ++i
)
1228 DWORD value
= ctx
->poison_data_buffer
.states
[i
];
1229 hr
= IDirect3DDevice8_GetRenderState(device
, render_state_indices
[i
], &value
);
1230 ok(SUCCEEDED(hr
), "GetRenderState returned %#x.\n", hr
);
1231 ok(value
== rsdata
->states
[i
], "Chain stage %u, render state %#x: expected %#x, got %#x.\n",
1232 chain_stage
, render_state_indices
[i
], rsdata
->states
[i
], value
);
1236 static inline DWORD
to_dword(float fl
)
1238 union {float f
; DWORD d
;} ret
;
1244 static void render_state_default_data_init(const struct render_state_arg
*rsarg
, struct render_state_data
*data
)
1246 DWORD zenable
= rsarg
->device_pparams
->EnableAutoDepthStencil
? D3DZB_TRUE
: D3DZB_FALSE
;
1247 unsigned int idx
= 0;
1249 data
->states
[idx
++] = zenable
; /* ZENABLE */
1250 data
->states
[idx
++] = D3DFILL_SOLID
; /* FILLMODE */
1251 data
->states
[idx
++] = D3DSHADE_GOURAUD
; /* SHADEMODE */
1252 data
->states
[idx
++] = TRUE
; /* ZWRITEENABLE */
1253 data
->states
[idx
++] = FALSE
; /* ALPHATESTENABLE */
1254 data
->states
[idx
++] = TRUE
; /* LASTPIXEL */
1255 data
->states
[idx
++] = D3DBLEND_ONE
; /* SRCBLEND */
1256 data
->states
[idx
++] = D3DBLEND_ZERO
; /* DESTBLEND */
1257 data
->states
[idx
++] = D3DCULL_CCW
; /* CULLMODE */
1258 data
->states
[idx
++] = D3DCMP_LESSEQUAL
; /* ZFUNC */
1259 data
->states
[idx
++] = 0; /* ALPHAREF */
1260 data
->states
[idx
++] = D3DCMP_ALWAYS
; /* ALPHAFUNC */
1261 data
->states
[idx
++] = FALSE
; /* DITHERENABLE */
1262 data
->states
[idx
++] = FALSE
; /* ALPHABLENDENABLE */
1263 data
->states
[idx
++] = FALSE
; /* FOGENABLE */
1264 data
->states
[idx
++] = FALSE
; /* SPECULARENABLE */
1265 data
->states
[idx
++] = 0; /* FOGCOLOR */
1266 data
->states
[idx
++] = D3DFOG_NONE
; /* FOGTABLEMODE */
1267 data
->states
[idx
++] = to_dword(0.0f
); /* FOGSTART */
1268 data
->states
[idx
++] = to_dword(1.0f
); /* FOGEND */
1269 data
->states
[idx
++] = to_dword(1.0f
); /* FOGDENSITY */
1270 data
->states
[idx
++] = FALSE
; /* RANGEFOGENABLE */
1271 data
->states
[idx
++] = FALSE
; /* STENCILENABLE */
1272 data
->states
[idx
++] = D3DSTENCILOP_KEEP
; /* STENCILFAIL */
1273 data
->states
[idx
++] = D3DSTENCILOP_KEEP
; /* STENCILZFAIL */
1274 data
->states
[idx
++] = D3DSTENCILOP_KEEP
; /* STENCILPASS */
1275 data
->states
[idx
++] = D3DCMP_ALWAYS
; /* STENCILFUNC */
1276 data
->states
[idx
++] = 0; /* STENCILREF */
1277 data
->states
[idx
++] = 0xFFFFFFFF; /* STENCILMASK */
1278 data
->states
[idx
++] = 0xFFFFFFFF; /* STENCILWRITEMASK */
1279 data
->states
[idx
++] = 0xFFFFFFFF; /* TEXTUREFACTOR */
1280 data
->states
[idx
++] = 0; /* WRAP 0 */
1281 data
->states
[idx
++] = 0; /* WRAP 1 */
1282 data
->states
[idx
++] = 0; /* WRAP 2 */
1283 data
->states
[idx
++] = 0; /* WRAP 3 */
1284 data
->states
[idx
++] = 0; /* WRAP 4 */
1285 data
->states
[idx
++] = 0; /* WRAP 5 */
1286 data
->states
[idx
++] = 0; /* WRAP 6 */
1287 data
->states
[idx
++] = 0; /* WRAP 7 */
1288 data
->states
[idx
++] = TRUE
; /* CLIPPING */
1289 data
->states
[idx
++] = TRUE
; /* LIGHTING */
1290 data
->states
[idx
++] = 0; /* AMBIENT */
1291 data
->states
[idx
++] = D3DFOG_NONE
; /* FOGVERTEXMODE */
1292 data
->states
[idx
++] = TRUE
; /* COLORVERTEX */
1293 data
->states
[idx
++] = TRUE
; /* LOCALVIEWER */
1294 data
->states
[idx
++] = FALSE
; /* NORMALIZENORMALS */
1295 data
->states
[idx
++] = D3DMCS_COLOR1
; /* DIFFUSEMATERIALSOURCE */
1296 data
->states
[idx
++] = D3DMCS_COLOR2
; /* SPECULARMATERIALSOURCE */
1297 data
->states
[idx
++] = D3DMCS_MATERIAL
; /* AMBIENTMATERIALSOURCE */
1298 data
->states
[idx
++] = D3DMCS_MATERIAL
; /* EMISSIVEMATERIALSOURCE */
1299 data
->states
[idx
++] = D3DVBF_DISABLE
; /* VERTEXBLEND */
1300 data
->states
[idx
++] = 0; /* CLIPPLANEENABLE */
1301 if (0) data
->states
[idx
++] = to_dword(1.0f
); /* POINTSIZE, driver dependent, increase array size to enable */
1302 data
->states
[idx
++] = to_dword(0.0f
); /* POINTSIZEMIN */
1303 data
->states
[idx
++] = FALSE
; /* POINTSPRITEENABLE */
1304 data
->states
[idx
++] = FALSE
; /* POINTSCALEENABLE */
1305 data
->states
[idx
++] = to_dword(1.0f
); /* POINTSCALE_A */
1306 data
->states
[idx
++] = to_dword(0.0f
); /* POINTSCALE_B */
1307 data
->states
[idx
++] = to_dword(0.0f
); /* POINTSCALE_C */
1308 data
->states
[idx
++] = TRUE
; /* MULTISAMPLEANTIALIAS */
1309 data
->states
[idx
++] = 0xFFFFFFFF; /* MULTISAMPLEMASK */
1310 data
->states
[idx
++] = D3DPATCHEDGE_DISCRETE
; /* PATCHEDGESTYLE */
1311 if (0) data
->states
[idx
++] = 0xbaadcafe; /* DEBUGMONITORTOKEN, not recorded in the stateblock */
1312 data
->states
[idx
++] = to_dword(rsarg
->pointsize_max
); /* POINTSIZE_MAX */
1313 data
->states
[idx
++] = FALSE
; /* INDEXEDVERTEXBLENDENABLE */
1314 data
->states
[idx
++] = 0x0000000F; /* COLORWRITEENABLE */
1315 data
->states
[idx
++] = to_dword(0.0f
); /* TWEENFACTOR */
1316 data
->states
[idx
++] = D3DBLENDOP_ADD
; /* BLENDOP */
1319 static void render_state_poison_data_init(struct render_state_data
*data
)
1323 for (i
= 0; i
< sizeof(render_state_indices
) / sizeof(*render_state_indices
); ++i
)
1325 data
->states
[i
] = 0x1337c0de;
1329 static void render_state_test_data_init(struct render_state_data
*data
)
1331 unsigned int idx
= 0;
1333 data
->states
[idx
++] = D3DZB_USEW
; /* ZENABLE */
1334 data
->states
[idx
++] = D3DFILL_WIREFRAME
; /* FILLMODE */
1335 data
->states
[idx
++] = D3DSHADE_PHONG
; /* SHADEMODE */
1336 data
->states
[idx
++] = FALSE
; /* ZWRITEENABLE */
1337 data
->states
[idx
++] = TRUE
; /* ALPHATESTENABLE */
1338 data
->states
[idx
++] = FALSE
; /* LASTPIXEL */
1339 data
->states
[idx
++] = D3DBLEND_SRCALPHASAT
; /* SRCBLEND */
1340 data
->states
[idx
++] = D3DBLEND_INVDESTALPHA
; /* DESTBLEND */
1341 data
->states
[idx
++] = D3DCULL_CW
; /* CULLMODE */
1342 data
->states
[idx
++] = D3DCMP_NOTEQUAL
; /* ZFUNC */
1343 data
->states
[idx
++] = 10; /* ALPHAREF */
1344 data
->states
[idx
++] = D3DCMP_GREATER
; /* ALPHAFUNC */
1345 data
->states
[idx
++] = TRUE
; /* DITHERENABLE */
1346 data
->states
[idx
++] = TRUE
; /* ALPHABLENDENABLE */
1347 data
->states
[idx
++] = TRUE
; /* FOGENABLE */
1348 data
->states
[idx
++] = TRUE
; /* SPECULARENABLE */
1349 data
->states
[idx
++] = 1 << 31; /* FOGCOLOR */
1350 data
->states
[idx
++] = D3DFOG_EXP
; /* FOGTABLEMODE */
1351 data
->states
[idx
++] = to_dword(0.1f
); /* FOGSTART */
1352 data
->states
[idx
++] = to_dword(0.8f
); /* FOGEND */
1353 data
->states
[idx
++] = to_dword(0.5f
); /* FOGDENSITY */
1354 data
->states
[idx
++] = TRUE
; /* RANGEFOGENABLE */
1355 data
->states
[idx
++] = TRUE
; /* STENCILENABLE */
1356 data
->states
[idx
++] = D3DSTENCILOP_INCRSAT
; /* STENCILFAIL */
1357 data
->states
[idx
++] = D3DSTENCILOP_REPLACE
; /* STENCILZFAIL */
1358 data
->states
[idx
++] = D3DSTENCILOP_INVERT
; /* STENCILPASS */
1359 data
->states
[idx
++] = D3DCMP_LESS
; /* STENCILFUNC */
1360 data
->states
[idx
++] = 10; /* STENCILREF */
1361 data
->states
[idx
++] = 0xFF00FF00; /* STENCILMASK */
1362 data
->states
[idx
++] = 0x00FF00FF; /* STENCILWRITEMASK */
1363 data
->states
[idx
++] = 0xF0F0F0F0; /* TEXTUREFACTOR */
1364 data
->states
[idx
++] = D3DWRAPCOORD_0
| D3DWRAPCOORD_2
; /* WRAP 0 */
1365 data
->states
[idx
++] = D3DWRAPCOORD_1
| D3DWRAPCOORD_3
; /* WRAP 1 */
1366 data
->states
[idx
++] = D3DWRAPCOORD_2
| D3DWRAPCOORD_3
; /* WRAP 2 */
1367 data
->states
[idx
++] = D3DWRAPCOORD_3
| D3DWRAPCOORD_0
; /* WRAP 4 */
1368 data
->states
[idx
++] = D3DWRAPCOORD_0
| D3DWRAPCOORD_1
| D3DWRAPCOORD_2
; /* WRAP 5 */
1369 data
->states
[idx
++] = D3DWRAPCOORD_1
| D3DWRAPCOORD_3
| D3DWRAPCOORD_2
; /* WRAP 6 */
1370 data
->states
[idx
++] = D3DWRAPCOORD_2
| D3DWRAPCOORD_1
| D3DWRAPCOORD_0
; /* WRAP 7 */
1371 data
->states
[idx
++] = D3DWRAPCOORD_1
| D3DWRAPCOORD_0
| D3DWRAPCOORD_2
| D3DWRAPCOORD_3
; /* WRAP 8 */
1372 data
->states
[idx
++] = FALSE
; /* CLIPPING */
1373 data
->states
[idx
++] = FALSE
; /* LIGHTING */
1374 data
->states
[idx
++] = 255 << 16; /* AMBIENT */
1375 data
->states
[idx
++] = D3DFOG_EXP2
; /* FOGVERTEXMODE */
1376 data
->states
[idx
++] = FALSE
; /* COLORVERTEX */
1377 data
->states
[idx
++] = FALSE
; /* LOCALVIEWER */
1378 data
->states
[idx
++] = TRUE
; /* NORMALIZENORMALS */
1379 data
->states
[idx
++] = D3DMCS_COLOR2
; /* DIFFUSEMATERIALSOURCE */
1380 data
->states
[idx
++] = D3DMCS_MATERIAL
; /* SPECULARMATERIALSOURCE */
1381 data
->states
[idx
++] = D3DMCS_COLOR1
; /* AMBIENTMATERIALSOURCE */
1382 data
->states
[idx
++] = D3DMCS_COLOR2
; /* EMISSIVEMATERIALSOURCE */
1383 data
->states
[idx
++] = D3DVBF_3WEIGHTS
; /* VERTEXBLEND */
1384 data
->states
[idx
++] = 0xf1f1f1f1; /* CLIPPLANEENABLE */
1385 if (0) data
->states
[idx
++] = to_dword(32.0f
);/* POINTSIZE, driver dependent, increase array size to enable */
1386 data
->states
[idx
++] = to_dword(0.7f
); /* POINTSIZEMIN */
1387 data
->states
[idx
++] = TRUE
; /* POINTSPRITEENABLE */
1388 data
->states
[idx
++] = TRUE
; /* POINTSCALEENABLE */
1389 data
->states
[idx
++] = to_dword(0.7f
); /* POINTSCALE_A */
1390 data
->states
[idx
++] = to_dword(0.5f
); /* POINTSCALE_B */
1391 data
->states
[idx
++] = to_dword(0.4f
); /* POINTSCALE_C */
1392 data
->states
[idx
++] = FALSE
; /* MULTISAMPLEANTIALIAS */
1393 data
->states
[idx
++] = 0xABCDDBCA; /* MULTISAMPLEMASK */
1394 data
->states
[idx
++] = D3DPATCHEDGE_CONTINUOUS
; /* PATCHEDGESTYLE */
1395 if (0) data
->states
[idx
++] = D3DDMT_DISABLE
; /* DEBUGMONITORTOKEN, not recorded in the stateblock */
1396 data
->states
[idx
++] = to_dword(77.0f
); /* POINTSIZE_MAX */
1397 data
->states
[idx
++] = TRUE
; /* INDEXEDVERTEXBLENDENABLE */
1398 data
->states
[idx
++] = 0x00000009; /* COLORWRITEENABLE */
1399 data
->states
[idx
++] = to_dword(0.2f
); /* TWEENFACTOR */
1400 data
->states
[idx
++] = D3DBLENDOP_REVSUBTRACT
;/* BLENDOP */
1403 static HRESULT
render_state_test_init(IDirect3DDevice8
*device
, struct state_test
*test
)
1405 static const DWORD states_vertex
[] =
1408 D3DRS_AMBIENTMATERIALSOURCE
,
1410 D3DRS_CLIPPLANEENABLE
,
1413 D3DRS_DIFFUSEMATERIALSOURCE
,
1414 D3DRS_EMISSIVEMATERIALSOURCE
,
1421 D3DRS_FOGVERTEXMODE
,
1422 D3DRS_INDEXEDVERTEXBLENDENABLE
,
1425 D3DRS_MULTISAMPLEANTIALIAS
,
1426 D3DRS_MULTISAMPLEMASK
,
1427 D3DRS_NORMALIZENORMALS
,
1428 D3DRS_PATCHEDGESTYLE
,
1432 D3DRS_POINTSCALEENABLE
,
1434 D3DRS_POINTSIZE_MAX
,
1435 D3DRS_POINTSIZE_MIN
,
1436 D3DRS_POINTSPRITEENABLE
,
1437 D3DRS_RANGEFOGENABLE
,
1439 D3DRS_SPECULARENABLE
,
1440 D3DRS_SPECULARMATERIALSOURCE
,
1445 static const DWORD states_pixel
[] =
1447 D3DRS_ALPHABLENDENABLE
,
1450 D3DRS_ALPHATESTENABLE
,
1452 D3DRS_COLORWRITEENABLE
,
1462 D3DRS_STENCILENABLE
,
1468 D3DRS_STENCILWRITEMASK
,
1470 D3DRS_TEXTUREFACTOR
,
1484 const struct render_state_arg
*rsarg
= test
->test_arg
;
1487 struct render_state_context
*ctx
= HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, sizeof(*ctx
));
1488 if (!ctx
) return E_FAIL
;
1489 test
->test_context
= ctx
;
1491 test
->default_data
= &ctx
->default_data_buffer
;
1492 test
->initial_data
= &ctx
->default_data_buffer
;
1493 test
->test_data_in
= &ctx
->test_data_all_buffer
;
1494 test
->test_data_out_all
= &ctx
->test_data_all_buffer
;
1495 test
->test_data_out_vertex
= &ctx
->test_data_vertex_buffer
;
1496 test
->test_data_out_pixel
= &ctx
->test_data_pixel_buffer
;
1498 render_state_default_data_init(rsarg
, &ctx
->default_data_buffer
);
1499 render_state_test_data_init(&ctx
->test_data_all_buffer
);
1500 render_state_poison_data_init(&ctx
->poison_data_buffer
);
1502 for (i
= 0; i
< sizeof(render_state_indices
) / sizeof(*render_state_indices
); ++i
)
1504 ctx
->test_data_vertex_buffer
.states
[i
] = ctx
->default_data_buffer
.states
[i
];
1505 for (j
= 0; j
< sizeof(states_vertex
) / sizeof(*states_vertex
); ++j
)
1507 if (render_state_indices
[i
] == states_vertex
[j
])
1509 ctx
->test_data_vertex_buffer
.states
[i
] = ctx
->test_data_all_buffer
.states
[i
];
1514 ctx
->test_data_pixel_buffer
.states
[i
] = ctx
->default_data_buffer
.states
[i
];
1515 for (j
= 0; j
< sizeof(states_pixel
) / sizeof(*states_pixel
); ++j
)
1517 if (render_state_indices
[i
] == states_pixel
[j
])
1519 ctx
->test_data_pixel_buffer
.states
[i
] = ctx
->test_data_all_buffer
.states
[i
];
1528 static void render_state_test_cleanup(IDirect3DDevice8
*device
, struct state_test
*test
)
1530 HeapFree(GetProcessHeap(), 0, test
->test_context
);
1533 static void render_states_queue_test(struct state_test
*test
, const struct render_state_arg
*test_arg
)
1535 test
->init
= render_state_test_init
;
1536 test
->cleanup
= render_state_test_cleanup
;
1537 test
->apply_data
= render_state_apply_data
;
1538 test
->check_data
= render_state_check_data
;
1539 test
->test_name
= "set_get_render_states";
1540 test
->test_arg
= test_arg
;
1543 /* resource tests */
1545 struct resource_test_arg
1553 struct resource_test_data
1557 IDirect3DIndexBuffer8
*ib
;
1558 IDirect3DVertexBuffer8
**vb
;
1559 IDirect3DTexture8
**tex
;
1562 struct resource_test_context
1564 struct resource_test_data default_data
;
1565 struct resource_test_data test_data_all
;
1566 struct resource_test_data test_data_vertex
;
1567 struct resource_test_data test_data_pixel
;
1568 struct resource_test_data poison_data
;
1571 static void resource_apply_data(IDirect3DDevice8
*device
, const struct state_test
*test
, const void *data
)
1573 const struct resource_test_arg
*arg
= test
->test_arg
;
1574 const struct resource_test_data
*d
= data
;
1578 hr
= IDirect3DDevice8_SetVertexShader(device
, d
->vs
);
1579 ok(SUCCEEDED(hr
), "SetVertexShader (%u) returned %#x.\n", d
->vs
, hr
);
1581 hr
= IDirect3DDevice8_SetPixelShader(device
, d
->ps
);
1582 ok(SUCCEEDED(hr
), "SetPixelShader (%u) returned %#x.\n", d
->ps
, hr
);
1584 hr
= IDirect3DDevice8_SetIndices(device
, d
->ib
, 0);
1585 ok(SUCCEEDED(hr
), "SetIndices (%p) returned %#x.\n", d
->ib
, hr
);
1587 for (i
= 0; i
< arg
->stream_count
; ++i
)
1589 hr
= IDirect3DDevice8_SetStreamSource(device
, i
, d
->vb
[i
], 64);
1590 ok(SUCCEEDED(hr
), "SetStreamSource (%u, %p, 64) returned %#x.\n",
1594 for (i
= 0; i
< arg
->tex_count
; ++i
)
1596 hr
= IDirect3DDevice8_SetTexture(device
, i
, (IDirect3DBaseTexture8
*)d
->tex
[i
]);
1597 ok(SUCCEEDED(hr
), "SetTexture (%u, %p) returned %#x.\n", i
, d
->tex
[i
], hr
);
1601 static void resource_check_data(IDirect3DDevice8
*device
, const struct state_test
*test
,
1602 const void *expected_data
, unsigned int chain_stage
)
1604 const struct resource_test_context
*ctx
= test
->test_context
;
1605 const struct resource_test_data
*poison
= &ctx
->poison_data
;
1606 const struct resource_test_arg
*arg
= test
->test_arg
;
1607 const struct resource_test_data
*d
= expected_data
;
1614 hr
= IDirect3DDevice8_GetVertexShader(device
, &v
);
1615 ok(SUCCEEDED(hr
), "GetVertexShader returned %#x.\n", hr
);
1616 ok(v
== d
->vs
, "Chain stage %u, expected vertex shader %#x, received %#x.\n",
1617 chain_stage
, d
->vs
, v
);
1620 hr
= IDirect3DDevice8_GetPixelShader(device
, &v
);
1621 ok(SUCCEEDED(hr
), "GetPixelShader returned %#x.\n", hr
);
1622 ok(v
== d
->ps
, "Chain stage %u, expected pixel shader %#x, received %#x.\n",
1623 chain_stage
, d
->ps
, v
);
1626 hr
= IDirect3DDevice8_GetIndices(device
, (IDirect3DIndexBuffer8
**)&ptr
, &v
);
1627 ok(SUCCEEDED(hr
), "GetIndices returned %#x.\n", hr
);
1628 ok(ptr
== d
->ib
, "Chain stage %u, expected index buffer %p, received %p.\n",
1629 chain_stage
, d
->ib
, ptr
);
1630 if (SUCCEEDED(hr
) && ptr
)
1632 IDirect3DIndexBuffer8_Release((IDirect3DIndexBuffer8
*)ptr
);
1635 for (i
= 0; i
< arg
->stream_count
; ++i
)
1637 ptr
= poison
->vb
[i
];
1638 hr
= IDirect3DDevice8_GetStreamSource(device
, i
, (IDirect3DVertexBuffer8
**)&ptr
, &v
);
1639 ok(SUCCEEDED(hr
), "GetStreamSource (%u) returned %#x.\n", i
, hr
);
1640 ok(ptr
== d
->vb
[i
], "Chain stage %u, stream %u, expected vertex buffer %p, received %p.\n",
1641 chain_stage
, i
, d
->vb
[i
], ptr
);
1642 if (SUCCEEDED(hr
) && ptr
)
1644 IDirect3DIndexBuffer8_Release((IDirect3DVertexBuffer8
*)ptr
);
1648 for (i
= 0; i
< arg
->tex_count
; ++i
)
1650 ptr
= poison
->tex
[i
];
1651 hr
= IDirect3DDevice8_GetTexture(device
, i
, (IDirect3DBaseTexture8
**)&ptr
);
1652 ok(SUCCEEDED(hr
), "SetTexture (%u) returned %#x.\n", i
, hr
);
1653 ok(ptr
== d
->tex
[i
], "Chain stage %u, texture stage %u, expected texture %p, received %p.\n",
1654 chain_stage
, i
, d
->tex
[i
], ptr
);
1655 if (SUCCEEDED(hr
) && ptr
)
1657 IDirect3DBaseTexture8_Release((IDirect3DBaseTexture8
*)ptr
);
1662 static void resource_default_data_init(struct resource_test_data
*data
, const struct resource_test_arg
*arg
)
1669 data
->vb
= HeapAlloc(GetProcessHeap(), 0, arg
->stream_count
* sizeof(*data
->vb
));
1670 for (i
= 0; i
< arg
->stream_count
; ++i
)
1674 data
->tex
= HeapAlloc(GetProcessHeap(), 0, arg
->tex_count
* sizeof(*data
->tex
));
1675 for (i
= 0; i
< arg
->tex_count
; ++i
)
1677 data
->tex
[i
] = NULL
;
1681 static void resource_test_data_init(IDirect3DDevice8
*device
,
1682 struct resource_test_data
*data
, const struct resource_test_arg
*arg
)
1684 static const DWORD vs_code
[] =
1686 0xfffe0101, /* vs_1_1 */
1687 0x00000009, 0xc0010000, 0x90e40000, 0xa0e40000, /* dp4 oPos.x, v0, c0 */
1688 0x00000009, 0xc0020000, 0x90e40000, 0xa0e40001, /* dp4 oPos.y, v0, c1 */
1689 0x00000009, 0xc0040000, 0x90e40000, 0xa0e40002, /* dp4 oPos.z, v0, c2 */
1690 0x00000009, 0xc0080000, 0x90e40000, 0xa0e40003, /* dp4 oPos.w, v0, c3 */
1691 0x0000ffff, /* END */
1693 static const DWORD ps_code
[] =
1695 0xffff0101, /* ps_1_1 */
1696 0x00000051, 0xa00f0001, 0x3f800000, 0x00000000, 0x00000000, 0x00000000, /* def c1 = 1.0, 0.0, 0.0, 0.0 */
1697 0x00000042, 0xb00f0000, /* tex t0 */
1698 0x00000008, 0x800f0000, 0xa0e40001, 0xa0e40000, /* dp3 r0, c1, c0 */
1699 0x00000005, 0x800f0000, 0x90e40000, 0x80e40000, /* mul r0, v0, r0 */
1700 0x00000005, 0x800f0000, 0xb0e40000, 0x80e40000, /* mul r0, t0, r0 */
1701 0x0000ffff, /* END */
1703 static const DWORD decl
[] =
1706 D3DVSD_REG(D3DVSDE_POSITION
, D3DVSDT_FLOAT3
),
1707 D3DVSD_REG(D3DVSDE_DIFFUSE
, D3DVSDT_D3DCOLOR
),
1714 if (arg
->vs_version
)
1716 hr
= IDirect3DDevice8_CreateVertexShader(device
, decl
, vs_code
, &data
->vs
, 0);
1717 ok(SUCCEEDED(hr
), "CreateVertexShader returned hr %#x.\n", hr
);
1720 if (arg
->ps_version
)
1722 hr
= IDirect3DDevice8_CreatePixelShader(device
, ps_code
, &data
->ps
);
1723 ok(SUCCEEDED(hr
), "CreatePixelShader returned hr %#x.\n", hr
);
1726 hr
= IDirect3DDevice8_CreateIndexBuffer(device
, 64, D3DUSAGE_DYNAMIC
, D3DFMT_INDEX32
, D3DPOOL_DEFAULT
, &data
->ib
);
1727 ok(SUCCEEDED(hr
), "CreateIndexBuffer returned hr %#x.\n", hr
);
1729 data
->vb
= HeapAlloc(GetProcessHeap(), 0, arg
->stream_count
* sizeof(*data
->vb
));
1730 for (i
= 0; i
< arg
->stream_count
; ++i
)
1732 hr
= IDirect3DDevice8_CreateVertexBuffer(device
, 64, D3DUSAGE_DYNAMIC
,
1733 0, D3DPOOL_DEFAULT
, &data
->vb
[i
]);
1734 ok(SUCCEEDED(hr
), "CreateVertexBuffer (%u) returned hr %#x.\n", i
, hr
);
1737 data
->tex
= HeapAlloc(GetProcessHeap(), 0, arg
->tex_count
* sizeof(*data
->tex
));
1738 for (i
= 0; i
< arg
->tex_count
; ++i
)
1740 hr
= IDirect3DDevice8_CreateTexture(device
, 64, 64, 0, 0,
1741 D3DFMT_A8R8G8B8
, D3DPOOL_DEFAULT
, &data
->tex
[i
]);
1742 ok(SUCCEEDED(hr
), "CreateTexture (%u) returned hr %#x.\n", i
, hr
);
1746 static void resource_poison_data_init(struct resource_test_data
*data
, const struct resource_test_arg
*arg
)
1748 DWORD_PTR poison
= 0xdeadbeef;
1751 data
->vs
= poison
++;
1752 data
->ps
= poison
++;
1753 data
->ib
= (IDirect3DIndexBuffer8
*)poison
++;
1754 data
->vb
= HeapAlloc(GetProcessHeap(), 0, arg
->stream_count
* sizeof(*data
->vb
));
1755 for (i
= 0; i
< arg
->stream_count
; ++i
)
1757 data
->vb
[i
] = (IDirect3DVertexBuffer8
*)poison
++;
1759 data
->tex
= HeapAlloc(GetProcessHeap(), 0, arg
->tex_count
* sizeof(*data
->tex
));
1760 for (i
= 0; i
< arg
->tex_count
; ++i
)
1762 data
->tex
[i
] = (IDirect3DTexture8
*)poison
++;
1766 static HRESULT
resource_test_init(IDirect3DDevice8
*device
, struct state_test
*test
)
1768 const struct resource_test_arg
*arg
= test
->test_arg
;
1769 struct resource_test_context
*ctx
;
1771 ctx
= HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, sizeof(*ctx
));
1772 if (!ctx
) return E_OUTOFMEMORY
;
1774 test
->test_context
= ctx
;
1775 test
->test_data_in
= &ctx
->test_data_all
;
1776 test
->test_data_out_all
= &ctx
->test_data_all
;
1777 test
->test_data_out_vertex
= &ctx
->test_data_vertex
;
1778 test
->test_data_out_pixel
= &ctx
->test_data_pixel
;
1779 test
->default_data
= &ctx
->default_data
;
1780 test
->initial_data
= &ctx
->default_data
;
1782 resource_default_data_init(&ctx
->default_data
, arg
);
1783 resource_test_data_init(device
, &ctx
->test_data_all
, arg
);
1784 resource_default_data_init(&ctx
->test_data_vertex
, arg
);
1785 resource_default_data_init(&ctx
->test_data_pixel
, arg
);
1786 resource_poison_data_init(&ctx
->poison_data
, arg
);
1788 ctx
->test_data_vertex
.vs
= ctx
->test_data_all
.vs
;
1789 ctx
->test_data_pixel
.ps
= ctx
->test_data_all
.ps
;
1794 static void resource_test_cleanup(IDirect3DDevice8
*device
, struct state_test
*test
)
1796 struct resource_test_context
*ctx
= test
->test_context
;
1797 const struct resource_test_arg
*arg
= test
->test_arg
;
1801 resource_apply_data(device
, test
, &ctx
->default_data
);
1803 if (ctx
->test_data_all
.vs
)
1805 hr
= IDirect3DDevice8_DeleteVertexShader(device
, ctx
->test_data_all
.vs
);
1806 ok(SUCCEEDED(hr
), "DeleteVertexShader (%u) returned %#x.\n", ctx
->test_data_all
.vs
, hr
);
1809 if (ctx
->test_data_all
.ps
)
1811 hr
= IDirect3DDevice8_DeletePixelShader(device
, ctx
->test_data_all
.ps
);
1812 ok(SUCCEEDED(hr
), "DeletePixelShader (%u) returned %#x.\n", ctx
->test_data_all
.ps
, hr
);
1815 IDirect3DIndexBuffer8_Release(ctx
->test_data_all
.ib
);
1816 for (i
= 0; i
< arg
->stream_count
; ++i
)
1818 IDirect3DVertexBuffer8_Release(ctx
->test_data_all
.vb
[i
]);
1821 for (i
= 0; i
< arg
->tex_count
; ++i
)
1823 IDirect3DBaseTexture8_Release(ctx
->test_data_all
.tex
[i
]);
1826 HeapFree(GetProcessHeap(), 0, ctx
->default_data
.vb
);
1827 HeapFree(GetProcessHeap(), 0, ctx
->default_data
.tex
);
1828 HeapFree(GetProcessHeap(), 0, ctx
->test_data_all
.vb
);
1829 HeapFree(GetProcessHeap(), 0, ctx
->test_data_all
.tex
);
1830 HeapFree(GetProcessHeap(), 0, ctx
->test_data_vertex
.vb
);
1831 HeapFree(GetProcessHeap(), 0, ctx
->test_data_vertex
.tex
);
1832 HeapFree(GetProcessHeap(), 0, ctx
->test_data_pixel
.vb
);
1833 HeapFree(GetProcessHeap(), 0, ctx
->test_data_pixel
.tex
);
1834 HeapFree(GetProcessHeap(), 0, ctx
->poison_data
.vb
);
1835 HeapFree(GetProcessHeap(), 0, ctx
->poison_data
.tex
);
1836 HeapFree(GetProcessHeap(), 0, ctx
);
1839 static void resource_test_queue(struct state_test
*test
, const struct resource_test_arg
*test_arg
)
1841 test
->init
= resource_test_init
;
1842 test
->cleanup
= resource_test_cleanup
;
1843 test
->apply_data
= resource_apply_data
;
1844 test
->check_data
= resource_check_data
;
1845 test
->test_name
= "set_get_resources";
1846 test
->test_arg
= test_arg
;
1849 /* =================== Main state tests function =============================== */
1851 static void test_state_management(IDirect3DDevice8
*device
, D3DPRESENT_PARAMETERS
*device_pparams
)
1856 /* Test count: 2 for shader constants
1859 * 1 for render states
1862 struct state_test tests
[6];
1863 unsigned int tcount
= 0;
1865 struct shader_constant_arg pshader_constant_arg
;
1866 struct shader_constant_arg vshader_constant_arg
;
1867 struct resource_test_arg resource_test_arg
;
1868 struct render_state_arg render_state_arg
;
1869 struct light_arg light_arg
;
1871 hr
= IDirect3DDevice8_GetDeviceCaps(device
, &caps
);
1872 ok(SUCCEEDED(hr
), "GetDeviceCaps returned %#x.\n", hr
);
1873 if (FAILED(hr
)) return;
1875 texture_stages
= caps
.MaxTextureBlendStages
;
1877 /* Zero test memory */
1878 memset(tests
, 0, sizeof(tests
));
1880 if (caps
.VertexShaderVersion
& 0xffff)
1882 vshader_constant_arg
.idx
= 0;
1883 vshader_constant_arg
.pshader
= FALSE
;
1884 shader_constants_queue_test(&tests
[tcount
++], &vshader_constant_arg
);
1887 if (caps
.PixelShaderVersion
& 0xffff)
1889 pshader_constant_arg
.idx
= 0;
1890 pshader_constant_arg
.pshader
= TRUE
;
1891 shader_constants_queue_test(&tests
[tcount
++], &pshader_constant_arg
);
1895 lights_queue_test(&tests
[tcount
++], &light_arg
);
1897 transform_queue_test(&tests
[tcount
++]);
1899 render_state_arg
.device_pparams
= device_pparams
;
1900 render_state_arg
.pointsize_max
= caps
.MaxPointSize
;
1901 render_states_queue_test(&tests
[tcount
++], &render_state_arg
);
1903 resource_test_arg
.vs_version
= caps
.VertexShaderVersion
& 0xffff;
1904 resource_test_arg
.ps_version
= caps
.PixelShaderVersion
& 0xffff;
1905 resource_test_arg
.stream_count
= caps
.MaxStreams
;
1906 resource_test_arg
.tex_count
= caps
.MaxTextureBlendStages
;
1907 resource_test_queue(&tests
[tcount
++], &resource_test_arg
);
1909 execute_test_chain_all(device
, tests
, tcount
);
1912 static void test_shader_constant_apply(IDirect3DDevice8
*device
)
1914 static const float initial
[] = {0.0f
, 0.0f
, 0.0f
, 0.0f
};
1915 static const float vs_const
[] = {1.0f
, 2.0f
, 3.0f
, 4.0f
};
1916 static const float ps_const
[] = {5.0f
, 6.0f
, 7.0f
, 8.0f
};
1917 DWORD vs_version
, ps_version
;
1923 hr
= IDirect3DDevice8_GetDeviceCaps(device
, &caps
);
1924 ok(SUCCEEDED(hr
), "GetDeviceCaps returned %#x.\n", hr
);
1925 vs_version
= caps
.VertexShaderVersion
& 0xffff;
1926 ps_version
= caps
.PixelShaderVersion
& 0xffff;
1930 hr
= IDirect3DDevice8_SetVertexShaderConstant(device
, 0, initial
, 1);
1931 ok(SUCCEEDED(hr
), "SetVertexShaderConstant returned %#x.\n", hr
);
1932 hr
= IDirect3DDevice8_SetVertexShaderConstant(device
, 1, initial
, 1);
1933 ok(SUCCEEDED(hr
), "SetVertexShaderConstant returned %#x.\n", hr
);
1935 hr
= IDirect3DDevice8_GetVertexShaderConstant(device
, 0, ret
, 1);
1936 ok(SUCCEEDED(hr
), "GetVertexShaderConstant returned %#x.\n", hr
);
1937 ok(!memcmp(ret
, initial
, sizeof(initial
)),
1938 "GetVertexShaderConstant got {%f, %f, %f, %f}, expected {%f, %f, %f, %f}\n",
1939 ret
[0], ret
[1], ret
[2], ret
[3], initial
[0], initial
[1], initial
[2], initial
[3]);
1940 hr
= IDirect3DDevice8_GetVertexShaderConstant(device
, 1, ret
, 1);
1941 ok(SUCCEEDED(hr
), "GetVertexShaderConstant returned %#x\n", hr
);
1942 ok(!memcmp(ret
, initial
, sizeof(initial
)),
1943 "GetVertexShaderConstant got {%f, %f, %f, %f}, expected {%f, %f, %f, %f}\n",
1944 ret
[0], ret
[1], ret
[2], ret
[3], initial
[0], initial
[1], initial
[2], initial
[3]);
1946 hr
= IDirect3DDevice8_SetVertexShaderConstant(device
, 0, vs_const
, 1);
1947 ok(SUCCEEDED(hr
), "SetVertexShaderConstant returned %#x.\n", hr
);
1951 hr
= IDirect3DDevice8_SetPixelShaderConstant(device
, 0, initial
, 1);
1952 ok(SUCCEEDED(hr
), "SetPixelShaderConstant returned %#x.\n", hr
);
1953 hr
= IDirect3DDevice8_SetPixelShaderConstant(device
, 1, initial
, 1);
1954 ok(SUCCEEDED(hr
), "SetPixelShaderConstant returned %#x.\n", hr
);
1956 hr
= IDirect3DDevice8_GetPixelShaderConstant(device
, 0, ret
, 1);
1957 ok(SUCCEEDED(hr
), "GetPixelShaderConstant returned %#x.\n", hr
);
1958 ok(!memcmp(ret
, initial
, sizeof(initial
)),
1959 "GetpixelShaderConstant got {%f, %f, %f, %f}, expected {%f, %f, %f, %f}\n",
1960 ret
[0], ret
[1], ret
[2], ret
[3], initial
[0], initial
[1], initial
[2], initial
[3]);
1961 hr
= IDirect3DDevice8_GetPixelShaderConstant(device
, 1, ret
, 1);
1962 ok(SUCCEEDED(hr
), "GetPixelShaderConstant returned %#x.\n", hr
);
1963 ok(!memcmp(ret
, initial
, sizeof(initial
)),
1964 "GetPixelShaderConstant got {%f, %f, %f, %f}, expected {%f, %f, %f, %f}\n",
1965 ret
[0], ret
[1], ret
[2], ret
[3], initial
[0], initial
[1], initial
[2], initial
[3]);
1967 hr
= IDirect3DDevice8_SetPixelShaderConstant(device
, 0, ps_const
, 1);
1968 ok(SUCCEEDED(hr
), "SetPixelShaderConstant returned %#x.\n", hr
);
1971 hr
= IDirect3DDevice8_BeginStateBlock(device
);
1972 ok(SUCCEEDED(hr
), "BeginStateBlock returned %#x\n", hr
);
1976 hr
= IDirect3DDevice8_SetVertexShaderConstant(device
, 1, vs_const
, 1);
1977 ok(SUCCEEDED(hr
), "SetVertexShaderConstant returned %#x.\n", hr
);
1981 hr
= IDirect3DDevice8_SetPixelShaderConstant(device
, 1, ps_const
, 1);
1982 ok(SUCCEEDED(hr
), "SetPixelShaderConstant returned %#x.\n", hr
);
1985 hr
= IDirect3DDevice8_EndStateBlock(device
, &stateblock
);
1986 ok(SUCCEEDED(hr
), "EndStateBlock returned %#x\n", hr
);
1990 hr
= IDirect3DDevice8_GetVertexShaderConstant(device
, 0, ret
, 1);
1991 ok(SUCCEEDED(hr
), "GetVertexShaderConstant returned %#x.\n", hr
);
1992 ok(!memcmp(ret
, vs_const
, sizeof(vs_const
)),
1993 "GetVertexShaderConstant got {%f, %f, %f, %f}, expected {%f, %f, %f, %f}\n",
1994 ret
[0], ret
[1], ret
[2], ret
[3], vs_const
[0], vs_const
[1], vs_const
[2], vs_const
[3]);
1995 hr
= IDirect3DDevice8_GetVertexShaderConstant(device
, 1, ret
, 1);
1996 ok(SUCCEEDED(hr
), "GetVertexShaderConstant returned %#x.\n", hr
);
1997 ok(!memcmp(ret
, initial
, sizeof(initial
)),
1998 "GetVertexShaderConstant got {%f, %f, %f, %f}, expected {%f, %f, %f, %f}\n",
1999 ret
[0], ret
[1], ret
[2], ret
[3], initial
[0], initial
[1], initial
[2], initial
[3]);
2003 hr
= IDirect3DDevice8_GetPixelShaderConstant(device
, 0, ret
, 1);
2004 ok(SUCCEEDED(hr
), "GetPixelShaderConstant returned %#x.\n", hr
);
2005 ok(!memcmp(ret
, ps_const
, sizeof(ps_const
)),
2006 "GetPixelShaderConstant got {%f, %f, %f, %f}, expected {%f, %f, %f, %f}\n",
2007 ret
[0], ret
[1], ret
[2], ret
[3], ps_const
[0], ps_const
[1], ps_const
[2], ps_const
[3]);
2008 hr
= IDirect3DDevice8_GetPixelShaderConstant(device
, 1, ret
, 1);
2009 ok(SUCCEEDED(hr
), "GetPixelShaderConstant returned %#x.\n", hr
);
2010 ok(!memcmp(ret
, initial
, sizeof(initial
)),
2011 "GetPixelShaderConstant got {%f, %f, %f, %f}, expected {%f, %f, %f, %f}\n",
2012 ret
[0], ret
[1], ret
[2], ret
[3], initial
[0], initial
[1], initial
[2], initial
[3]);
2015 /* Apply doesn't overwrite constants that aren't explicitly set on the source stateblock. */
2016 hr
= IDirect3DDevice8_ApplyStateBlock(device
, stateblock
);
2017 ok(SUCCEEDED(hr
), "Apply returned %#x\n", hr
);
2021 hr
= IDirect3DDevice8_GetVertexShaderConstant(device
, 0, ret
, 1);
2022 ok(SUCCEEDED(hr
), "GetVertexShaderConstant returned %#x.\n", hr
);
2023 ok(!memcmp(ret
, vs_const
, sizeof(vs_const
)),
2024 "GetVertexShaderConstant got {%f, %f, %f, %f}, expected {%f, %f, %f, %f}\n",
2025 ret
[0], ret
[1], ret
[2], ret
[3], vs_const
[0], vs_const
[1], vs_const
[2], vs_const
[3]);
2026 hr
= IDirect3DDevice8_GetVertexShaderConstant(device
, 1, ret
, 1);
2027 ok(SUCCEEDED(hr
), "GetVertexShaderConstant returned %#x.\n", hr
);
2028 ok(!memcmp(ret
, vs_const
, sizeof(vs_const
)),
2029 "GetVertexShaderConstant got {%f, %f, %f, %f}, expected {%f, %f, %f, %f}\n",
2030 ret
[0], ret
[1], ret
[2], ret
[3], vs_const
[0], vs_const
[1], vs_const
[2], vs_const
[3]);
2034 hr
= IDirect3DDevice8_GetPixelShaderConstant(device
, 0, ret
, 1);
2035 ok(SUCCEEDED(hr
), "GetPixelShaderConstant returned %#x.\n", hr
);
2036 ok(!memcmp(ret
, ps_const
, sizeof(ps_const
)),
2037 "GetPixelShaderConstant got {%f, %f, %f, %f}, expected {%f, %f, %f, %f}\n",
2038 ret
[0], ret
[1], ret
[2], ret
[3], ps_const
[0], ps_const
[1], ps_const
[2], ps_const
[3]);
2039 hr
= IDirect3DDevice8_GetPixelShaderConstant(device
, 1, ret
, 1);
2040 ok(SUCCEEDED(hr
), "GetPixelShaderConstant returned %#x.\n", hr
);
2041 ok(!memcmp(ret
, ps_const
, sizeof(ps_const
)),
2042 "GetPixelShaderConstant got {%f, %f, %f, %f}, expected {%f, %f, %f, %f}\n",
2043 ret
[0], ret
[1], ret
[2], ret
[3], ps_const
[0], ps_const
[1], ps_const
[2], ps_const
[3]);
2046 IDirect3DDevice8_DeleteStateBlock(device
, stateblock
);
2049 START_TEST(stateblock
)
2051 IDirect3DDevice8
*device
= NULL
;
2052 D3DPRESENT_PARAMETERS device_pparams
;
2053 HMODULE d3d8_module
;
2057 d3d8_module
= LoadLibraryA("d3d8.dll");
2060 skip("Could not load d3d8.dll\n");
2064 hr
= init_d3d8(d3d8_module
, &device
, &device_pparams
);
2067 FreeLibrary(d3d8_module
);
2071 test_begin_end_state_block(device
);
2072 test_state_management(device
, &device_pparams
);
2073 test_shader_constant_apply(device
);
2075 refcount
= IDirect3DDevice8_Release(device
);
2076 ok(!refcount
, "Device has %u references left\n", refcount
);
2078 FreeLibrary(d3d8_module
);