| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2024-10-05 10:26:20 -05:00
										 |  |  |  * Copyright 2021-2024 Elizabeth Figura for CodeWeavers | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |  * | 
					
						
							|  |  |  |  * This library is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU Lesser General Public | 
					
						
							|  |  |  |  * License as published by the Free Software Foundation; either | 
					
						
							|  |  |  |  * version 2.1 of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This library is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
					
						
							|  |  |  |  * Lesser General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU Lesser General Public | 
					
						
							|  |  |  |  * License along with this library; if not, write to the Free Software | 
					
						
							|  |  |  |  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-18 13:53:51 +01:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | #define COBJMACROS
 | 
					
						
							|  |  |  | #define CONST_VTABLE
 | 
					
						
							|  |  |  | #define VKD3D_TEST_NO_DEFS
 | 
					
						
							|  |  |  | #include <d3d9.h>
 | 
					
						
							|  |  |  | #include "vkd3d_d3dcommon.h"
 | 
					
						
							|  |  |  | #include "vkd3d_d3dcompiler.h"
 | 
					
						
							|  |  |  | #include "shader_runner.h"
 | 
					
						
							|  |  |  | #include "vkd3d_test.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct d3d9_resource | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct resource r; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:37 -05:00
										 |  |  |     IDirect3DSurface9 *surface; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     IDirect3DTexture9 *texture; | 
					
						
							|  |  |  |     IDirect3DVertexBuffer9 *vb; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct d3d9_resource *d3d9_resource(struct resource *r) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return CONTAINING_RECORD(r, struct d3d9_resource, r); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct d3d9_shader_runner | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct shader_runner r; | 
					
						
							| 
									
										
										
										
											2024-02-19 16:47:11 +01:00
										 |  |  |     struct shader_runner_caps caps; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     IDirect3DDevice9 *device; | 
					
						
							|  |  |  |     HWND window; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct d3d9_shader_runner *d3d9_shader_runner(struct shader_runner *r) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return CONTAINING_RECORD(r, struct d3d9_shader_runner, r); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static IDirect3D9 *(WINAPI *pDirect3DCreate9)(UINT sdk_version); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-23 22:12:25 +02:00
										 |  |  | static HRESULT set_viewport(IDirect3DDevice9 *device, unsigned int x, unsigned int y, | 
					
						
							|  |  |  |         unsigned int width, unsigned int height, float min_depth, float max_depth) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     D3DVIEWPORT9 vp = | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         .X = x, | 
					
						
							|  |  |  |         .Y = y, | 
					
						
							|  |  |  |         .Width = width, | 
					
						
							|  |  |  |         .Height = height, | 
					
						
							|  |  |  |         .MinZ = min_depth, | 
					
						
							|  |  |  |         .MaxZ = max_depth, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return IDirect3DDevice9_SetViewport(device, &vp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | static void init_adapter_info(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     D3DADAPTER_IDENTIFIER9 identifier; | 
					
						
							|  |  |  |     IDirect3D9 *d3d; | 
					
						
							|  |  |  |     HRESULT hr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     d3d = pDirect3DCreate9(D3D_SDK_VERSION); | 
					
						
							|  |  |  |     ok(!!d3d, "Failed to create a D3D object.\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-21 17:46:36 -06:00
										 |  |  |     hr = IDirect3D9_GetAdapterIdentifier(d3d, test_options.adapter_idx, 0, &identifier); | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     ok(hr == S_OK, "Failed to get adapter identifier, hr %#lx.\n", hr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     trace("Driver string: %s.\n", identifier.Driver); | 
					
						
							|  |  |  |     trace("Device: %s, %04lx:%04lx.\n", identifier.Description, identifier.VendorId, identifier.DeviceId); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (identifier.VendorId == 0x1414 && identifier.DeviceId == 0x008c) | 
					
						
							|  |  |  |         trace("Using WARP device.\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     IDirect3D9_Release(d3d); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool init_test_context(struct d3d9_shader_runner *runner) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     D3DPRESENT_PARAMETERS present_parameters = | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         .Windowed = TRUE, | 
					
						
							|  |  |  |         .SwapEffect = D3DSWAPEFFECT_DISCARD, | 
					
						
							|  |  |  |         .BackBufferWidth = RENDER_TARGET_WIDTH, | 
					
						
							|  |  |  |         .BackBufferHeight = RENDER_TARGET_HEIGHT, | 
					
						
							|  |  |  |         .BackBufferFormat = D3DFMT_A8R8G8B8, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     RECT rect = {0, 0, RENDER_TARGET_WIDTH, RENDER_TARGET_HEIGHT}; | 
					
						
							|  |  |  |     IDirect3D9 *d3d; | 
					
						
							|  |  |  |     D3DCAPS9 caps; | 
					
						
							|  |  |  |     HRESULT hr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     memset(runner, 0, sizeof(*runner)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     AdjustWindowRect(&rect, WS_OVERLAPPEDWINDOW, FALSE); | 
					
						
							|  |  |  |     runner->window = CreateWindowA("static", "d3dcompiler_test", WS_OVERLAPPEDWINDOW, | 
					
						
							|  |  |  |             0, 0, rect.right - rect.left, rect.bottom - rect.top, NULL, NULL, NULL, NULL); | 
					
						
							|  |  |  |     ok(!!runner->window, "Failed to create a window.\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     d3d = pDirect3DCreate9(D3D_SDK_VERSION); | 
					
						
							|  |  |  |     ok(!!d3d, "Failed to create a D3D object.\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     present_parameters.hDeviceWindow = runner->window; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-11 12:34:51 +02:00
										 |  |  |     hr = IDirect3D9_CreateDevice(d3d, test_options.adapter_idx, D3DDEVTYPE_HAL, runner->window, | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |             D3DCREATE_HARDWARE_VERTEXPROCESSING, &present_parameters, &runner->device); | 
					
						
							|  |  |  |     IDirect3D9_Release(d3d); | 
					
						
							|  |  |  |     if (FAILED(hr)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         skip("Failed to create a 3D device, hr %#lx.\n", hr); | 
					
						
							|  |  |  |         DestroyWindow(runner->window); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     hr = IDirect3DDevice9_GetDeviceCaps(runner->device, &caps); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to get device caps, hr %#lx.\n", hr); | 
					
						
							|  |  |  |     if (caps.PixelShaderVersion < D3DPS_VERSION(2, 0) || caps.VertexShaderVersion < D3DVS_VERSION(2, 0)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         skip("No shader model 2 support.\n"); | 
					
						
							|  |  |  |         IDirect3DDevice9_Release(runner->device); | 
					
						
							|  |  |  |         DestroyWindow(runner->window); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-19 17:01:00 +01:00
										 |  |  |     runner->caps.runner = "d3d9.dll"; | 
					
						
							| 
									
										
										
										
											2025-03-06 22:00:57 +01:00
										 |  |  |     runner->caps.compiler = HLSL_COMPILER; | 
					
						
							| 
									
										
										
										
											2024-02-19 16:47:11 +01:00
										 |  |  |     runner->caps.minimum_shader_model = SHADER_MODEL_2_0; | 
					
						
							|  |  |  |     runner->caps.maximum_shader_model = SHADER_MODEL_3_0; | 
					
						
							| 
									
										
										
										
											2024-10-05 07:46:00 -05:00
										 |  |  |     runner->caps.shader_caps[SHADER_CAP_CLIP_PLANES] = true; | 
					
						
							| 
									
										
										
										
											2024-10-05 10:26:20 -05:00
										 |  |  |     runner->caps.shader_caps[SHADER_CAP_FOG] = true; | 
					
						
							| 
									
										
										
										
											2024-10-05 07:46:00 -05:00
										 |  |  |     runner->caps.shader_caps[SHADER_CAP_POINT_SIZE] = true; | 
					
						
							| 
									
										
										
										
											2024-02-19 16:47:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void destroy_test_context(struct d3d9_shader_runner *runner) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ULONG ref; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ref = IDirect3DDevice9_Release(runner->device); | 
					
						
							|  |  |  |     ok(!ref, "Device has %lu references left.\n", ref); | 
					
						
							|  |  |  |     DestroyWindow(runner->window); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static D3DTEXTUREADDRESS sampler_address_to_d3d9(D3D12_TEXTURE_ADDRESS_MODE address) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (address) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         case D3D12_TEXTURE_ADDRESS_MODE_WRAP: | 
					
						
							|  |  |  |             return D3DTADDRESS_WRAP; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case D3D12_TEXTURE_ADDRESS_MODE_MIRROR: | 
					
						
							|  |  |  |             return D3DTADDRESS_MIRROR; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case D3D12_TEXTURE_ADDRESS_MODE_CLAMP: | 
					
						
							|  |  |  |             return D3DTADDRESS_CLAMP; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case D3D12_TEXTURE_ADDRESS_MODE_BORDER: | 
					
						
							|  |  |  |             return D3DTADDRESS_BORDER; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case D3D12_TEXTURE_ADDRESS_MODE_MIRROR_ONCE: | 
					
						
							|  |  |  |             return D3DTADDRESS_MIRRORONCE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-31 13:25:24 +02:00
										 |  |  |     vkd3d_unreachable(); | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct resource *d3d9_runner_create_resource(struct shader_runner *r, const struct resource_params *params) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct d3d9_shader_runner *runner = d3d9_shader_runner(r); | 
					
						
							|  |  |  |     IDirect3DDevice9 *device = runner->device; | 
					
						
							|  |  |  |     struct d3d9_resource *resource; | 
					
						
							|  |  |  |     D3DLOCKED_RECT map_desc; | 
					
						
							|  |  |  |     D3DFORMAT format; | 
					
						
							|  |  |  |     HRESULT hr; | 
					
						
							|  |  |  |     void *data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resource = calloc(1, sizeof(*resource)); | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:35 -05:00
										 |  |  |     init_resource(&resource->r, params); | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-17 16:01:04 -05:00
										 |  |  |     switch (params->desc.format) | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:37 -05:00
										 |  |  |         case DXGI_FORMAT_R32G32B32A32_FLOAT: | 
					
						
							|  |  |  |             format = D3DFMT_A32B32G32R32F; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case DXGI_FORMAT_R32_FLOAT: | 
					
						
							|  |  |  |             format = D3DFMT_R32F; | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:37 -05:00
										 |  |  |         default: | 
					
						
							|  |  |  |             format = D3DFMT_UNKNOWN; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-17 16:01:04 -05:00
										 |  |  |     switch (params->desc.type) | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:37 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |         case RESOURCE_TYPE_RENDER_TARGET: | 
					
						
							| 
									
										
										
										
											2024-06-17 16:01:04 -05:00
										 |  |  |             hr = IDirect3DDevice9_CreateRenderTarget(device, params->desc.width, params->desc.height, | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:37 -05:00
										 |  |  |                     format, D3DMULTISAMPLE_NONE, 0, FALSE, &resource->surface, NULL); | 
					
						
							|  |  |  |             ok(hr == D3D_OK, "Failed to create render target, hr %#lx.\n", hr); | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-15 10:33:15 +10:00
										 |  |  |         case RESOURCE_TYPE_DEPTH_STENCIL: | 
					
						
							|  |  |  |             fatal_error("DSVs are not supported.\n"); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:37 -05:00
										 |  |  |         case RESOURCE_TYPE_TEXTURE: | 
					
						
							| 
									
										
										
										
											2023-04-30 19:23:00 -05:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2024-06-17 16:01:04 -05:00
										 |  |  |             if (params->desc.dimension == RESOURCE_DIMENSION_BUFFER) | 
					
						
							| 
									
										
										
										
											2024-01-23 20:26:11 -03:00
										 |  |  |             { | 
					
						
							|  |  |  |                 fatal_error("Buffer resources are not supported.\n"); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-30 19:23:00 -05:00
										 |  |  |             unsigned int src_buffer_offset = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-17 16:01:04 -05:00
										 |  |  |             hr = IDirect3DDevice9_CreateTexture(device, params->desc.width, params->desc.height, | 
					
						
							|  |  |  |                     params->desc.level_count, 0, format, D3DPOOL_MANAGED, &resource->texture, NULL); | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |             ok(hr == D3D_OK, "Failed to create texture, hr %#lx.\n", hr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-17 16:01:04 -05:00
										 |  |  |             for (unsigned int level = 0; level < params->desc.level_count; ++level) | 
					
						
							| 
									
										
										
										
											2023-04-30 19:23:00 -05:00
										 |  |  |             { | 
					
						
							| 
									
										
										
										
											2024-06-17 16:01:04 -05:00
										 |  |  |                 unsigned int level_width = get_level_dimension(params->desc.width, level); | 
					
						
							|  |  |  |                 unsigned int level_height = get_level_dimension(params->desc.height, level); | 
					
						
							|  |  |  |                 unsigned int src_row_pitch = level_width * params->desc.texel_size; | 
					
						
							| 
									
										
										
										
											2023-04-30 19:23:00 -05:00
										 |  |  |                 unsigned int src_slice_pitch = level_height * src_row_pitch; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 hr = IDirect3DTexture9_LockRect(resource->texture, level, &map_desc, NULL, 0); | 
					
						
							|  |  |  |                 ok(hr == D3D_OK, "Failed to map texture, hr %#lx.\n", hr); | 
					
						
							|  |  |  |                 for (unsigned int y = 0; y < level_height; ++y) | 
					
						
							|  |  |  |                     memcpy(&((char *)map_desc.pBits)[y * map_desc.Pitch], | 
					
						
							|  |  |  |                             ¶ms->data[src_buffer_offset + y * src_row_pitch], src_row_pitch); | 
					
						
							|  |  |  |                 hr = IDirect3DTexture9_UnlockRect(resource->texture, level); | 
					
						
							|  |  |  |                 ok(hr == D3D_OK, "Failed to unmap texture, hr %#lx.\n", hr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 src_buffer_offset += src_slice_pitch; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2023-04-30 19:23:00 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:35 -05:00
										 |  |  |         case RESOURCE_TYPE_UAV: | 
					
						
							|  |  |  |             fatal_error("UAVs are not supported.\n"); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |         case RESOURCE_TYPE_VERTEX_BUFFER: | 
					
						
							|  |  |  |             hr = IDirect3DDevice9_CreateVertexBuffer(device, params->data_size, | 
					
						
							|  |  |  |                     D3DUSAGE_DYNAMIC, 0, D3DPOOL_DEFAULT, &resource->vb, NULL); | 
					
						
							|  |  |  |             ok(hr == D3D_OK, "Failed to create vertex buffer, hr %#lx.\n", hr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             hr = IDirect3DVertexBuffer9_Lock(resource->vb, 0, 0, &data, D3DLOCK_DISCARD); | 
					
						
							|  |  |  |             ok(hr == D3D_OK, "Failed to map texture, hr %#lx.\n", hr); | 
					
						
							|  |  |  |             memcpy(data, params->data, params->data_size); | 
					
						
							|  |  |  |             hr = IDirect3DVertexBuffer9_Unlock(resource->vb); | 
					
						
							|  |  |  |             ok(hr == D3D_OK, "Failed to unmap texture, hr %#lx.\n", hr); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return &resource->r; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void d3d9_runner_destroy_resource(struct shader_runner *r, struct resource *res) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct d3d9_resource *resource = d3d9_resource(res); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:37 -05:00
										 |  |  |     if (resource->surface) | 
					
						
							|  |  |  |         IDirect3DSurface9_Release(resource->surface); | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     if (resource->texture) | 
					
						
							|  |  |  |         IDirect3DTexture9_Release(resource->texture); | 
					
						
							|  |  |  |     if (resource->vb) | 
					
						
							|  |  |  |         IDirect3DVertexBuffer9_Release(resource->vb); | 
					
						
							|  |  |  |     free(resource); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static D3DDECLTYPE vertex_decl_type_from_format(DXGI_FORMAT format) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (format) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-10-03 12:35:11 -05:00
										 |  |  |         case DXGI_FORMAT_R32_FLOAT: | 
					
						
							|  |  |  |             return D3DDECLTYPE_FLOAT1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |         case DXGI_FORMAT_R32G32_FLOAT: | 
					
						
							|  |  |  |             return D3DDECLTYPE_FLOAT2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case DXGI_FORMAT_R32G32B32A32_FLOAT: | 
					
						
							|  |  |  |             return D3DDECLTYPE_FLOAT4; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             fatal_error("Cannot translate format %#x to a d3d9 vertex buffer format.\n", format); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static D3DDECLUSAGE vertex_decl_usage_from_name(const char *name) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-10-05 11:45:25 -05:00
										 |  |  |     if (!strcasecmp(name, "color")) | 
					
						
							|  |  |  |         return D3DDECLUSAGE_COLOR; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     if (!strcasecmp(name, "position") || !strcasecmp(name, "sv_position")) | 
					
						
							|  |  |  |         return D3DDECLUSAGE_POSITION; | 
					
						
							| 
									
										
										
										
											2024-10-03 12:35:11 -05:00
										 |  |  |     if (!strcasecmp(name, "psize")) | 
					
						
							|  |  |  |         return D3DDECLUSAGE_PSIZE; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     if (!strcasecmp(name, "texcoord")) | 
					
						
							|  |  |  |         return D3DDECLUSAGE_TEXCOORD; | 
					
						
							|  |  |  |     fatal_error("Cannot translate usage \"%s\" to a d3d9 usage.\n", name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 16:37:31 -05:00
										 |  |  | static bool d3d9_runner_dispatch(struct shader_runner *r, unsigned int x, unsigned int y, unsigned int z) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     fatal_error("Compute shaders are not supported.\n"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-05 10:26:20 -05:00
										 |  |  | static uint32_t d3d_color_from_vec4(const struct vec4 *v) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return vkd3d_make_u32(vkd3d_make_u16(v->z * 255.0f, v->y * 255.0f), | 
					
						
							|  |  |  |             vkd3d_make_u16(v->x * 255.0f, v->w * 255.0f)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-15 10:33:15 +10:00
										 |  |  | static void d3d9_runner_clear(struct shader_runner *r, struct resource *resource, const struct vec4 *clear_value) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-05-17 19:53:12 -05:00
										 |  |  |     struct d3d9_shader_runner *runner = d3d9_shader_runner(r); | 
					
						
							|  |  |  |     HRESULT hr; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-05 10:26:20 -05:00
										 |  |  |     hr = IDirect3DDevice9_ColorFill(runner->device, d3d9_resource(resource)->surface, | 
					
						
							|  |  |  |             NULL, d3d_color_from_vec4(clear_value)); | 
					
						
							| 
									
										
										
										
											2024-05-17 19:53:12 -05:00
										 |  |  |     ok(hr == S_OK, "Got hr %#lx.\n", hr); | 
					
						
							| 
									
										
										
										
											2024-04-15 10:33:15 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-14 12:52:32 +02:00
										 |  |  | static bool d3d9_runner_draw(struct shader_runner *r, | 
					
						
							| 
									
										
										
										
											2024-04-22 12:55:44 +10:00
										 |  |  |         D3D_PRIMITIVE_TOPOLOGY primitive_topology, unsigned int vertex_count, unsigned int instance_count) | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | { | 
					
						
							|  |  |  |     static const D3DVERTEXELEMENT9 decl_element_end = D3DDECL_END(); | 
					
						
							|  |  |  |     struct d3d9_shader_runner *runner = d3d9_shader_runner(r); | 
					
						
							|  |  |  |     IDirect3DVertexDeclaration9 *vertex_declaration; | 
					
						
							|  |  |  |     IDirect3DDevice9 *device = runner->device; | 
					
						
							| 
									
										
										
										
											2024-10-23 22:12:25 +02:00
										 |  |  |     unsigned int fb_width, fb_height, i, j; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     D3DVERTEXELEMENT9 *decl_elements; | 
					
						
							|  |  |  |     ID3D10Blob *vs_code, *ps_code; | 
					
						
							|  |  |  |     IDirect3DVertexShader9 *vs; | 
					
						
							|  |  |  |     IDirect3DPixelShader9 *ps; | 
					
						
							| 
									
										
										
										
											2024-11-13 19:16:39 -06:00
										 |  |  |     D3DMATRIX proj_matrix; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     HRESULT hr; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-22 12:55:44 +10:00
										 |  |  |     if (instance_count > 1) | 
					
						
							|  |  |  |         fatal_error("Unhandled instance count %u.\n", instance_count); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-24 15:58:32 -05:00
										 |  |  |     if (!(vs_code = compile_hlsl(&runner->r, SHADER_TYPE_VS))) | 
					
						
							| 
									
										
										
										
											2022-04-14 12:52:32 +02:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-24 15:58:32 -05:00
										 |  |  |     if (!(ps_code = compile_hlsl(&runner->r, SHADER_TYPE_PS))) | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |         ID3D10Blob_Release(vs_code); | 
					
						
							| 
									
										
										
										
											2022-04-14 12:52:32 +02:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (runner->r.uniform_count) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-02-26 20:02:39 -03:00
										 |  |  |         hr = IDirect3DDevice9_SetVertexShaderConstantF(device, 0, | 
					
						
							|  |  |  |                 (const float *)runner->r.uniforms, runner->r.uniform_count / 4); | 
					
						
							|  |  |  |         ok(hr == D3D_OK, "Failed to set uniforms, hr %#lx.\n", hr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |         hr = IDirect3DDevice9_SetPixelShaderConstantF(device, 0, | 
					
						
							|  |  |  |                 (const float *)runner->r.uniforms, runner->r.uniform_count / 4); | 
					
						
							|  |  |  |         ok(hr == D3D_OK, "Failed to set uniforms, hr %#lx.\n", hr); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     decl_elements = calloc(runner->r.input_element_count + 1, sizeof(*decl_elements)); | 
					
						
							|  |  |  |     for (i = 0; i < runner->r.input_element_count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         const struct input_element *src_element = &runner->r.input_elements[i]; | 
					
						
							|  |  |  |         D3DVERTEXELEMENT9 *dst_element = &decl_elements[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         dst_element->Stream = src_element->slot; | 
					
						
							|  |  |  |         dst_element->Type = vertex_decl_type_from_format(src_element->format); | 
					
						
							|  |  |  |         dst_element->Method = D3DDECLMETHOD_DEFAULT; | 
					
						
							|  |  |  |         dst_element->Usage = vertex_decl_usage_from_name(src_element->name); | 
					
						
							|  |  |  |         dst_element->UsageIndex = src_element->index; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* The offset will be filled below. */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     decl_elements[runner->r.input_element_count] = decl_element_end; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-23 22:12:25 +02:00
										 |  |  |     fb_width = ~0u; | 
					
						
							|  |  |  |     fb_height = ~0u; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     for (i = 0; i < runner->r.resource_count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         struct d3d9_resource *resource = d3d9_resource(runner->r.resources[i]); | 
					
						
							|  |  |  |         unsigned int stride = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-17 16:01:04 -05:00
										 |  |  |         switch (resource->r.desc.type) | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:37 -05:00
										 |  |  |             case RESOURCE_TYPE_RENDER_TARGET: | 
					
						
							| 
									
										
										
										
											2024-06-17 16:01:04 -05:00
										 |  |  |                 hr = IDirect3DDevice9_SetRenderTarget(device, resource->r.desc.slot, resource->surface); | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:37 -05:00
										 |  |  |                 ok(hr == D3D_OK, "Failed to set render target, hr %#lx.\n", hr); | 
					
						
							| 
									
										
										
										
											2024-10-23 22:12:25 +02:00
										 |  |  |                 if (resource->r.desc.width < fb_width) | 
					
						
							|  |  |  |                     fb_width = resource->r.desc.width; | 
					
						
							|  |  |  |                 if (resource->r.desc.height < fb_height) | 
					
						
							|  |  |  |                     fb_height = resource->r.desc.height; | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:37 -05:00
										 |  |  |                 break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-15 10:33:15 +10:00
										 |  |  |             case RESOURCE_TYPE_DEPTH_STENCIL: | 
					
						
							|  |  |  |                 vkd3d_unreachable(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |             case RESOURCE_TYPE_TEXTURE: | 
					
						
							| 
									
										
										
										
											2024-06-17 16:01:04 -05:00
										 |  |  |                 assert(resource->r.desc.dimension != RESOURCE_DIMENSION_BUFFER); | 
					
						
							| 
									
										
										
										
											2024-01-23 20:26:11 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-17 16:01:04 -05:00
										 |  |  |                 hr = IDirect3DDevice9_SetTexture(device, resource->r.desc.slot, (IDirect3DBaseTexture9 *)resource->texture); | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |                 ok(hr == D3D_OK, "Failed to set texture, hr %#lx.\n", hr); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:35 -05:00
										 |  |  |             case RESOURCE_TYPE_UAV: | 
					
						
							| 
									
										
										
										
											2022-08-31 13:25:24 +02:00
										 |  |  |                 vkd3d_unreachable(); | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:35 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |             case RESOURCE_TYPE_VERTEX_BUFFER: | 
					
						
							|  |  |  |                 for (j = 0; j < runner->r.input_element_count; ++j) | 
					
						
							|  |  |  |                 { | 
					
						
							| 
									
										
										
										
											2024-06-17 16:01:04 -05:00
										 |  |  |                     if (runner->r.input_elements[j].slot == resource->r.desc.slot) | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |                     { | 
					
						
							|  |  |  |                         decl_elements[j].Offset = stride; | 
					
						
							|  |  |  |                         stride += runner->r.input_elements[j].texel_size; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-17 16:01:04 -05:00
										 |  |  |                 hr = IDirect3DDevice9_SetStreamSource(device, resource->r.desc.slot, resource->vb, 0, stride); | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |                 ok(hr == D3D_OK, "Failed to set vertex buffer, hr %#lx.\n", hr); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-23 22:12:25 +02:00
										 |  |  |     hr = set_viewport(device, 0, 0, fb_width, fb_height, 0.0f, 1.0f); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set viewport, hr %#lx.\n", hr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     for (i = 0; i < runner->r.sampler_count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         const struct sampler *sampler = &runner->r.samplers[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         hr = IDirect3DDevice9_SetSamplerState(device, sampler->slot, | 
					
						
							|  |  |  |                 D3DSAMP_ADDRESSU, sampler_address_to_d3d9(sampler->u_address)); | 
					
						
							|  |  |  |         ok(hr == D3D_OK, "Failed to set sampler state, hr %#lx.\n", hr); | 
					
						
							|  |  |  |         hr = IDirect3DDevice9_SetSamplerState(device, sampler->slot, | 
					
						
							|  |  |  |                 D3DSAMP_ADDRESSV, sampler_address_to_d3d9(sampler->v_address)); | 
					
						
							|  |  |  |         ok(hr == D3D_OK, "Failed to set sampler state, hr %#lx.\n", hr); | 
					
						
							|  |  |  |         hr = IDirect3DDevice9_SetSamplerState(device, sampler->slot, | 
					
						
							|  |  |  |                 D3DSAMP_ADDRESSW, sampler_address_to_d3d9(sampler->w_address)); | 
					
						
							|  |  |  |         ok(hr == D3D_OK, "Failed to set sampler state, hr %#lx.\n", hr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         hr = IDirect3DDevice9_SetSamplerState(device, sampler->slot, D3DSAMP_MINFILTER, | 
					
						
							|  |  |  |                 (sampler->filter & 0x1) ? D3DTEXF_LINEAR : D3DTEXF_POINT); | 
					
						
							|  |  |  |         ok(hr == D3D_OK, "Failed to set sampler state, hr %#lx.\n", hr); | 
					
						
							|  |  |  |         hr = IDirect3DDevice9_SetSamplerState(device, sampler->slot, D3DSAMP_MAGFILTER, | 
					
						
							|  |  |  |                 (sampler->filter & 0x4) ? D3DTEXF_LINEAR : D3DTEXF_POINT); | 
					
						
							|  |  |  |         ok(hr == D3D_OK, "Failed to set sampler state, hr %#lx.\n", hr); | 
					
						
							|  |  |  |         hr = IDirect3DDevice9_SetSamplerState(device, sampler->slot, D3DSAMP_MIPFILTER, | 
					
						
							|  |  |  |                 (sampler->filter & 0x10) ? D3DTEXF_LINEAR : D3DTEXF_POINT); | 
					
						
							|  |  |  |         ok(hr == D3D_OK, "Failed to set sampler state, hr %#lx.\n", hr); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-17 19:53:12 -05:00
										 |  |  |     hr = IDirect3DDevice9_SetRenderState(device, D3DRS_ALPHATESTENABLE, TRUE); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set render state, hr %#lx.\n", hr); | 
					
						
							|  |  |  |     /* The members of enum vkd3d_shader_parameter_alpha_test_func are
 | 
					
						
							|  |  |  |      * compatible with D3DCMPFUNC. */ | 
					
						
							|  |  |  |     hr = IDirect3DDevice9_SetRenderState(device, D3DRS_ALPHAFUNC, runner->r.alpha_test_func); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set render state, hr %#lx.\n", hr); | 
					
						
							|  |  |  |     hr = IDirect3DDevice9_SetRenderState(device, D3DRS_ALPHAREF, runner->r.alpha_test_ref * 255.0); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set render state, hr %#lx.\n", hr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-11 17:31:02 -05:00
										 |  |  |     hr = IDirect3DDevice9_SetRenderState(device, D3DRS_SHADEMODE, | 
					
						
							|  |  |  |             runner->r.flat_shading ? D3DSHADE_FLAT : D3DSHADE_GOURAUD); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set render state, hr %#lx.\n", hr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-16 19:49:30 -05:00
										 |  |  |     hr = IDirect3DDevice9_SetRenderState(device, D3DRS_CLIPPING, TRUE); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set render state, hr %#lx.\n", hr); | 
					
						
							|  |  |  |     hr = IDirect3DDevice9_SetRenderState(device, D3DRS_CLIPPLANEENABLE, runner->r.clip_plane_mask); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set render state, hr %#lx.\n", hr); | 
					
						
							|  |  |  |     for (unsigned int i = 0; i < 8; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         hr = IDirect3DDevice9_SetClipPlane(device, i, &runner->r.clip_planes[i].x); | 
					
						
							|  |  |  |         ok(hr == D3D_OK, "Failed to set clip plane, hr %#lx.\n", hr); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-03 16:23:04 -05:00
										 |  |  |     hr = IDirect3DDevice9_SetRenderState(device, D3DRS_POINTSIZE, float_to_int(runner->r.point_size)); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set render state, hr %#lx.\n", hr); | 
					
						
							| 
									
										
										
										
											2024-10-03 17:55:33 -05:00
										 |  |  |     hr = IDirect3DDevice9_SetRenderState(device, D3DRS_POINTSIZE_MIN, float_to_int(runner->r.point_size_min)); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set render state, hr %#lx.\n", hr); | 
					
						
							|  |  |  |     hr = IDirect3DDevice9_SetRenderState(device, D3DRS_POINTSIZE_MAX, float_to_int(runner->r.point_size_max)); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set render state, hr %#lx.\n", hr); | 
					
						
							| 
									
										
										
										
											2024-10-15 19:04:25 -05:00
										 |  |  |     hr = IDirect3DDevice9_SetRenderState(device, D3DRS_POINTSPRITEENABLE, runner->r.point_sprite); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set render state, hr %#lx.\n", hr); | 
					
						
							| 
									
										
										
										
											2024-10-03 16:23:04 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-05 10:26:20 -05:00
										 |  |  |     hr = IDirect3DDevice9_SetRenderState(device, D3DRS_FOGENABLE, (runner->r.fog_mode != FOG_MODE_DISABLE)); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set render state, hr %#lx.\n", hr); | 
					
						
							|  |  |  |     if (runner->r.fog_mode != FOG_MODE_DISABLE) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         hr = IDirect3DDevice9_SetRenderState(device, D3DRS_FOGTABLEMODE, runner->r.fog_mode); | 
					
						
							|  |  |  |         ok(hr == D3D_OK, "Failed to set render state, hr %#lx.\n", hr); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-11-13 19:16:39 -06:00
										 |  |  |     memset(&proj_matrix, 0, sizeof(proj_matrix)); | 
					
						
							|  |  |  |     proj_matrix._11 = proj_matrix._22 = proj_matrix._33 = proj_matrix._44 = 1.0f; | 
					
						
							|  |  |  |     if (!runner->r.ortho_fog) | 
					
						
							|  |  |  |         proj_matrix._44 = 1.01f; | 
					
						
							|  |  |  |     hr = IDirect3DDevice9_SetTransform(device, D3DTS_PROJECTION, &proj_matrix); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set projection matrix, hr %#lx.\n", hr); | 
					
						
							| 
									
										
										
										
											2024-10-05 10:26:20 -05:00
										 |  |  |     hr = IDirect3DDevice9_SetRenderState(device, D3DRS_FOGCOLOR, d3d_color_from_vec4(&runner->r.fog_colour)); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set render state, hr %#lx.\n", hr); | 
					
						
							| 
									
										
										
										
											2024-11-13 19:16:39 -06:00
										 |  |  |     hr = IDirect3DDevice9_SetRenderState(device, D3DRS_FOGSTART, float_to_int(runner->r.fog_start)); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set render state, hr %#lx.\n", hr); | 
					
						
							|  |  |  |     hr = IDirect3DDevice9_SetRenderState(device, D3DRS_FOGEND, float_to_int(runner->r.fog_end)); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set render state, hr %#lx.\n", hr); | 
					
						
							| 
									
										
										
										
											2024-11-13 19:39:35 -06:00
										 |  |  |     hr = IDirect3DDevice9_SetRenderState(device, D3DRS_FOGDENSITY, float_to_int(runner->r.fog_density)); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set render state, hr %#lx.\n", hr); | 
					
						
							| 
									
										
										
										
											2024-10-05 10:26:20 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     hr = IDirect3DDevice9_CreateVertexDeclaration(device, decl_elements, &vertex_declaration); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to create vertex declaration, hr %#lx.\n", hr); | 
					
						
							|  |  |  |     hr = IDirect3DDevice9_CreateVertexShader(device, ID3D10Blob_GetBufferPointer(vs_code), &vs); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to create vertex shader, hr %#lx.\n", hr); | 
					
						
							|  |  |  |     hr = IDirect3DDevice9_CreatePixelShader(device, ID3D10Blob_GetBufferPointer(ps_code), &ps); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to create pixel shader, hr %#lx.\n", hr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     hr = IDirect3DDevice9_SetVertexDeclaration(device, vertex_declaration); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set vertex declaration, hr %#lx.\n", hr); | 
					
						
							|  |  |  |     hr = IDirect3DDevice9_SetVertexShader(device, vs); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set vertex shader, hr %#lx.\n", hr); | 
					
						
							|  |  |  |     hr = IDirect3DDevice9_SetPixelShader(device, ps); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set pixel shader, hr %#lx.\n", hr); | 
					
						
							| 
									
										
										
										
											2023-04-10 20:45:15 +02:00
										 |  |  |     hr = IDirect3DDevice9_SetRenderState(device, D3DRS_CULLMODE, D3DCULL_NONE); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to set cull mode, hr %#lx.\n", hr); | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     hr = IDirect3DDevice9_BeginScene(device); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to draw, hr %#lx.\n", hr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (primitive_topology) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-10-03 12:35:11 -05:00
										 |  |  |         case D3D_PRIMITIVE_TOPOLOGY_POINTLIST: | 
					
						
							|  |  |  |             hr = IDirect3DDevice9_DrawPrimitive(device, D3DPT_POINTLIST, 0, vertex_count); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |         case D3D_PRIMITIVE_TOPOLOGY_TRIANGLELIST: | 
					
						
							|  |  |  |             hr = IDirect3DDevice9_DrawPrimitive(device, D3DPT_TRIANGLELIST, 0, vertex_count / 3); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case D3D_PRIMITIVE_TOPOLOGY_TRIANGLESTRIP: | 
					
						
							|  |  |  |             hr = IDirect3DDevice9_DrawPrimitive(device, D3DPT_TRIANGLESTRIP, 0, vertex_count - 2); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             fatal_error("Cannot translate topology %#x to a d3d9 topology.\n", primitive_topology); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to draw, hr %#lx.\n", hr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     hr = IDirect3DDevice9_EndScene(device); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to draw, hr %#lx.\n", hr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     IDirect3DVertexDeclaration9_Release(vertex_declaration); | 
					
						
							|  |  |  |     IDirect3DVertexShader9_Release(vs); | 
					
						
							|  |  |  |     IDirect3DPixelShader9_Release(ps); | 
					
						
							| 
									
										
										
										
											2022-04-14 12:52:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-21 00:21:00 +02:00
										 |  |  | static bool d3d9_runner_copy(struct shader_runner *r, struct resource *src, struct resource *dst) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-17 15:45:38 +02:00
										 |  |  | struct d3d9_resource_readback | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-05-17 15:45:38 +02:00
										 |  |  |     struct resource_readback rb; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     IDirect3DSurface9 *surface; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-01 12:31:06 +10:00
										 |  |  | static struct resource_readback *d3d9_runner_get_resource_readback(struct shader_runner *r, | 
					
						
							|  |  |  |         struct resource *res, unsigned int sub_resource_idx) | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-05-17 15:45:41 +02:00
										 |  |  |     struct d3d9_shader_runner *runner = d3d9_shader_runner(r); | 
					
						
							|  |  |  |     struct d3d9_resource_readback *rb = malloc(sizeof(*rb)); | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:37 -05:00
										 |  |  |     struct d3d9_resource *resource = d3d9_resource(res); | 
					
						
							| 
									
										
										
										
											2022-05-17 15:45:38 +02:00
										 |  |  |     D3DLOCKED_RECT map_desc; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     D3DSURFACE_DESC desc; | 
					
						
							|  |  |  |     HRESULT hr; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-17 16:01:04 -05:00
										 |  |  |     assert(resource->r.desc.type == RESOURCE_TYPE_RENDER_TARGET); | 
					
						
							| 
									
										
										
										
											2024-05-01 12:31:06 +10:00
										 |  |  |     if (sub_resource_idx) | 
					
						
							|  |  |  |         fatal_error("Unsupported sub-resource index %u.\n", sub_resource_idx); | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:37 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     hr = IDirect3DSurface9_GetDesc(resource->surface, &desc); | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     ok(hr == D3D_OK, "Failed to get surface desc, hr %#lx.\n", hr); | 
					
						
							|  |  |  |     hr = IDirect3DDevice9Ex_CreateOffscreenPlainSurface(runner->device, desc.Width, | 
					
						
							|  |  |  |             desc.Height, desc.Format, D3DPOOL_SYSTEMMEM, &rb->surface, NULL); | 
					
						
							|  |  |  |     ok(hr == D3D_OK, "Failed to create surface, hr %#lx.\n", hr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:37 -05:00
										 |  |  |     hr = IDirect3DDevice9Ex_GetRenderTargetData(runner->device, resource->surface, rb->surface); | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     ok(hr == D3D_OK, "Failed to get render target data, hr %#lx.\n", hr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-17 15:45:38 +02:00
										 |  |  |     hr = IDirect3DSurface9_LockRect(rb->surface, &map_desc, NULL, D3DLOCK_READONLY); | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     ok(hr == D3D_OK, "Failed to lock surface, hr %#lx.\n", hr); | 
					
						
							| 
									
										
										
										
											2022-05-17 15:45:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     rb->rb.data = map_desc.pBits; | 
					
						
							|  |  |  |     rb->rb.row_pitch = map_desc.Pitch; | 
					
						
							|  |  |  |     rb->rb.width = desc.Width; | 
					
						
							|  |  |  |     rb->rb.height = desc.Height; | 
					
						
							|  |  |  |     rb->rb.depth = 1; | 
					
						
							| 
									
										
										
										
											2022-05-17 15:45:41 +02:00
										 |  |  |     return &rb->rb; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-17 15:45:41 +02:00
										 |  |  | static void d3d9_runner_release_readback(struct shader_runner *r, struct resource_readback *rb) | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2022-05-17 15:45:41 +02:00
										 |  |  |     struct d3d9_resource_readback *d3d9_rb = CONTAINING_RECORD(rb, struct d3d9_resource_readback, rb); | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-17 15:45:41 +02:00
										 |  |  |     IDirect3DSurface9_UnlockRect(d3d9_rb->surface); | 
					
						
							|  |  |  |     IDirect3DSurface9_Release(d3d9_rb->surface); | 
					
						
							|  |  |  |     free(d3d9_rb); | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct shader_runner_ops d3d9_runner_ops = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     .create_resource = d3d9_runner_create_resource, | 
					
						
							|  |  |  |     .destroy_resource = d3d9_runner_destroy_resource, | 
					
						
							| 
									
										
										
										
											2021-08-13 16:37:31 -05:00
										 |  |  |     .dispatch = d3d9_runner_dispatch, | 
					
						
							| 
									
										
										
										
											2024-04-15 10:33:15 +10:00
										 |  |  |     .clear = d3d9_runner_clear, | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     .draw = d3d9_runner_draw, | 
					
						
							| 
									
										
										
										
											2024-10-21 00:21:00 +02:00
										 |  |  |     .copy = d3d9_runner_copy, | 
					
						
							| 
									
										
										
										
											2022-06-08 18:18:37 -05:00
										 |  |  |     .get_resource_readback = d3d9_runner_get_resource_readback, | 
					
						
							| 
									
										
										
										
											2022-05-17 15:45:41 +02:00
										 |  |  |     .release_readback = d3d9_runner_release_readback, | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-21 17:46:36 -06:00
										 |  |  | void run_shader_tests_d3d9(void) | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | { | 
					
						
							|  |  |  |     struct d3d9_shader_runner runner; | 
					
						
							|  |  |  |     HMODULE d3d9_module; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-13 22:07:10 +01:00
										 |  |  |     if (test_skipping_execution("d3d9.dll", | 
					
						
							|  |  |  |             HLSL_COMPILER, SHADER_MODEL_2_0, SHADER_MODEL_3_0)) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-19 16:47:11 +01:00
										 |  |  |     if (!(d3d9_module = LoadLibraryA("d3d9.dll"))) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-19 16:47:11 +01:00
										 |  |  |     pDirect3DCreate9 = (void *)GetProcAddress(d3d9_module, "Direct3DCreate9"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     init_adapter_info(); | 
					
						
							|  |  |  |     if (init_test_context(&runner)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         run_shader_tests(&runner.r, &runner.caps, &d3d9_runner_ops, NULL); | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |         destroy_test_context(&runner); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-02-19 16:47:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-07 18:58:17 -05:00
										 |  |  |     FreeLibrary(d3d9_module); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2023-01-18 13:53:51 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #endif
 |