| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * FX (Direct3D 9/10/11 effect) support | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright 2023 Nikolay Sivov for CodeWeavers | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "hlsl.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | static inline size_t put_u32_unaligned(struct vkd3d_bytecode_buffer *buffer, uint32_t value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return bytecode_put_bytes_unaligned(buffer, &value, sizeof(value)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-11 15:52:30 +01:00
										 |  |  | enum fx_2_type_constants | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* Assignment types */ | 
					
						
							|  |  |  |     FX_2_ASSIGNMENT_CODE_BLOB = 0x0, | 
					
						
							|  |  |  |     FX_2_ASSIGNMENT_PARAMETER = 0x1, | 
					
						
							|  |  |  |     FX_2_ASSIGNMENT_ARRAY_SELECTOR = 0x2, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  | enum state_property_component_type | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     FX_BOOL, | 
					
						
							|  |  |  |     FX_FLOAT, | 
					
						
							|  |  |  |     FX_UINT, | 
					
						
							|  |  |  |     FX_UINT8, | 
					
						
							|  |  |  |     FX_DEPTHSTENCIL, | 
					
						
							|  |  |  |     FX_RASTERIZER, | 
					
						
							|  |  |  |     FX_DOMAINSHADER, | 
					
						
							|  |  |  |     FX_HULLSHADER, | 
					
						
							|  |  |  |     FX_COMPUTESHADER, | 
					
						
							|  |  |  |     FX_TEXTURE, | 
					
						
							|  |  |  |     FX_DEPTHSTENCILVIEW, | 
					
						
							|  |  |  |     FX_RENDERTARGETVIEW, | 
					
						
							|  |  |  |     FX_BLEND, | 
					
						
							|  |  |  |     FX_VERTEXSHADER, | 
					
						
							|  |  |  |     FX_PIXELSHADER, | 
					
						
							|  |  |  |     FX_GEOMETRYSHADER, | 
					
						
							|  |  |  |     FX_COMPONENT_TYPE_COUNT, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 16:33:57 +02:00
										 |  |  | enum fxlvm_constants | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     FX_FXLC_COMP_COUNT_MASK = 0xffff, | 
					
						
							|  |  |  |     FX_FXLC_OPCODE_MASK = 0x7ff, | 
					
						
							|  |  |  |     FX_FXLC_OPCODE_SHIFT = 20, | 
					
						
							|  |  |  |     FX_FXLC_IS_SCALAR_MASK = 0x80000000, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-13 12:01:19 +02:00
										 |  |  |     FX_FXLC_REG_UNUSED = 0, | 
					
						
							| 
									
										
										
										
											2025-06-07 16:33:57 +02:00
										 |  |  |     FX_FXLC_REG_LITERAL = 1, | 
					
						
							|  |  |  |     FX_FXLC_REG_CB = 2, | 
					
						
							| 
									
										
										
										
											2025-06-13 12:01:19 +02:00
										 |  |  |     FX_FXLC_REG_INPUT = 3, | 
					
						
							| 
									
										
										
										
											2025-06-07 16:33:57 +02:00
										 |  |  |     FX_FXLC_REG_OUTPUT = 4, | 
					
						
							|  |  |  |     FX_FXLC_REG_TEMP = 7, | 
					
						
							| 
									
										
										
										
											2025-06-09 16:50:31 +02:00
										 |  |  |     FX_FXLC_REG_MAX = FX_FXLC_REG_TEMP, | 
					
						
							| 
									
										
										
										
											2025-06-07 16:33:57 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  | struct rhs_named_value | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const char *name; | 
					
						
							|  |  |  |     unsigned int value; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct fx_assignment | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t id; | 
					
						
							|  |  |  |     uint32_t lhs_index; | 
					
						
							|  |  |  |     uint32_t type; | 
					
						
							|  |  |  |     uint32_t value; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-17 11:36:34 +02:00
										 |  |  | struct fx_4_binary_type | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t name; | 
					
						
							|  |  |  |     uint32_t class; | 
					
						
							|  |  |  |     uint32_t element_count; | 
					
						
							|  |  |  |     uint32_t unpacked_size; | 
					
						
							|  |  |  |     uint32_t stride; | 
					
						
							|  |  |  |     uint32_t packed_size; | 
					
						
							|  |  |  |     uint32_t typeinfo; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-22 02:17:25 +01:00
										 |  |  | struct fx_5_shader | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t offset; | 
					
						
							|  |  |  |     uint32_t sodecl[4]; | 
					
						
							|  |  |  |     uint32_t sodecl_count; | 
					
						
							|  |  |  |     uint32_t rast_stream; | 
					
						
							|  |  |  |     uint32_t iface_bindings_count; | 
					
						
							|  |  |  |     uint32_t iface_bindings; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-12 14:33:05 +01:00
										 |  |  | struct string_entry | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct rb_entry entry; | 
					
						
							|  |  |  |     /* String points to original data, should not be freed. */ | 
					
						
							|  |  |  |     const char *string; | 
					
						
							|  |  |  |     uint32_t offset; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | struct type_entry | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct list entry; | 
					
						
							|  |  |  |     const char *name; | 
					
						
							|  |  |  |     uint32_t elements_count; | 
					
						
							| 
									
										
										
										
											2024-09-08 04:42:11 +02:00
										 |  |  |     uint32_t modifiers; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     uint32_t offset; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-12 14:33:05 +01:00
										 |  |  | static int string_storage_compare(const void *key, const struct rb_entry *entry) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct string_entry *string_entry = RB_ENTRY_VALUE(entry, struct string_entry, entry); | 
					
						
							|  |  |  |     const char *string = key; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return strcmp(string, string_entry->string); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void string_storage_destroy(struct rb_entry *entry, void *context) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct string_entry *string_entry = RB_ENTRY_VALUE(entry, struct string_entry, entry); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_free(string_entry); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  | struct function_component | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const char *name; | 
					
						
							|  |  |  |     bool lhs_has_index; | 
					
						
							|  |  |  |     unsigned int lhs_index; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct state_block_function_info | 
					
						
							| 
									
										
										
										
											2024-06-06 17:50:05 -04:00
										 |  |  | { | 
					
						
							|  |  |  |     const char *name; | 
					
						
							|  |  |  |     unsigned int min_args, max_args; | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |     const struct function_component components[3]; | 
					
						
							| 
									
										
										
										
											2024-08-05 20:32:24 +02:00
										 |  |  |     unsigned int min_profile; | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  | } | 
					
						
							|  |  |  | function_info[] = | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-08-05 20:32:24 +02:00
										 |  |  |     {"SetBlendState",        3, 3, { { "AB_BlendFactor" }, { "AB_SampleMask" }, { "BlendState" } }, 4 }, | 
					
						
							|  |  |  |     {"SetDepthStencilState", 2, 2, { { "DS_StencilRef" }, { "DepthStencilState" } }, 4 }, | 
					
						
							|  |  |  |     {"SetRasterizerState",   1, 1, { { "RasterizerState" } }, 4 }, | 
					
						
							|  |  |  |     {"SetVertexShader",      1, 1, { { "VertexShader" } }, 4 }, | 
					
						
							|  |  |  |     {"SetDomainShader",      1, 1, { { "DomainShader" } }, 5 }, | 
					
						
							|  |  |  |     {"SetHullShader",        1, 1, { { "HullShader" } }, 5 }, | 
					
						
							|  |  |  |     {"SetGeometryShader",    1, 1, { { "GeometryShader" } }, 4 }, | 
					
						
							|  |  |  |     {"SetPixelShader",       1, 1, { { "PixelShader" } }, 4 }, | 
					
						
							|  |  |  |     {"SetComputeShader",     1, 1, { { "ComputeShader" } }, 4 }, | 
					
						
							|  |  |  |     {"OMSetRenderTargets",   2, 9, { {0} }, 4 }, | 
					
						
							| 
									
										
										
										
											2024-06-06 17:50:05 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct state_block_function_info *get_state_block_function_info(const char *name) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |     for (unsigned int i = 0; i < ARRAY_SIZE(function_info); ++i) | 
					
						
							| 
									
										
										
										
											2024-06-06 17:50:05 -04:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |         if (!strcmp(name, function_info[i].name)) | 
					
						
							|  |  |  |             return &function_info[i]; | 
					
						
							| 
									
										
										
										
											2024-06-06 17:50:05 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  | static void add_function_component(struct function_component **components, const char *name, | 
					
						
							|  |  |  |         bool lhs_has_index, unsigned int lhs_index) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct function_component *comp = *components; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     comp->name = name; | 
					
						
							|  |  |  |     comp->lhs_has_index = lhs_has_index; | 
					
						
							|  |  |  |     comp->lhs_index = lhs_index; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     *components = *components + 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void get_state_block_function_components(const struct state_block_function_info *info, | 
					
						
							|  |  |  |         struct function_component *components, unsigned int comp_count) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     unsigned int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-13 14:18:47 +02:00
										 |  |  |     VKD3D_ASSERT(comp_count <= info->max_args); | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (info->min_args == info->max_args) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         const struct function_component *c = info->components; | 
					
						
							|  |  |  |         for (i = 0; i < comp_count; ++i, ++c) | 
					
						
							|  |  |  |             add_function_component(&components, c->name, c->lhs_has_index, c->lhs_index); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!strcmp(info->name, "OMSetRenderTargets")) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         for (i = 0; i < comp_count - 2; ++i) | 
					
						
							|  |  |  |             add_function_component(&components, "RenderTargetView", true, i + 1); | 
					
						
							| 
									
										
										
										
											2024-08-07 18:35:51 +02:00
										 |  |  |         add_function_component(&components, "DepthStencilView", false, 0); | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |         add_function_component(&components, "RenderTargetView", true, 0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-06 17:50:05 -04:00
										 |  |  | bool hlsl_validate_state_block_entry(struct hlsl_ctx *ctx, struct hlsl_state_block_entry *entry, | 
					
						
							|  |  |  |         const struct vkd3d_shader_location *loc) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (entry->is_function_call) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         const struct state_block_function_info *info = get_state_block_function_info(entry->name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!info) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             hlsl_error(ctx, loc, VKD3D_SHADER_ERROR_HLSL_INVALID_STATE_BLOCK_ENTRY, | 
					
						
							|  |  |  |                     "Invalid state block function '%s'.", entry->name); | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (entry->args_count < info->min_args || entry->args_count > info->max_args) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (info->min_args == info->max_args) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 hlsl_error(ctx, loc, VKD3D_SHADER_ERROR_HLSL_INVALID_STATE_BLOCK_ENTRY, | 
					
						
							|  |  |  |                         "Invalid argument count for state block function '%s' (expected %u).", | 
					
						
							|  |  |  |                         entry->name, info->min_args); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 hlsl_error(ctx, loc, VKD3D_SHADER_ERROR_HLSL_INVALID_STATE_BLOCK_ENTRY, | 
					
						
							|  |  |  |                         "Invalid argument count for state block function '%s' (expected from %u to %u).", | 
					
						
							|  |  |  |                         entry->name, info->min_args, info->max_args); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | struct fx_write_context; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct fx_write_context_ops | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t (*write_string)(const char *string, struct fx_write_context *fx); | 
					
						
							|  |  |  |     void (*write_technique)(struct hlsl_ir_var *var, struct fx_write_context *fx); | 
					
						
							| 
									
										
										
										
											2023-11-15 23:31:01 +01:00
										 |  |  |     void (*write_pass)(struct hlsl_ir_var *var, struct fx_write_context *fx); | 
					
						
							| 
									
										
										
										
											2024-06-15 23:44:08 +02:00
										 |  |  |     void (*write_annotation)(struct hlsl_ir_var *var, struct fx_write_context *fx); | 
					
						
							| 
									
										
										
										
											2025-06-16 23:35:08 +02:00
										 |  |  |     void (*write_state_assignment)(const struct hlsl_ir_var *var, | 
					
						
							|  |  |  |             struct hlsl_state_block_entry *entry, struct fx_write_context *fx); | 
					
						
							| 
									
										
										
										
											2024-02-25 01:08:46 +01:00
										 |  |  |     bool are_child_effects_supported; | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | struct fx_write_context | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-01-12 14:33:05 +01:00
										 |  |  |     struct hlsl_ctx *ctx; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  |     struct vkd3d_bytecode_buffer unstructured; | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer structured; | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |     struct vkd3d_bytecode_buffer objects; | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-12 14:33:05 +01:00
										 |  |  |     struct rb_tree strings; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     struct list types; | 
					
						
							| 
									
										
										
										
											2024-01-12 14:33:05 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  |     unsigned int min_technique_version; | 
					
						
							|  |  |  |     unsigned int max_technique_version; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  |     uint32_t technique_count; | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  |     uint32_t group_count; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     uint32_t buffer_count; | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  |     uint32_t shared_buffer_count; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     uint32_t numeric_variable_count; | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  |     uint32_t shared_numeric_variable_count; | 
					
						
							| 
									
										
										
										
											2024-02-09 22:42:23 +01:00
										 |  |  |     uint32_t object_variable_count; | 
					
						
							| 
									
										
										
										
											2024-02-25 01:08:46 +01:00
										 |  |  |     uint32_t shared_object_count; | 
					
						
							| 
									
										
										
										
											2024-04-20 16:30:05 +02:00
										 |  |  |     uint32_t shader_count; | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |     uint32_t parameter_count; | 
					
						
							| 
									
										
										
										
											2024-05-06 23:29:15 +02:00
										 |  |  |     uint32_t dsv_count; | 
					
						
							| 
									
										
										
										
											2024-04-20 16:19:54 +02:00
										 |  |  |     uint32_t rtv_count; | 
					
						
							| 
									
										
										
										
											2024-04-20 16:28:34 +02:00
										 |  |  |     uint32_t texture_count; | 
					
						
							| 
									
										
										
										
											2024-04-19 11:40:14 +02:00
										 |  |  |     uint32_t uav_count; | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |     uint32_t sampler_state_count; | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  |     uint32_t depth_stencil_state_count; | 
					
						
							| 
									
										
										
										
											2024-06-10 13:18:43 +02:00
										 |  |  |     uint32_t rasterizer_state_count; | 
					
						
							| 
									
										
										
										
											2024-08-07 12:59:24 +02:00
										 |  |  |     uint32_t blend_state_count; | 
					
						
							| 
									
										
										
										
											2024-08-13 23:33:16 +02:00
										 |  |  |     uint32_t string_count; | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  |     int status; | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-25 01:08:46 +01:00
										 |  |  |     bool child_effect; | 
					
						
							| 
									
										
										
										
											2024-03-28 11:39:04 +01:00
										 |  |  |     bool include_empty_buffers; | 
					
						
							| 
									
										
										
										
											2024-02-25 01:08:46 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |     const struct fx_write_context_ops *ops; | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | static void set_status(struct fx_write_context *fx, int status) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (fx->status < 0) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     if (status < 0) | 
					
						
							|  |  |  |         fx->status = status; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | static uint32_t write_string(const char *string, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return fx->ops->write_string(string, fx); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-15 23:31:01 +01:00
										 |  |  | static void write_pass(struct hlsl_ir_var *var, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     fx->ops->write_pass(var, fx); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 23:35:08 +02:00
										 |  |  | static void write_state_assignment(const struct hlsl_ir_var *var, | 
					
						
							|  |  |  |         struct hlsl_state_block_entry *entry, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     fx->ops->write_state_assignment(var, entry, fx); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uint32_t write_state_block(struct hlsl_ir_var *var, | 
					
						
							|  |  |  |         unsigned int block_index, struct fx_write_context *fx); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-15 23:44:08 +02:00
										 |  |  | static uint32_t write_annotations(struct hlsl_scope *scope, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							|  |  |  |     struct hlsl_ir_var *v; | 
					
						
							|  |  |  |     uint32_t count = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!scope) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LIST_FOR_EACH_ENTRY(v, &scope->vars, struct hlsl_ir_var, scope_entry) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (!v->default_values) | 
					
						
							|  |  |  |             hlsl_error(ctx, &v->loc, VKD3D_SHADER_ERROR_HLSL_INVALID_SYNTAX, | 
					
						
							|  |  |  |                     "Annotation variable is missing default value."); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         fx->ops->write_annotation(v, fx); | 
					
						
							|  |  |  |         ++count; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return count; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_fx_4_annotations(struct hlsl_scope *scope, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							|  |  |  |     uint32_t count_offset, count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     count_offset = put_u32(buffer, 0); | 
					
						
							|  |  |  |     count = write_annotations(scope, fx); | 
					
						
							|  |  |  |     set_u32(buffer, count_offset, count); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-11 11:31:21 +01:00
										 |  |  | static uint32_t write_fx_4_type(const struct hlsl_type *type, struct fx_write_context *fx); | 
					
						
							| 
									
										
										
										
											2024-04-21 20:11:05 +02:00
										 |  |  | static const char * get_fx_4_type_name(const struct hlsl_type *type); | 
					
						
							| 
									
										
										
										
											2024-06-15 23:44:08 +02:00
										 |  |  | static void write_fx_4_annotation(struct hlsl_ir_var *var, struct fx_write_context *fx); | 
					
						
							| 
									
										
										
										
											2024-03-11 11:31:21 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | static uint32_t write_type(const struct hlsl_type *type, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-09-08 04:42:11 +02:00
										 |  |  |     unsigned int elements_count, modifiers; | 
					
						
							| 
									
										
										
										
											2024-04-21 20:11:05 +02:00
										 |  |  |     const struct hlsl_type *element_type; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     struct type_entry *type_entry; | 
					
						
							|  |  |  |     const char *name; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-01 10:48:29 +02:00
										 |  |  |     VKD3D_ASSERT(fx->ctx->profile->major_version >= 4); | 
					
						
							| 
									
										
										
										
											2024-03-11 11:31:21 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     if (type->class == HLSL_CLASS_ARRAY) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         elements_count = hlsl_get_multiarray_size(type); | 
					
						
							| 
									
										
										
										
											2024-04-21 20:11:05 +02:00
										 |  |  |         element_type = hlsl_get_multiarray_element_type(type); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         elements_count = 0; | 
					
						
							| 
									
										
										
										
											2024-04-21 20:11:05 +02:00
										 |  |  |         element_type = type; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-21 20:11:05 +02:00
										 |  |  |     name = get_fx_4_type_name(element_type); | 
					
						
							| 
									
										
										
										
											2024-09-08 04:42:11 +02:00
										 |  |  |     modifiers = element_type->modifiers & HLSL_MODIFIERS_MAJORITY_MASK; | 
					
						
							| 
									
										
										
										
											2024-04-21 20:11:05 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-22 18:06:03 +02:00
										 |  |  |     /* We don't try to reuse nameless types; they will get the same
 | 
					
						
							|  |  |  |      * "<unnamed>" name, but are not available for the type cache. */ | 
					
						
							|  |  |  |     if (name) | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-05-22 18:06:03 +02:00
										 |  |  |         LIST_FOR_EACH_ENTRY(type_entry, &fx->types, struct type_entry, entry) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (strcmp(type_entry->name, name)) | 
					
						
							|  |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-22 18:06:03 +02:00
										 |  |  |             if (type_entry->elements_count != elements_count) | 
					
						
							|  |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-22 18:06:03 +02:00
										 |  |  |             if (type_entry->modifiers != modifiers) | 
					
						
							|  |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2024-09-08 04:42:11 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-22 18:06:03 +02:00
										 |  |  |             return type_entry->offset; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!(type_entry = hlsl_alloc(fx->ctx, sizeof(*type_entry)))) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-11 11:31:21 +01:00
										 |  |  |     type_entry->offset = write_fx_4_type(type, fx); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     type_entry->name = name; | 
					
						
							|  |  |  |     type_entry->elements_count = elements_count; | 
					
						
							| 
									
										
										
										
											2024-09-08 04:42:11 +02:00
										 |  |  |     type_entry->modifiers = modifiers; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-22 18:06:03 +02:00
										 |  |  |     if (name) | 
					
						
							|  |  |  |         list_add_tail(&fx->types, &type_entry->entry); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return type_entry->offset; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | static void fx_write_context_init(struct hlsl_ctx *ctx, const struct fx_write_context_ops *ops, | 
					
						
							|  |  |  |         struct fx_write_context *fx) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     unsigned int version = ctx->profile->major_version; | 
					
						
							| 
									
										
										
										
											2024-04-23 21:38:06 +02:00
										 |  |  |     struct hlsl_ir_var *var; | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     memset(fx, 0, sizeof(*fx)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-12 14:33:05 +01:00
										 |  |  |     fx->ctx = ctx; | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |     fx->ops = ops; | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  |     if (version == 2) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx->min_technique_version = 9; | 
					
						
							|  |  |  |         fx->max_technique_version = 9; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (version == 4) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx->min_technique_version = 10; | 
					
						
							|  |  |  |         fx->max_technique_version = 10; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (version == 5) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx->min_technique_version = 10; | 
					
						
							|  |  |  |         fx->max_technique_version = 11; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-01-12 14:33:05 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     rb_init(&fx->strings, string_storage_compare); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     list_init(&fx->types); | 
					
						
							| 
									
										
										
										
											2024-02-25 01:08:46 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     fx->child_effect = fx->ops->are_child_effects_supported && ctx->child_effect; | 
					
						
							| 
									
										
										
										
											2024-03-28 11:39:04 +01:00
										 |  |  |     fx->include_empty_buffers = version == 4 && ctx->include_empty_buffers; | 
					
						
							| 
									
										
										
										
											2024-03-13 23:39:27 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-23 21:38:06 +02:00
										 |  |  |     LIST_FOR_EACH_ENTRY(var, &ctx->globals->vars, struct hlsl_ir_var, scope_entry) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (var->storage_modifiers & HLSL_STORAGE_UNIFORM) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             list_add_tail(&ctx->extern_vars, &var->extern_entry); | 
					
						
							|  |  |  |             var->is_uniform = 1; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-13 23:39:27 +01:00
										 |  |  |     hlsl_calculate_buffer_offsets(fx->ctx); | 
					
						
							| 
									
										
										
										
											2024-01-12 14:33:05 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int fx_write_context_cleanup(struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     struct type_entry *type, *next_type; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-12 14:33:05 +01:00
										 |  |  |     rb_destroy(&fx->strings, string_storage_destroy, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     LIST_FOR_EACH_ENTRY_SAFE(type, next_type, &fx->types, struct type_entry, entry) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         list_remove(&type->entry); | 
					
						
							|  |  |  |         vkd3d_free(type); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-23 02:11:38 +01:00
										 |  |  |     return fx->ctx->result; | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool technique_matches_version(const struct hlsl_ir_var *var, const struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const struct hlsl_type *type = var->data_type; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 17:15:48 -06:00
										 |  |  |     if (type->class != HLSL_CLASS_TECHNIQUE) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return type->e.version >= fx->min_technique_version && type->e.version <= fx->max_technique_version; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | static uint32_t write_fx_4_string(const char *string, struct fx_write_context *fx) | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-01-12 14:33:05 +01:00
										 |  |  |     struct string_entry *string_entry; | 
					
						
							|  |  |  |     struct rb_entry *entry; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  |     /* NULLs are emitted as empty strings using the same 4 bytes at the start of the section. */ | 
					
						
							| 
									
										
										
										
											2024-01-12 14:33:05 +01:00
										 |  |  |     if (!string) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ((entry = rb_get(&fx->strings, string))) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         string_entry = RB_ENTRY_VALUE(entry, struct string_entry, entry); | 
					
						
							|  |  |  |         return string_entry->offset; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!(string_entry = hlsl_alloc(fx->ctx, sizeof(*string_entry)))) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 14:21:46 +01:00
										 |  |  |     string_entry->offset = bytecode_put_bytes_unaligned(&fx->unstructured, string, strlen(string) + 1); | 
					
						
							| 
									
										
										
										
											2024-01-12 14:33:05 +01:00
										 |  |  |     string_entry->string = string; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     rb_put(&fx->strings, string, &string_entry->entry); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return string_entry->offset; | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 16:40:37 +02:00
										 |  |  | static void fx_4_decompose_state_blocks(struct hlsl_ir_var *var, struct fx_write_context *fx); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-15 23:31:01 +01:00
										 |  |  | static void write_fx_4_pass(struct hlsl_ir_var *var, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							| 
									
										
										
										
											2025-06-16 23:26:35 +02:00
										 |  |  |     uint32_t name_offset, count_offset, count; | 
					
						
							| 
									
										
										
										
											2023-11-15 23:31:01 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     name_offset = write_string(var->name, fx); | 
					
						
							|  |  |  |     put_u32(buffer, name_offset); | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |     count_offset = put_u32(buffer, 0); | 
					
						
							| 
									
										
										
										
											2023-11-15 23:31:01 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 16:40:37 +02:00
										 |  |  |     fx_4_decompose_state_blocks(var, fx); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-15 23:44:08 +02:00
										 |  |  |     write_fx_4_annotations(var->annotations, fx); | 
					
						
							| 
									
										
										
										
											2025-06-16 23:35:08 +02:00
										 |  |  |     count = write_state_block(var, 0, fx); | 
					
						
							| 
									
										
										
										
											2025-06-16 23:26:35 +02:00
										 |  |  |     set_u32(buffer, count_offset, count); | 
					
						
							| 
									
										
										
										
											2023-11-15 23:31:01 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  | static void write_fx_2_annotations(struct hlsl_ir_var *var, uint32_t count_offset, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							|  |  |  |     uint32_t count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     count = write_annotations(var->annotations, fx); | 
					
						
							|  |  |  |     set_u32(buffer, count_offset, count); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  | static const struct rhs_named_value fx_2_zenable_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "USEW", 2 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_fillmode_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "POINT", 1 }, | 
					
						
							|  |  |  |     { "WIREFRAME", 2 }, | 
					
						
							|  |  |  |     { "SOLID", 3 }, | 
					
						
							|  |  |  |     { NULL }, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_shademode_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "FLAT", 1 }, | 
					
						
							|  |  |  |     { "GOURAUD", 2 }, | 
					
						
							|  |  |  |     { "PHONG", 3 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_blendmode_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "ZERO", 1 }, | 
					
						
							|  |  |  |     { "ONE", 2 }, | 
					
						
							|  |  |  |     { "SRCCOLOR", 3 }, | 
					
						
							|  |  |  |     { "INVSRCCOLOR", 4 }, | 
					
						
							|  |  |  |     { "SRCALPHA", 5 }, | 
					
						
							|  |  |  |     { "INVSRCALPHA", 6 }, | 
					
						
							|  |  |  |     { "DESTALPHA", 7 }, | 
					
						
							|  |  |  |     { "INVDESTALPHA", 8 }, | 
					
						
							|  |  |  |     { "DESTCOLOR", 9 }, | 
					
						
							|  |  |  |     { "INVDESTCOLOR", 10 }, | 
					
						
							|  |  |  |     { "SRCALPHASAT", 11 }, | 
					
						
							|  |  |  |     { "BOTHSRCALPHA", 12 }, | 
					
						
							|  |  |  |     { "BOTHINVSRCALPHA", 13 }, | 
					
						
							|  |  |  |     { "BLENDFACTOR", 14 }, | 
					
						
							|  |  |  |     { "INVBLENDFACTOR", 15 }, | 
					
						
							|  |  |  |     { "SRCCOLOR2", 16 }, | 
					
						
							|  |  |  |     { "INVSRCCOLOR2", 17 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_cullmode_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "NONE", 1 }, | 
					
						
							|  |  |  |     { "CW", 2 }, | 
					
						
							|  |  |  |     { "CCW", 3 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_cmpfunc_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "NEVER", 1 }, | 
					
						
							|  |  |  |     { "LESS", 2 }, | 
					
						
							|  |  |  |     { "EQUAL", 3 }, | 
					
						
							|  |  |  |     { "LESSEQUAL", 4 }, | 
					
						
							|  |  |  |     { "GREATER", 5 }, | 
					
						
							|  |  |  |     { "NOTEQUAL", 6 }, | 
					
						
							|  |  |  |     { "GREATEREQUAL", 7 }, | 
					
						
							|  |  |  |     { "ALWAYS", 8 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_fogmode_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "NONE", 0 }, | 
					
						
							|  |  |  |     { "EXP", 1 }, | 
					
						
							|  |  |  |     { "EXP2", 2 }, | 
					
						
							|  |  |  |     { "LINEAR", 3 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_stencilcaps_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "KEEP", 0x1 }, | 
					
						
							|  |  |  |     { "ZERO", 0x2 }, | 
					
						
							|  |  |  |     { "REPLACE", 0x4 }, | 
					
						
							|  |  |  |     { "INCRSAT", 0x8 }, | 
					
						
							|  |  |  |     { "DECRSAT", 0x10 }, | 
					
						
							|  |  |  |     { "INVERT", 0x20 }, | 
					
						
							|  |  |  |     { "INCR", 0x40 }, | 
					
						
							|  |  |  |     { "DECR", 0x80 }, | 
					
						
							|  |  |  |     { "TWOSIDED", 0x100 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_wrap_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "COORD_0", 0x1 }, | 
					
						
							|  |  |  |     { "COORD_1", 0x2 }, | 
					
						
							|  |  |  |     { "COORD_2", 0x4 }, | 
					
						
							|  |  |  |     { "COORD_3", 0x8 }, | 
					
						
							|  |  |  |     { "U", 0x1 }, | 
					
						
							|  |  |  |     { "V", 0x2 }, | 
					
						
							|  |  |  |     { "W", 0x4 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_materialcolorsource_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "MATERIAL", 0 }, | 
					
						
							|  |  |  |     { "COORD1", 1 }, | 
					
						
							|  |  |  |     { "COORD2", 2 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_vertexblend_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "DISABLE", 0 }, | 
					
						
							|  |  |  |     { "1WEIGHTS", 1 }, | 
					
						
							|  |  |  |     { "2WEIGHTS", 2 }, | 
					
						
							|  |  |  |     { "3WEIGHTS", 3 }, | 
					
						
							|  |  |  |     { "TWEENING", 255 }, | 
					
						
							|  |  |  |     { "0WEIGHTS", 256 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_clipplane_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "CLIPPLANE0", 0x1 }, | 
					
						
							|  |  |  |     { "CLIPPLANE1", 0x2 }, | 
					
						
							|  |  |  |     { "CLIPPLANE2", 0x4 }, | 
					
						
							|  |  |  |     { "CLIPPLANE3", 0x8 }, | 
					
						
							|  |  |  |     { "CLIPPLANE4", 0x10 }, | 
					
						
							|  |  |  |     { "CLIPPLANE5", 0x20 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_patchedgestyle_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "DISCRETE", 0 }, | 
					
						
							|  |  |  |     { "CONTINUOUS", 1 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_colorwriteenable_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "RED", 0x1 }, | 
					
						
							|  |  |  |     { "GREEN", 0x2 }, | 
					
						
							|  |  |  |     { "BLUE", 0x4 }, | 
					
						
							|  |  |  |     { "ALPHA", 0x8 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_blendop_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "ADD", 1 }, | 
					
						
							|  |  |  |     { "SUBTRACT", 2 }, | 
					
						
							|  |  |  |     { "REVSUBTRACT", 3 }, | 
					
						
							|  |  |  |     { "MIN", 4 }, | 
					
						
							|  |  |  |     { "MAX", 5 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_degree_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "LINEAR", 1 }, | 
					
						
							|  |  |  |     { "QUADRATIC", 2 }, | 
					
						
							|  |  |  |     { "CUBIC", 3 }, | 
					
						
							|  |  |  |     { "QUINTIC", 4 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_textureop_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "DISABLE", 1 }, | 
					
						
							|  |  |  |     { "SELECTARG1", 2 }, | 
					
						
							|  |  |  |     { "SELECTARG2", 3 }, | 
					
						
							|  |  |  |     { "MODULATE", 4 }, | 
					
						
							|  |  |  |     { "MODULATE2X", 5 }, | 
					
						
							|  |  |  |     { "MODULATE4X", 6 }, | 
					
						
							|  |  |  |     { "ADD", 7 }, | 
					
						
							|  |  |  |     { "ADDSIGNED", 8 }, | 
					
						
							|  |  |  |     { "ADDSIGNED2X", 9 }, | 
					
						
							|  |  |  |     { "SUBTRACT", 10 }, | 
					
						
							|  |  |  |     { "ADDSMOOTH", 11 }, | 
					
						
							|  |  |  |     { "BLENDDIFFUSEALPHA", 12 }, | 
					
						
							|  |  |  |     { "BLENDTEXTUREALPHA", 13 }, | 
					
						
							|  |  |  |     { "BLENDFACTORALPHA", 14 }, | 
					
						
							|  |  |  |     { "BLENDTEXTUREALPHAPM", 15 }, | 
					
						
							|  |  |  |     { "BLENDCURRENTALPHA", 16 }, | 
					
						
							|  |  |  |     { "PREMODULATE", 17 }, | 
					
						
							|  |  |  |     { "MODULATEALPHA_ADDCOLOR", 18 }, | 
					
						
							|  |  |  |     { "MODULATECOLOR_ADDALPHA", 19 }, | 
					
						
							|  |  |  |     { "MODULATEINVALPHA_ADDCOLOR", 20 }, | 
					
						
							|  |  |  |     { "MODULATEINVCOLOR_ADDALPHA", 21 }, | 
					
						
							|  |  |  |     { "BUMPENVMAP", 22 }, | 
					
						
							|  |  |  |     { "BUMPENVMAPLUMINANCE", 23 }, | 
					
						
							|  |  |  |     { "DOTPRODUCT3", 24 }, | 
					
						
							|  |  |  |     { "MULTIPLYADD", 25 }, | 
					
						
							|  |  |  |     { "LERP", 26 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_colorarg_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "DIFFUSE", 0x0 }, | 
					
						
							|  |  |  |     { "CURRENT", 0x1 }, | 
					
						
							|  |  |  |     { "TEXTURE", 0x2 }, | 
					
						
							|  |  |  |     { "TFACTOR", 0x3 }, | 
					
						
							|  |  |  |     { "SPECULAR", 0x4 }, | 
					
						
							|  |  |  |     { "TEMP", 0x5 }, | 
					
						
							|  |  |  |     { "CONSTANT", 0x6 }, | 
					
						
							|  |  |  |     { "COMPLEMENT", 0x10 }, | 
					
						
							|  |  |  |     { "ALPHAREPLICATE", 0x20 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_texturetransform_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "DISABLE", 0 }, | 
					
						
							|  |  |  |     { "COUNT1", 1 }, | 
					
						
							|  |  |  |     { "COUNT2", 2 }, | 
					
						
							|  |  |  |     { "COUNT3", 3 }, | 
					
						
							|  |  |  |     { "COUNT4", 4 }, | 
					
						
							|  |  |  |     { "PROJECTED", 256 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_lighttype_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "POINT", 1 }, | 
					
						
							|  |  |  |     { "SPOT", 2 }, | 
					
						
							|  |  |  |     { "DIRECTIONAL", 3 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-10 17:58:18 +01:00
										 |  |  | static const struct rhs_named_value fx_2_address_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "WRAP", 1 }, | 
					
						
							|  |  |  |     { "MIRROR", 2 }, | 
					
						
							|  |  |  |     { "CLAMP", 3 }, | 
					
						
							|  |  |  |     { "BORDER", 4 }, | 
					
						
							|  |  |  |     { "MIRROR_ONCE", 5 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct rhs_named_value fx_2_filter_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "NONE", 0 }, | 
					
						
							|  |  |  |     { "POINT", 1 }, | 
					
						
							|  |  |  |     { "LINEAR", 2 }, | 
					
						
							|  |  |  |     { "ANISOTROPIC", 3 }, | 
					
						
							|  |  |  |     { "PYRAMIDALQUAD", 6 }, | 
					
						
							|  |  |  |     { "GAUSSIANQUAD", 7 }, | 
					
						
							|  |  |  |     { "CONVOLUTIONMONO", 8 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 08:36:17 +02:00
										 |  |  | struct fx_state | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     const char *name; | 
					
						
							| 
									
										
										
										
											2025-06-16 08:36:17 +02:00
										 |  |  |     enum hlsl_type_class container; | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  |     enum hlsl_type_class class; | 
					
						
							|  |  |  |     enum state_property_component_type type; | 
					
						
							|  |  |  |     unsigned int dimx; | 
					
						
							|  |  |  |     uint32_t array_size; | 
					
						
							|  |  |  |     uint32_t id; | 
					
						
							|  |  |  |     const struct rhs_named_value *values; | 
					
						
							| 
									
										
										
										
											2025-06-04 20:28:33 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 08:36:17 +02:00
										 |  |  | static const struct fx_state fx_2_pass_states[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "ZEnable",          HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 0, fx_2_zenable_values }, | 
					
						
							|  |  |  |     { "FillMode",         HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 1, fx_2_fillmode_values }, | 
					
						
							|  |  |  |     { "ShadeMode",        HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 2, fx_2_shademode_values }, | 
					
						
							|  |  |  |     { "ZWriteEnable",     HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 3 }, | 
					
						
							|  |  |  |     { "AlphaTestEnable",  HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 4 }, | 
					
						
							|  |  |  |     { "LastPixel",        HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 5 }, | 
					
						
							|  |  |  |     { "SrcBlend",         HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 6, fx_2_blendmode_values }, | 
					
						
							|  |  |  |     { "DestBlend",        HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 7, fx_2_blendmode_values }, | 
					
						
							|  |  |  |     { "CullMode",         HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 8, fx_2_cullmode_values }, | 
					
						
							|  |  |  |     { "ZFunc",            HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 9, fx_2_cmpfunc_values }, | 
					
						
							|  |  |  |     { "AlphaRef",         HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 10 }, | 
					
						
							|  |  |  |     { "AlphaFunc",        HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 11, fx_2_cmpfunc_values }, | 
					
						
							|  |  |  |     { "DitherEnable",     HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 12 }, | 
					
						
							|  |  |  |     { "AlphaBlendEnable", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 13 }, | 
					
						
							|  |  |  |     { "FogEnable",        HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 14 }, | 
					
						
							|  |  |  |     { "SpecularEnable",   HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 15 }, | 
					
						
							|  |  |  |     { "FogColor",         HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 16 }, | 
					
						
							|  |  |  |     { "FogTableMode",     HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 17, fx_2_fogmode_values }, | 
					
						
							|  |  |  |     { "FogStart",         HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 18 }, | 
					
						
							|  |  |  |     { "FogEnd",           HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 19 }, | 
					
						
							|  |  |  |     { "FogDensity",       HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 20 }, | 
					
						
							|  |  |  |     { "RangeFogEnable",   HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 21 }, | 
					
						
							|  |  |  |     { "StencilEnable",    HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 22 }, | 
					
						
							|  |  |  |     { "StencilFail",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 23, fx_2_stencilcaps_values }, | 
					
						
							|  |  |  |     { "StencilZFail",     HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 24, fx_2_stencilcaps_values }, | 
					
						
							|  |  |  |     { "StencilPass",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 25, fx_2_stencilcaps_values }, | 
					
						
							|  |  |  |     { "StencilFunc",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 26, fx_2_cmpfunc_values }, | 
					
						
							|  |  |  |     { "StencilRef",       HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 27 }, | 
					
						
							|  |  |  |     { "StencilMask",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 28 }, | 
					
						
							|  |  |  |     { "StencilWriteMask", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 29 }, | 
					
						
							|  |  |  |     { "TextureFactor",    HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 30 }, | 
					
						
							|  |  |  |     { "Wrap0",            HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 31, fx_2_wrap_values }, | 
					
						
							|  |  |  |     { "Wrap1",            HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 32, fx_2_wrap_values }, | 
					
						
							|  |  |  |     { "Wrap2",            HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 33, fx_2_wrap_values }, | 
					
						
							|  |  |  |     { "Wrap3",            HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 34, fx_2_wrap_values }, | 
					
						
							|  |  |  |     { "Wrap4",            HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 35, fx_2_wrap_values }, | 
					
						
							|  |  |  |     { "Wrap5",            HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 36, fx_2_wrap_values }, | 
					
						
							|  |  |  |     { "Wrap6",            HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 37, fx_2_wrap_values }, | 
					
						
							|  |  |  |     { "Wrap7",            HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 38, fx_2_wrap_values }, | 
					
						
							|  |  |  |     { "Wrap8",            HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 39, fx_2_wrap_values }, | 
					
						
							|  |  |  |     { "Wrap9",            HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 40, fx_2_wrap_values }, | 
					
						
							|  |  |  |     { "Wrap10",           HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 41, fx_2_wrap_values }, | 
					
						
							|  |  |  |     { "Wrap11",           HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 42, fx_2_wrap_values }, | 
					
						
							|  |  |  |     { "Wrap12",           HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 43, fx_2_wrap_values }, | 
					
						
							|  |  |  |     { "Wrap13",           HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 44, fx_2_wrap_values }, | 
					
						
							|  |  |  |     { "Wrap14",           HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 45, fx_2_wrap_values }, | 
					
						
							|  |  |  |     { "Wrap15",           HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 46, fx_2_wrap_values }, | 
					
						
							|  |  |  |     { "Clipping",         HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 47 }, | 
					
						
							|  |  |  |     { "Lighting",         HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 48 }, | 
					
						
							|  |  |  |     { "Ambient",          HLSL_CLASS_PASS, HLSL_CLASS_VECTOR, FX_FLOAT, 4, 1, 49 }, | 
					
						
							|  |  |  |     { "FogVertexMode",    HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 50, fx_2_fogmode_values }, | 
					
						
							|  |  |  |     { "ColorVertex",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 51 }, | 
					
						
							|  |  |  |     { "LocalViewer",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 52 }, | 
					
						
							|  |  |  |     { "NormalizeNormals", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 53 }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "DiffuseMaterialSource",  HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 1, 54, fx_2_materialcolorsource_values }, | 
					
						
							|  |  |  |     { "SpecularMaterialSource", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 1, 55, fx_2_materialcolorsource_values }, | 
					
						
							|  |  |  |     { "AmbientMaterialSource",  HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 1, 56, fx_2_materialcolorsource_values }, | 
					
						
							|  |  |  |     { "EmissiveMaterialSource", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 1, 57, fx_2_materialcolorsource_values }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "VertexBlend",       HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 58, fx_2_vertexblend_values }, | 
					
						
							|  |  |  |     { "ClipPlaneEnable",   HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 59, fx_2_clipplane_values }, | 
					
						
							|  |  |  |     { "PointSize",         HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 60 }, | 
					
						
							|  |  |  |     { "PointSize_Min",     HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 61 }, | 
					
						
							|  |  |  |     { "PointSize_Max",     HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 62 }, | 
					
						
							|  |  |  |     { "PointSpriteEnable", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 63 }, | 
					
						
							|  |  |  |     { "PointScaleEnable",  HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 64 }, | 
					
						
							|  |  |  |     { "PointScale_A",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 65 }, | 
					
						
							|  |  |  |     { "PointScale_B",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 66 }, | 
					
						
							|  |  |  |     { "PointScale_C",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 67 }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "MultiSampleAntialias",     HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 68 }, | 
					
						
							|  |  |  |     { "MultiSampleMask",          HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 69 }, | 
					
						
							|  |  |  |     { "PatchEdgeStyle",           HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 70, fx_2_patchedgestyle_values }, | 
					
						
							|  |  |  |     { "DebugMonitorToken",        HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 71 }, | 
					
						
							|  |  |  |     { "IndexedVertexBlendEnable", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 72 }, | 
					
						
							|  |  |  |     { "ColorWriteEnable",         HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 73, fx_2_colorwriteenable_values }, | 
					
						
							|  |  |  |     { "TweenFactor",              HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 74 }, | 
					
						
							|  |  |  |     { "BlendOp",                  HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 75, fx_2_blendop_values }, | 
					
						
							|  |  |  |     { "PositionDegree",           HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 76, fx_2_degree_values }, | 
					
						
							|  |  |  |     { "NormalDegree",             HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 77, fx_2_degree_values }, | 
					
						
							|  |  |  |     { "ScissorTestEnable",        HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 78 }, | 
					
						
							|  |  |  |     { "SlopeScaleDepthBias",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 79 }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "AntialiasedLineEnable",     HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 80 }, | 
					
						
							|  |  |  |     { "MinTessellationLevel",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 81 }, | 
					
						
							|  |  |  |     { "MaxTessellationLevel",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 82 }, | 
					
						
							|  |  |  |     { "AdaptiveTess_X",            HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 83 }, | 
					
						
							|  |  |  |     { "AdaptiveTess_Y",            HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 84 }, | 
					
						
							|  |  |  |     { "AdaptiveTess_Z",            HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 85 }, | 
					
						
							|  |  |  |     { "AdaptiveTess_W",            HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 86 }, | 
					
						
							|  |  |  |     { "EnableAdaptiveTessellation",HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 87 }, | 
					
						
							|  |  |  |     { "TwoSidedStencilMode",       HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 88 }, | 
					
						
							|  |  |  |     { "StencilFail",               HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 89, fx_2_stencilcaps_values }, | 
					
						
							|  |  |  |     { "StencilZFail",              HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 90, fx_2_stencilcaps_values }, | 
					
						
							|  |  |  |     { "StencilPass",               HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 91, fx_2_stencilcaps_values }, | 
					
						
							|  |  |  |     { "StencilFunc",               HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 92, fx_2_cmpfunc_values }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "ColorWriteEnable1",        HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 93, fx_2_colorwriteenable_values }, | 
					
						
							|  |  |  |     { "ColorWriteEnable2",        HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 94, fx_2_colorwriteenable_values }, | 
					
						
							|  |  |  |     { "ColorWriteEnable3",        HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 95, fx_2_colorwriteenable_values }, | 
					
						
							|  |  |  |     { "BlendFactor",              HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 96 }, | 
					
						
							|  |  |  |     { "SRGBWriteEnable",          HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 97 }, | 
					
						
							|  |  |  |     { "DepthBias",                HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 98 }, | 
					
						
							|  |  |  |     { "SeparateAlphaBlendEnable", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 99 }, | 
					
						
							|  |  |  |     { "SrcBlendAlpha",            HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 100, fx_2_blendmode_values }, | 
					
						
							|  |  |  |     { "DestBlendAlpha",           HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 101, fx_2_blendmode_values }, | 
					
						
							|  |  |  |     { "BlendOpAlpha",             HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 102, fx_2_blendmode_values }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "ColorOp",               HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 8, 103, fx_2_textureop_values }, | 
					
						
							|  |  |  |     { "ColorArg0",             HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 8, 104, fx_2_colorarg_values }, | 
					
						
							|  |  |  |     { "ColorArg1",             HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 8, 105, fx_2_colorarg_values }, | 
					
						
							|  |  |  |     { "ColorArg2",             HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 8, 106, fx_2_colorarg_values }, | 
					
						
							|  |  |  |     { "AlphaOp",               HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 8, 107, fx_2_textureop_values }, | 
					
						
							|  |  |  |     { "AlphaArg0",             HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 8, 108, fx_2_colorarg_values }, | 
					
						
							|  |  |  |     { "AlphaArg1",             HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 8, 109, fx_2_colorarg_values }, | 
					
						
							|  |  |  |     { "AlphaArg2",             HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 8, 110, fx_2_colorarg_values }, | 
					
						
							|  |  |  |     { "ResultArg",             HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 8, 111, fx_2_colorarg_values }, | 
					
						
							|  |  |  |     { "BumpEnvMat00",          HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 8, 112 }, | 
					
						
							|  |  |  |     { "BumpEnvMat01",          HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 8, 113 }, | 
					
						
							|  |  |  |     { "BumpEnvMat10",          HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 8, 114 }, | 
					
						
							|  |  |  |     { "BumpEnvMat11",          HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 8, 115 }, | 
					
						
							|  |  |  |     { "TexCoordIndex",         HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 8, 116 }, | 
					
						
							|  |  |  |     { "BumpEnvLScale",         HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 8, 117 }, | 
					
						
							|  |  |  |     { "BumpEnvLOffset",        HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 8, 118 }, | 
					
						
							|  |  |  |     { "TextureTransformFlags", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 8, 119, fx_2_texturetransform_values }, | 
					
						
							|  |  |  |     { "Constant",              HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 8, 120 }, | 
					
						
							|  |  |  |     { "PatchSegments",         HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 1, 121 }, | 
					
						
							|  |  |  |     { "FVF",                   HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT, 1, 1, 122 }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "ProjectionTransform", HLSL_CLASS_PASS, HLSL_CLASS_MATRIX, FX_FLOAT, 4, 1, 123 }, | 
					
						
							|  |  |  |     { "ViewTransform",       HLSL_CLASS_PASS, HLSL_CLASS_MATRIX, FX_FLOAT, 4, 1, 124 }, | 
					
						
							|  |  |  |     { "WorldTransform",      HLSL_CLASS_PASS, HLSL_CLASS_MATRIX, FX_FLOAT, 4, 256, 125 }, | 
					
						
							|  |  |  |     { "TextureTransform",    HLSL_CLASS_PASS, HLSL_CLASS_MATRIX, FX_FLOAT, 4, 8, 126 }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-22 11:21:27 +02:00
										 |  |  |     { "MaterialDiffuse",   HLSL_CLASS_PASS, HLSL_CLASS_VECTOR, FX_FLOAT, 4, 1, 127 }, | 
					
						
							|  |  |  |     { "MaterialAmbient",   HLSL_CLASS_PASS, HLSL_CLASS_VECTOR, FX_FLOAT, 4, 1, 128 }, | 
					
						
							| 
									
										
										
										
											2025-06-16 08:36:17 +02:00
										 |  |  |     { "MaterialSpecular",  HLSL_CLASS_PASS, HLSL_CLASS_VECTOR, FX_FLOAT, 4, 1, 129 }, | 
					
						
							|  |  |  |     { "MaterialEmissive",  HLSL_CLASS_PASS, HLSL_CLASS_VECTOR, FX_FLOAT, 4, 1, 130 }, | 
					
						
							|  |  |  |     { "MaterialPower",     HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 131 }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "LightType",         HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, ~0u, 132, fx_2_lighttype_values }, | 
					
						
							|  |  |  |     { "LightDiffuse",      HLSL_CLASS_PASS, HLSL_CLASS_VECTOR, FX_FLOAT, 4, ~0u, 133 }, | 
					
						
							|  |  |  |     { "LightSpecular",     HLSL_CLASS_PASS, HLSL_CLASS_VECTOR, FX_FLOAT, 4, ~0u, 134 }, | 
					
						
							|  |  |  |     { "LightAmbient",      HLSL_CLASS_PASS, HLSL_CLASS_VECTOR, FX_FLOAT, 4, ~0u, 135 }, | 
					
						
							|  |  |  |     { "LightPosition",     HLSL_CLASS_PASS, HLSL_CLASS_VECTOR, FX_FLOAT, 3, ~0u, 136 }, | 
					
						
							|  |  |  |     { "LightDirection",    HLSL_CLASS_PASS, HLSL_CLASS_VECTOR, FX_FLOAT, 3, ~0u, 137 }, | 
					
						
							|  |  |  |     { "LightRange",        HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 138 }, | 
					
						
							|  |  |  |     { "LightFalloff",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 139 }, | 
					
						
							|  |  |  |     { "LightAttenuation0", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 140 }, | 
					
						
							|  |  |  |     { "LightAttenuation1", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 141 }, | 
					
						
							|  |  |  |     { "LightAttenuation2", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 142 }, | 
					
						
							|  |  |  |     { "LightTheta",        HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 143 }, | 
					
						
							|  |  |  |     { "LightPhi",          HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 144 }, | 
					
						
							|  |  |  |     { "LightEnable",       HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 145 }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "VertexShader", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_VERTEXSHADER, 1, 1, 146 }, | 
					
						
							|  |  |  |     { "PixelShader",  HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_PIXELSHADER,  1, 1, 147 }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "VertexShaderConstantF", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 148 }, | 
					
						
							|  |  |  |     { "VertexShaderConstantB", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_BOOL,  1, ~0u, 149 }, | 
					
						
							|  |  |  |     { "VertexShaderConstantI", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, ~0u, 150 }, | 
					
						
							|  |  |  |     { "VertexShaderConstant",  HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 151 }, | 
					
						
							|  |  |  |     { "VertexShaderConstant1", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 152 }, | 
					
						
							|  |  |  |     { "VertexShaderConstant2", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 153 }, | 
					
						
							|  |  |  |     { "VertexShaderConstant3", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 154 }, | 
					
						
							|  |  |  |     { "VertexShaderConstant4", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 155 }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "PixelShaderConstantF", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 156 }, | 
					
						
							|  |  |  |     { "PixelShaderConstantB", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_BOOL,  1, ~0u, 157 }, | 
					
						
							|  |  |  |     { "PixelShaderConstantI", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, ~0u, 158 }, | 
					
						
							|  |  |  |     { "PixelShaderConstant",  HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 159 }, | 
					
						
							|  |  |  |     { "PixelShaderConstant1", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 160 }, | 
					
						
							|  |  |  |     { "PixelShaderConstant2", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 161 }, | 
					
						
							|  |  |  |     { "PixelShaderConstant3", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 162 }, | 
					
						
							|  |  |  |     { "PixelShaderConstant4", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_FLOAT, 1, ~0u, 163 }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "Texture",       HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_TEXTURE, 1, 261, 164 }, | 
					
						
							|  |  |  |     { "AddressU",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,    1, 261, 165, fx_2_address_values }, | 
					
						
							|  |  |  |     { "AddressV",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,    1, 261, 166, fx_2_address_values }, | 
					
						
							|  |  |  |     { "AddressW",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,    1, 261, 167, fx_2_address_values }, | 
					
						
							|  |  |  |     { "BorderColor",   HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,    1, 261, 168 }, | 
					
						
							|  |  |  |     { "MagFilter",     HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,    1, 261, 169, fx_2_filter_values }, | 
					
						
							|  |  |  |     { "MinFilter",     HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,    1, 261, 170, fx_2_filter_values }, | 
					
						
							|  |  |  |     { "MipFilter",     HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,    1, 261, 171, fx_2_filter_values }, | 
					
						
							|  |  |  |     { "MipMapLodBias", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,    1, 261, 172 }, | 
					
						
							|  |  |  |     { "MaxMipLevel",   HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,    1, 261, 173 }, | 
					
						
							|  |  |  |     { "MaxAnisotropy", HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,    1, 261, 174 }, | 
					
						
							|  |  |  |     { "SRGBTexture",   HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,    1, 261, 175 }, | 
					
						
							|  |  |  |     { "ElementIndex",  HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,    1, 261, 176 }, | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 08:36:17 +02:00
										 |  |  | static const struct fx_state fx_2_sampler_states[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "Texture",       HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR, FX_TEXTURE, 1, 1, 164 }, | 
					
						
							|  |  |  |     { "AddressU",      HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR, FX_UINT,    1, 1, 165, fx_2_address_values }, | 
					
						
							|  |  |  |     { "AddressV",      HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR, FX_UINT,    1, 1, 166, fx_2_address_values }, | 
					
						
							|  |  |  |     { "AddressW",      HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR, FX_UINT,    1, 1, 167, fx_2_address_values }, | 
					
						
							|  |  |  |     { "BorderColor",   HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR, FX_UINT,    1, 1, 168 }, | 
					
						
							|  |  |  |     { "MagFilter",     HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR, FX_UINT,    1, 1, 169, fx_2_filter_values }, | 
					
						
							|  |  |  |     { "MinFilter",     HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR, FX_UINT,    1, 1, 170, fx_2_filter_values }, | 
					
						
							|  |  |  |     { "MipFilter",     HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR, FX_UINT,    1, 1, 171, fx_2_filter_values }, | 
					
						
							|  |  |  |     { "MipMapLodBias", HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR, FX_UINT,    1, 1, 172 }, | 
					
						
							|  |  |  |     { "MaxMipLevel",   HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR, FX_UINT,    1, 1, 173 }, | 
					
						
							|  |  |  |     { "MaxAnisotropy", HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR, FX_UINT,    1, 1, 174 }, | 
					
						
							|  |  |  |     { "SRGBTexture",   HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR, FX_UINT,    1, 1, 175 }, | 
					
						
							|  |  |  |     { "ElementIndex",  HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR, FX_UINT,    1, 1, 176 }, | 
					
						
							| 
									
										
										
										
											2025-06-04 20:28:33 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-15 23:31:01 +01:00
										 |  |  | static void write_fx_2_pass(struct hlsl_ir_var *var, struct fx_write_context *fx) | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  |     uint32_t name_offset, annotation_count_offset; | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |     name_offset = write_string(var->name, fx); | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  |     put_u32(buffer, name_offset); | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  |     annotation_count_offset = put_u32(buffer, 0); | 
					
						
							| 
									
										
										
										
											2023-11-11 01:14:03 +01:00
										 |  |  |     put_u32(buffer, 0); /* Assignment count. */ | 
					
						
							| 
									
										
										
										
											2023-11-15 23:31:01 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  |     write_fx_2_annotations(var, annotation_count_offset, fx); | 
					
						
							| 
									
										
										
										
											2023-11-15 23:31:01 +01:00
										 |  |  |     /* TODO: assignments */ | 
					
						
							| 
									
										
										
										
											2024-07-11 14:02:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |     if (var->state_block_count && var->state_blocks[0]->count) | 
					
						
							|  |  |  |         hlsl_fixme(fx->ctx, &var->loc, "Write pass assignments."); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-11 14:02:31 +02:00
										 |  |  |     /* For some reason every pass adds to the total shader object count. */ | 
					
						
							|  |  |  |     fx->shader_count++; | 
					
						
							| 
									
										
										
										
											2023-11-11 01:14:03 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  | enum fx_4_type_constants | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     /* Numeric types encoding */ | 
					
						
							|  |  |  |     FX_4_NUMERIC_TYPE_FLOAT = 1, | 
					
						
							|  |  |  |     FX_4_NUMERIC_TYPE_INT   = 2, | 
					
						
							|  |  |  |     FX_4_NUMERIC_TYPE_UINT  = 3, | 
					
						
							|  |  |  |     FX_4_NUMERIC_TYPE_BOOL  = 4, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     FX_4_NUMERIC_CLASS_SCALAR = 1, | 
					
						
							|  |  |  |     FX_4_NUMERIC_CLASS_VECTOR = 2, | 
					
						
							|  |  |  |     FX_4_NUMERIC_CLASS_MATRIX = 3, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     FX_4_NUMERIC_BASE_TYPE_SHIFT = 3, | 
					
						
							|  |  |  |     FX_4_NUMERIC_ROWS_SHIFT = 8, | 
					
						
							|  |  |  |     FX_4_NUMERIC_COLUMNS_SHIFT = 11, | 
					
						
							|  |  |  |     FX_4_NUMERIC_COLUMN_MAJOR_MASK = 0x4000, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-30 10:20:26 +02:00
										 |  |  |     /* Variable flags */ | 
					
						
							|  |  |  |     FX_4_HAS_EXPLICIT_BIND_POINT = 0x4, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  |     /* Object types */ | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |     FX_4_OBJECT_TYPE_STRING = 0x1, | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |     FX_4_OBJECT_TYPE_BLEND_STATE = 0x2, | 
					
						
							|  |  |  |     FX_4_OBJECT_TYPE_DEPTH_STENCIL_STATE = 0x3, | 
					
						
							|  |  |  |     FX_4_OBJECT_TYPE_RASTERIZER_STATE = 0x4, | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |     FX_4_OBJECT_TYPE_PIXEL_SHADER = 0x5, | 
					
						
							|  |  |  |     FX_4_OBJECT_TYPE_VERTEX_SHADER = 0x6, | 
					
						
							|  |  |  |     FX_4_OBJECT_TYPE_GEOMETRY_SHADER = 0x7, | 
					
						
							|  |  |  |     FX_4_OBJECT_TYPE_GEOMETRY_SHADER_SO = 0x8, | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     FX_4_OBJECT_TYPE_TEXTURE = 0x9, | 
					
						
							|  |  |  |     FX_4_OBJECT_TYPE_TEXTURE_1D = 0xa, | 
					
						
							|  |  |  |     FX_4_OBJECT_TYPE_TEXTURE_1DARRAY = 0xb, | 
					
						
							|  |  |  |     FX_4_OBJECT_TYPE_TEXTURE_2D = 0xc, | 
					
						
							|  |  |  |     FX_4_OBJECT_TYPE_TEXTURE_2DARRAY = 0xd, | 
					
						
							|  |  |  |     FX_4_OBJECT_TYPE_TEXTURE_2DMS = 0xe, | 
					
						
							|  |  |  |     FX_4_OBJECT_TYPE_TEXTURE_2DMSARRAY = 0xf, | 
					
						
							|  |  |  |     FX_4_OBJECT_TYPE_TEXTURE_3D = 0x10, | 
					
						
							|  |  |  |     FX_4_OBJECT_TYPE_TEXTURE_CUBE = 0x11, | 
					
						
							| 
									
										
										
										
											2024-11-05 13:43:28 +01:00
										 |  |  |     FX_4_OBJECT_TYPE_RTV = 0x13, | 
					
						
							|  |  |  |     FX_4_OBJECT_TYPE_DSV = 0x14, | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |     FX_4_OBJECT_TYPE_SAMPLER_STATE = 0x15, | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  |     FX_4_OBJECT_TYPE_TEXTURE_CUBEARRAY = 0x17, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |     FX_5_OBJECT_TYPE_GEOMETRY_SHADER = 0x1b, | 
					
						
							|  |  |  |     FX_5_OBJECT_TYPE_COMPUTE_SHADER = 0x1c, | 
					
						
							|  |  |  |     FX_5_OBJECT_TYPE_HULL_SHADER = 0x1d, | 
					
						
							|  |  |  |     FX_5_OBJECT_TYPE_DOMAIN_SHADER = 0x1e, | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  |     FX_5_OBJECT_TYPE_UAV_1D = 0x1f, | 
					
						
							|  |  |  |     FX_5_OBJECT_TYPE_UAV_1DARRAY = 0x20, | 
					
						
							|  |  |  |     FX_5_OBJECT_TYPE_UAV_2D = 0x21, | 
					
						
							|  |  |  |     FX_5_OBJECT_TYPE_UAV_2DARRAY = 0x22, | 
					
						
							|  |  |  |     FX_5_OBJECT_TYPE_UAV_3D = 0x23, | 
					
						
							|  |  |  |     FX_5_OBJECT_TYPE_UAV_BUFFER = 0x24, | 
					
						
							| 
									
										
										
										
											2024-11-05 13:43:28 +01:00
										 |  |  |     FX_5_OBJECT_TYPE_SRV_RAW_BUFFER = 0x25, | 
					
						
							|  |  |  |     FX_5_OBJECT_TYPE_UAV_RAW_BUFFER = 0x26, | 
					
						
							|  |  |  |     FX_5_OBJECT_TYPE_SRV_STRUCTURED_BUFFER = 0x27, | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  |     FX_5_OBJECT_TYPE_UAV_STRUCTURED_BUFFER = 0x28, | 
					
						
							| 
									
										
										
										
											2024-11-05 13:43:28 +01:00
										 |  |  |     FX_5_OBJECT_TYPE_SRV_APPEND_STRUCTURED_BUFFER = 0x2b, | 
					
						
							|  |  |  |     FX_5_OBJECT_TYPE_SRV_CONSUME_STRUCTURED_BUFFER = 0x2c, | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  |     /* Types */ | 
					
						
							|  |  |  |     FX_4_TYPE_CLASS_NUMERIC = 1, | 
					
						
							|  |  |  |     FX_4_TYPE_CLASS_OBJECT = 2, | 
					
						
							|  |  |  |     FX_4_TYPE_CLASS_STRUCT = 3, | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* Assignment types */ | 
					
						
							|  |  |  |     FX_4_ASSIGNMENT_CONSTANT = 0x1, | 
					
						
							|  |  |  |     FX_4_ASSIGNMENT_VARIABLE = 0x2, | 
					
						
							|  |  |  |     FX_4_ASSIGNMENT_ARRAY_CONSTANT_INDEX = 0x3, | 
					
						
							|  |  |  |     FX_4_ASSIGNMENT_ARRAY_VARIABLE_INDEX = 0x4, | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |     FX_4_ASSIGNMENT_INDEX_EXPRESSION = 0x5, | 
					
						
							|  |  |  |     FX_4_ASSIGNMENT_VALUE_EXPRESSION = 0x6, | 
					
						
							| 
									
										
										
										
											2024-11-22 02:17:25 +01:00
										 |  |  |     FX_4_ASSIGNMENT_INLINE_SHADER = 0x7, | 
					
						
							|  |  |  |     FX_5_ASSIGNMENT_INLINE_SHADER = 0x8, | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const uint32_t fx_4_numeric_base_types[] = | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  |     [HLSL_TYPE_HALF ] = FX_4_NUMERIC_TYPE_FLOAT, | 
					
						
							|  |  |  |     [HLSL_TYPE_FLOAT] = FX_4_NUMERIC_TYPE_FLOAT, | 
					
						
							|  |  |  |     [HLSL_TYPE_INT  ] = FX_4_NUMERIC_TYPE_INT, | 
					
						
							|  |  |  |     [HLSL_TYPE_UINT ] = FX_4_NUMERIC_TYPE_UINT, | 
					
						
							|  |  |  |     [HLSL_TYPE_BOOL ] = FX_4_NUMERIC_TYPE_BOOL, | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | static uint32_t get_fx_4_numeric_type_description(const struct hlsl_type *type, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     static const uint32_t numeric_type_class[] = | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  |         [HLSL_CLASS_SCALAR] = FX_4_NUMERIC_CLASS_SCALAR, | 
					
						
							|  |  |  |         [HLSL_CLASS_VECTOR] = FX_4_NUMERIC_CLASS_VECTOR, | 
					
						
							|  |  |  |         [HLSL_CLASS_MATRIX] = FX_4_NUMERIC_CLASS_MATRIX, | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     }; | 
					
						
							| 
									
										
										
										
											2024-02-23 14:55:34 +01:00
										 |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     uint32_t value = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (type->class) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         case HLSL_CLASS_SCALAR: | 
					
						
							|  |  |  |         case HLSL_CLASS_VECTOR: | 
					
						
							|  |  |  |         case HLSL_CLASS_MATRIX: | 
					
						
							|  |  |  |             value |= numeric_type_class[type->class]; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         default: | 
					
						
							| 
									
										
										
										
											2024-03-14 13:31:00 +01:00
										 |  |  |             hlsl_fixme(ctx, &ctx->location, "Not implemented for type class %u.", type->class); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |             return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 17:30:51 -06:00
										 |  |  |     switch (type->e.numeric.type) | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         case HLSL_TYPE_FLOAT: | 
					
						
							| 
									
										
										
										
											2024-09-26 14:52:11 +02:00
										 |  |  |         case HLSL_TYPE_HALF: | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |         case HLSL_TYPE_INT: | 
					
						
							|  |  |  |         case HLSL_TYPE_UINT: | 
					
						
							|  |  |  |         case HLSL_TYPE_BOOL: | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  |             value |= (fx_4_numeric_base_types[type->e.numeric.type] << FX_4_NUMERIC_BASE_TYPE_SHIFT); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  |         default: | 
					
						
							| 
									
										
										
										
											2024-02-27 17:30:51 -06:00
										 |  |  |             hlsl_fixme(ctx, &ctx->location, "Not implemented for base type %u.", type->e.numeric.type); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |             return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-30 11:51:05 -05:00
										 |  |  |     value |= (type->e.numeric.dimy & 0x7) << FX_4_NUMERIC_ROWS_SHIFT; | 
					
						
							|  |  |  |     value |= (type->e.numeric.dimx & 0x7) << FX_4_NUMERIC_COLUMNS_SHIFT; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     if (type->modifiers & HLSL_MODIFIER_COLUMN_MAJOR) | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  |         value |= FX_4_NUMERIC_COLUMN_MAJOR_MASK; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-24 00:39:37 +01:00
										 |  |  | static const char * get_fx_4_type_name(const struct hlsl_type *type) | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-02-11 00:02:16 +01:00
										 |  |  |     static const char * const texture_type_names[] = | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         [HLSL_SAMPLER_DIM_GENERIC]   = "texture", | 
					
						
							|  |  |  |         [HLSL_SAMPLER_DIM_1D]        = "Texture1D", | 
					
						
							|  |  |  |         [HLSL_SAMPLER_DIM_1DARRAY]   = "Texture1DArray", | 
					
						
							|  |  |  |         [HLSL_SAMPLER_DIM_2D]        = "Texture2D", | 
					
						
							|  |  |  |         [HLSL_SAMPLER_DIM_2DARRAY]   = "Texture2DArray", | 
					
						
							|  |  |  |         [HLSL_SAMPLER_DIM_2DMS]      = "Texture2DMS", | 
					
						
							|  |  |  |         [HLSL_SAMPLER_DIM_2DMSARRAY] = "Texture2DMSArray", | 
					
						
							|  |  |  |         [HLSL_SAMPLER_DIM_3D]        = "Texture3D", | 
					
						
							|  |  |  |         [HLSL_SAMPLER_DIM_CUBE]      = "TextureCube", | 
					
						
							|  |  |  |         [HLSL_SAMPLER_DIM_CUBEARRAY] = "TextureCubeArray", | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2024-02-11 11:04:14 +01:00
										 |  |  |     static const char * const uav_type_names[] = | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         [HLSL_SAMPLER_DIM_1D]                = "RWTexture1D", | 
					
						
							|  |  |  |         [HLSL_SAMPLER_DIM_1DARRAY]           = "RWTexture1DArray", | 
					
						
							|  |  |  |         [HLSL_SAMPLER_DIM_2D]                = "RWTexture2D", | 
					
						
							|  |  |  |         [HLSL_SAMPLER_DIM_2DARRAY]           = "RWTexture2DArray", | 
					
						
							|  |  |  |         [HLSL_SAMPLER_DIM_3D]                = "RWTexture3D", | 
					
						
							|  |  |  |         [HLSL_SAMPLER_DIM_BUFFER]            = "RWBuffer", | 
					
						
							|  |  |  |         [HLSL_SAMPLER_DIM_STRUCTURED_BUFFER] = "RWStructuredBuffer", | 
					
						
							| 
									
										
										
										
											2024-11-05 13:43:28 +01:00
										 |  |  |         [HLSL_SAMPLER_DIM_RAW_BUFFER]        = "RWByteAddressBuffer", | 
					
						
							| 
									
										
										
										
											2024-02-11 11:04:14 +01:00
										 |  |  |     }; | 
					
						
							| 
									
										
										
										
											2024-02-24 00:39:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 13:31:20 -06:00
										 |  |  |     switch (type->class) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |         case HLSL_CLASS_SAMPLER: | 
					
						
							|  |  |  |             return "SamplerState"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 13:31:20 -06:00
										 |  |  |         case HLSL_CLASS_TEXTURE: | 
					
						
							|  |  |  |             return texture_type_names[type->sampler_dim]; | 
					
						
							| 
									
										
										
										
											2024-02-24 00:39:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 13:31:20 -06:00
										 |  |  |         case HLSL_CLASS_UAV: | 
					
						
							|  |  |  |             return uav_type_names[type->sampler_dim]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  |         case HLSL_CLASS_DEPTH_STENCIL_STATE: | 
					
						
							|  |  |  |             return "DepthStencilState"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 13:31:20 -06:00
										 |  |  |         case HLSL_CLASS_DEPTH_STENCIL_VIEW: | 
					
						
							|  |  |  |             return "DepthStencilView"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-27 13:34:52 -06:00
										 |  |  |         case HLSL_CLASS_RENDER_TARGET_VIEW: | 
					
						
							|  |  |  |             return "RenderTargetView"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 17:33:26 -06:00
										 |  |  |         case HLSL_CLASS_VERTEX_SHADER: | 
					
						
							|  |  |  |             return "VertexShader"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-21 15:59:23 -04:00
										 |  |  |         case HLSL_CLASS_GEOMETRY_SHADER: | 
					
						
							|  |  |  |             return "GeometryShader"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 19:41:15 -06:00
										 |  |  |         case HLSL_CLASS_PIXEL_SHADER: | 
					
						
							|  |  |  |             return "PixelShader"; | 
					
						
							| 
									
										
										
										
											2024-02-24 00:39:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-13 23:33:16 +02:00
										 |  |  |         case HLSL_CLASS_STRING: | 
					
						
							|  |  |  |             return "String"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-26 14:52:11 +02:00
										 |  |  |         case HLSL_CLASS_SCALAR: | 
					
						
							|  |  |  |         case HLSL_CLASS_VECTOR: | 
					
						
							|  |  |  |         case HLSL_CLASS_MATRIX: | 
					
						
							|  |  |  |             if (type->e.numeric.type == HLSL_TYPE_HALF) | 
					
						
							|  |  |  |                 return "float"; | 
					
						
							|  |  |  |             /* fall-through */ | 
					
						
							| 
									
										
										
										
											2024-02-24 00:39:37 +01:00
										 |  |  |         default: | 
					
						
							|  |  |  |             return type->name; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-31 01:24:51 +02:00
										 |  |  | static bool is_numeric_fx_4_type(const struct hlsl_type *type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     type = hlsl_get_multiarray_element_type(type); | 
					
						
							|  |  |  |     return type->class == HLSL_CLASS_STRUCT || hlsl_is_numeric_type(type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-24 00:39:37 +01:00
										 |  |  | static uint32_t write_fx_4_type(const struct hlsl_type *type, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-06-30 23:06:11 +02:00
										 |  |  |     struct field_offsets | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t name; | 
					
						
							|  |  |  |         uint32_t semantic; | 
					
						
							|  |  |  |         uint32_t offset; | 
					
						
							|  |  |  |         uint32_t type; | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     uint32_t name_offset, offset, unpacked_size, packed_size, stride, numeric_desc; | 
					
						
							| 
									
										
										
										
											2024-02-24 00:39:37 +01:00
										 |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->unstructured; | 
					
						
							| 
									
										
										
										
											2024-06-30 23:06:11 +02:00
										 |  |  |     struct field_offsets *field_offsets = NULL; | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     const struct hlsl_type *element_type; | 
					
						
							| 
									
										
										
										
											2024-06-30 23:06:11 +02:00
										 |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							| 
									
										
										
										
											2024-02-24 00:39:37 +01:00
										 |  |  |     uint32_t elements_count = 0; | 
					
						
							|  |  |  |     const char *name; | 
					
						
							| 
									
										
										
										
											2024-06-30 23:06:11 +02:00
										 |  |  |     size_t i; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (type->class == HLSL_CLASS_ARRAY) | 
					
						
							|  |  |  |         elements_count = hlsl_get_multiarray_size(type); | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     element_type = hlsl_get_multiarray_element_type(type); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     name = get_fx_4_type_name(element_type); | 
					
						
							| 
									
										
										
										
											2024-02-11 00:02:16 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-22 18:06:03 +02:00
										 |  |  |     name_offset = write_string(name ? name : "<unnamed>", fx); | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     if (element_type->class == HLSL_CLASS_STRUCT) | 
					
						
							| 
									
										
										
										
											2024-06-30 23:06:11 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |         if (!(field_offsets = hlsl_calloc(ctx, element_type->e.record.field_count, sizeof(*field_offsets)))) | 
					
						
							| 
									
										
										
										
											2024-06-30 23:06:11 +02:00
										 |  |  |             return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |         for (i = 0; i < element_type->e.record.field_count; ++i) | 
					
						
							| 
									
										
										
										
											2024-06-30 23:06:11 +02:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |             const struct hlsl_struct_field *field = &element_type->e.record.fields[i]; | 
					
						
							| 
									
										
										
										
											2024-06-30 23:06:11 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             field_offsets[i].name = write_string(field->name, fx); | 
					
						
							|  |  |  |             field_offsets[i].semantic = write_string(field->semantic.raw_name, fx); | 
					
						
							| 
									
										
										
										
											2024-08-31 00:45:36 +02:00
										 |  |  |             field_offsets[i].offset = field->reg_offset[HLSL_REGSET_NUMERIC] * sizeof(float); | 
					
						
							| 
									
										
										
										
											2024-06-30 23:06:11 +02:00
										 |  |  |             field_offsets[i].type = write_type(field->type, fx); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     offset = put_u32_unaligned(buffer, name_offset); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     switch (element_type->class) | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         case HLSL_CLASS_SCALAR: | 
					
						
							|  |  |  |         case HLSL_CLASS_VECTOR: | 
					
						
							|  |  |  |         case HLSL_CLASS_MATRIX: | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  |             put_u32_unaligned(buffer, FX_4_TYPE_CLASS_NUMERIC); | 
					
						
							| 
									
										
										
										
											2024-04-09 16:05:33 -05:00
										 |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  |         case HLSL_CLASS_DEPTH_STENCIL_STATE: | 
					
						
							| 
									
										
										
										
											2024-02-27 13:31:20 -06:00
										 |  |  |         case HLSL_CLASS_DEPTH_STENCIL_VIEW: | 
					
						
							| 
									
										
										
										
											2024-02-06 19:41:15 -06:00
										 |  |  |         case HLSL_CLASS_PIXEL_SHADER: | 
					
						
							| 
									
										
										
										
											2024-04-24 11:12:08 +02:00
										 |  |  |         case HLSL_CLASS_RASTERIZER_STATE: | 
					
						
							| 
									
										
										
										
											2024-02-27 13:34:52 -06:00
										 |  |  |         case HLSL_CLASS_RENDER_TARGET_VIEW: | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |         case HLSL_CLASS_SAMPLER: | 
					
						
							| 
									
										
										
										
											2024-02-05 20:25:57 -06:00
										 |  |  |         case HLSL_CLASS_TEXTURE: | 
					
						
							| 
									
										
										
										
											2024-02-05 20:32:37 -06:00
										 |  |  |         case HLSL_CLASS_UAV: | 
					
						
							| 
									
										
										
										
											2024-02-06 17:33:26 -06:00
										 |  |  |         case HLSL_CLASS_VERTEX_SHADER: | 
					
						
							| 
									
										
										
										
											2024-08-05 18:41:23 +02:00
										 |  |  |         case HLSL_CLASS_COMPUTE_SHADER: | 
					
						
							|  |  |  |         case HLSL_CLASS_DOMAIN_SHADER: | 
					
						
							|  |  |  |         case HLSL_CLASS_HULL_SHADER: | 
					
						
							| 
									
										
										
										
											2024-08-06 16:41:46 +02:00
										 |  |  |         case HLSL_CLASS_GEOMETRY_SHADER: | 
					
						
							| 
									
										
										
										
											2024-08-07 12:49:04 +02:00
										 |  |  |         case HLSL_CLASS_BLEND_STATE: | 
					
						
							| 
									
										
										
										
											2024-08-13 23:33:16 +02:00
										 |  |  |         case HLSL_CLASS_STRING: | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  |             put_u32_unaligned(buffer, FX_4_TYPE_CLASS_OBJECT); | 
					
						
							| 
									
										
										
										
											2024-04-09 16:05:33 -05:00
										 |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |         case HLSL_CLASS_STRUCT: | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  |             put_u32_unaligned(buffer, FX_4_TYPE_CLASS_STRUCT); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2024-04-09 16:05:33 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         case HLSL_CLASS_ARRAY: | 
					
						
							| 
									
										
										
										
											2024-02-05 20:35:22 -06:00
										 |  |  |         case HLSL_CLASS_EFFECT_GROUP: | 
					
						
							| 
									
										
											  
											
												vkd3d-shader/hlsl: Introduce the "error" type.
Currently, if an expression successfully parses according to the bison grammar,
but for one reason or another cannot generate a meaningful IR instruction, we
abort parsing with YYABORT. This includes, for example, an undefined variable or
function, invalid swizzle or field reference, or a constructor with a complex or
non-numeric data type.
Aborting parsing is unfortunate, however, because it means that any further
errors in the program cannot be caught by the programmer, increasing the number
of times they will need to fix errors and recompile.
The idea of this patch is that any such expression will instead generate an IR
node whose data type is of HLSL_CLASS_ERROR. Any further expression which would
consume an "error" typed instruction will instead immediately return an
expression of type "error" (probably the same one) instead of aborting or doing
any other type-checking.
Currently these "error" instructions should not pass the parsing stage, since
hlsl_compile_shader() will immediately notice that compilation has failed and
skip any optimization, lowering, or bytecode-writing.
A further direction to take this is to pre-allocate one "error" expression
immediately when creating the HLSL parser, and return that expression when we
fail to allocate an hlsl_ir_node of any type. This means we do not need to
handle allocation errors when constructing nodes, saving us quite a lot of error
handling (which is not only tedious but currently often broken, if nothing else
by virtue of neglecting cleanup of local variables).
											
										 
											2024-08-29 12:48:23 -05:00
										 |  |  |         case HLSL_CLASS_ERROR: | 
					
						
							| 
									
										
										
										
											2024-02-06 17:08:01 -06:00
										 |  |  |         case HLSL_CLASS_PASS: | 
					
						
							| 
									
										
										
										
											2024-02-06 17:15:48 -06:00
										 |  |  |         case HLSL_CLASS_TECHNIQUE: | 
					
						
							| 
									
										
										
										
											2024-05-28 00:31:51 +02:00
										 |  |  |         case HLSL_CLASS_CONSTANT_BUFFER: | 
					
						
							| 
									
										
										
										
											2024-07-23 15:30:27 +02:00
										 |  |  |         case HLSL_CLASS_NULL: | 
					
						
							| 
									
										
										
										
											2024-11-08 16:15:32 +01:00
										 |  |  |         case HLSL_CLASS_STREAM_OUTPUT: | 
					
						
							| 
									
										
										
										
											2024-04-09 16:05:33 -05:00
										 |  |  |             vkd3d_unreachable(); | 
					
						
							| 
									
										
										
										
											2024-04-09 16:42:00 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         case HLSL_CLASS_VOID: | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |             FIXME("Writing type class %u is not implemented.\n", element_type->class); | 
					
						
							| 
									
										
										
										
											2024-04-09 16:42:00 -05:00
										 |  |  |             set_status(fx, VKD3D_ERROR_NOT_IMPLEMENTED); | 
					
						
							|  |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-30 23:31:50 +02:00
										 |  |  |     /* Structures can only contain numeric fields, this is validated during variable declaration. */ | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     unpacked_size = type->reg_size[HLSL_REGSET_NUMERIC] * sizeof(float); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-30 23:31:50 +02:00
										 |  |  |     packed_size = 0; | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     if (is_numeric_fx_4_type(element_type)) | 
					
						
							|  |  |  |         packed_size = hlsl_type_component_count(element_type) * sizeof(float); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     if (elements_count) | 
					
						
							| 
									
										
										
										
											2024-06-30 23:31:50 +02:00
										 |  |  |         packed_size *= elements_count; | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     stride = element_type->reg_size[HLSL_REGSET_NUMERIC] * sizeof(float); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     stride = align(stride, 4 * sizeof(float)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     put_u32_unaligned(buffer, elements_count); | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     put_u32_unaligned(buffer, unpacked_size); | 
					
						
							| 
									
										
										
										
											2024-06-30 23:31:50 +02:00
										 |  |  |     put_u32_unaligned(buffer, stride); | 
					
						
							|  |  |  |     put_u32_unaligned(buffer, packed_size); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     if (element_type->class == HLSL_CLASS_STRUCT) | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |         put_u32_unaligned(buffer, element_type->e.record.field_count); | 
					
						
							|  |  |  |         for (i = 0; i < element_type->e.record.field_count; ++i) | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2024-06-30 23:06:11 +02:00
										 |  |  |             const struct field_offsets *field = &field_offsets[i]; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-30 23:06:11 +02:00
										 |  |  |             put_u32_unaligned(buffer, field->name); | 
					
						
							|  |  |  |             put_u32_unaligned(buffer, field->semantic); | 
					
						
							|  |  |  |             put_u32_unaligned(buffer, field->offset); | 
					
						
							|  |  |  |             put_u32_unaligned(buffer, field->type); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-06-30 23:39:30 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (ctx->profile->major_version == 5) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             put_u32_unaligned(buffer, 0); /* Base class type */ | 
					
						
							|  |  |  |             put_u32_unaligned(buffer, 0); /* Interface count */ | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     else if (element_type->class == HLSL_CLASS_TEXTURE) | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-02-11 00:02:16 +01:00
										 |  |  |         static const uint32_t texture_type[] = | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  |             [HLSL_SAMPLER_DIM_GENERIC]   = FX_4_OBJECT_TYPE_TEXTURE, | 
					
						
							|  |  |  |             [HLSL_SAMPLER_DIM_1D]        = FX_4_OBJECT_TYPE_TEXTURE_1D, | 
					
						
							|  |  |  |             [HLSL_SAMPLER_DIM_1DARRAY]   = FX_4_OBJECT_TYPE_TEXTURE_1DARRAY, | 
					
						
							|  |  |  |             [HLSL_SAMPLER_DIM_2D]        = FX_4_OBJECT_TYPE_TEXTURE_2D, | 
					
						
							|  |  |  |             [HLSL_SAMPLER_DIM_2DARRAY]   = FX_4_OBJECT_TYPE_TEXTURE_2DARRAY, | 
					
						
							|  |  |  |             [HLSL_SAMPLER_DIM_2DMS]      = FX_4_OBJECT_TYPE_TEXTURE_2DMS, | 
					
						
							|  |  |  |             [HLSL_SAMPLER_DIM_2DMSARRAY] = FX_4_OBJECT_TYPE_TEXTURE_2DMSARRAY, | 
					
						
							|  |  |  |             [HLSL_SAMPLER_DIM_3D]        = FX_4_OBJECT_TYPE_TEXTURE_3D, | 
					
						
							|  |  |  |             [HLSL_SAMPLER_DIM_CUBE]      = FX_4_OBJECT_TYPE_TEXTURE_CUBE, | 
					
						
							|  |  |  |             [HLSL_SAMPLER_DIM_CUBEARRAY] = FX_4_OBJECT_TYPE_TEXTURE_CUBEARRAY, | 
					
						
							| 
									
										
										
										
											2024-02-11 00:02:16 +01:00
										 |  |  |         }; | 
					
						
							| 
									
										
										
										
											2024-02-05 20:25:57 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |         put_u32_unaligned(buffer, texture_type[element_type->sampler_dim]); | 
					
						
							| 
									
										
										
										
											2024-02-05 20:25:57 -06:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     else if (element_type->class == HLSL_CLASS_SAMPLER) | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |         put_u32_unaligned(buffer, FX_4_OBJECT_TYPE_SAMPLER_STATE); | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     else if (element_type->class == HLSL_CLASS_UAV) | 
					
						
							| 
									
										
										
										
											2024-02-05 20:25:57 -06:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-02-11 11:04:14 +01:00
										 |  |  |         static const uint32_t uav_type[] = | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  |             [HLSL_SAMPLER_DIM_1D]                = FX_5_OBJECT_TYPE_UAV_1D, | 
					
						
							|  |  |  |             [HLSL_SAMPLER_DIM_1DARRAY]           = FX_5_OBJECT_TYPE_UAV_1DARRAY, | 
					
						
							|  |  |  |             [HLSL_SAMPLER_DIM_2D]                = FX_5_OBJECT_TYPE_UAV_2D, | 
					
						
							|  |  |  |             [HLSL_SAMPLER_DIM_2DARRAY]           = FX_5_OBJECT_TYPE_UAV_2DARRAY, | 
					
						
							|  |  |  |             [HLSL_SAMPLER_DIM_3D]                = FX_5_OBJECT_TYPE_UAV_3D, | 
					
						
							|  |  |  |             [HLSL_SAMPLER_DIM_BUFFER]            = FX_5_OBJECT_TYPE_UAV_BUFFER, | 
					
						
							|  |  |  |             [HLSL_SAMPLER_DIM_STRUCTURED_BUFFER] = FX_5_OBJECT_TYPE_UAV_STRUCTURED_BUFFER, | 
					
						
							| 
									
										
										
										
											2024-11-05 13:43:28 +01:00
										 |  |  |             [HLSL_SAMPLER_DIM_RAW_BUFFER]        = FX_5_OBJECT_TYPE_UAV_RAW_BUFFER, | 
					
						
							| 
									
										
										
										
											2024-02-11 11:04:14 +01:00
										 |  |  |         }; | 
					
						
							| 
									
										
										
										
											2024-02-09 22:42:23 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |         put_u32_unaligned(buffer, uav_type[element_type->sampler_dim]); | 
					
						
							| 
									
										
										
										
											2024-02-05 20:32:37 -06:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     else if (element_type->class == HLSL_CLASS_DEPTH_STENCIL_VIEW) | 
					
						
							| 
									
										
										
										
											2024-02-27 13:31:20 -06:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-11-05 13:43:28 +01:00
										 |  |  |         put_u32_unaligned(buffer, FX_4_OBJECT_TYPE_DSV); | 
					
						
							| 
									
										
										
										
											2024-02-27 13:31:20 -06:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     else if (element_type->class == HLSL_CLASS_RENDER_TARGET_VIEW) | 
					
						
							| 
									
										
										
										
											2024-02-27 13:34:52 -06:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-11-05 13:43:28 +01:00
										 |  |  |         put_u32_unaligned(buffer, FX_4_OBJECT_TYPE_RTV); | 
					
						
							| 
									
										
										
										
											2024-02-27 13:34:52 -06:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     else if (element_type->class == HLSL_CLASS_PIXEL_SHADER) | 
					
						
							| 
									
										
										
										
											2024-02-06 17:33:26 -06:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |         put_u32_unaligned(buffer, FX_4_OBJECT_TYPE_PIXEL_SHADER); | 
					
						
							| 
									
										
										
										
											2024-02-06 17:33:26 -06:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     else if (element_type->class == HLSL_CLASS_VERTEX_SHADER) | 
					
						
							| 
									
										
										
										
											2024-02-05 20:32:37 -06:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |         put_u32_unaligned(buffer, FX_4_OBJECT_TYPE_VERTEX_SHADER); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     else if (element_type->class == HLSL_CLASS_RASTERIZER_STATE) | 
					
						
							| 
									
										
										
										
											2024-06-10 13:18:43 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |         put_u32_unaligned(buffer, FX_4_OBJECT_TYPE_RASTERIZER_STATE); | 
					
						
							| 
									
										
										
										
											2024-06-10 13:18:43 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     else if (element_type->class == HLSL_CLASS_DEPTH_STENCIL_STATE) | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |         put_u32_unaligned(buffer, FX_4_OBJECT_TYPE_DEPTH_STENCIL_STATE); | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     else if (element_type->class == HLSL_CLASS_BLEND_STATE) | 
					
						
							| 
									
										
										
										
											2024-08-07 12:59:24 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |         put_u32_unaligned(buffer, FX_4_OBJECT_TYPE_BLEND_STATE); | 
					
						
							| 
									
										
										
										
											2024-08-07 12:59:24 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     else if (element_type->class == HLSL_CLASS_STRING) | 
					
						
							| 
									
										
										
										
											2024-08-13 23:33:16 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  |         put_u32_unaligned(buffer, FX_4_OBJECT_TYPE_STRING); | 
					
						
							| 
									
										
										
										
											2024-08-13 23:33:16 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     else if (hlsl_is_numeric_type(element_type)) | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |         numeric_desc = get_fx_4_numeric_type_description(element_type, fx); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |         put_u32_unaligned(buffer, numeric_desc); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     else if (element_type->class == HLSL_CLASS_COMPUTE_SHADER) | 
					
						
							| 
									
										
										
										
											2024-08-05 20:32:24 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |         put_u32_unaligned(buffer, FX_5_OBJECT_TYPE_COMPUTE_SHADER); | 
					
						
							| 
									
										
										
										
											2024-08-05 20:32:24 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     else if (element_type->class == HLSL_CLASS_HULL_SHADER) | 
					
						
							| 
									
										
										
										
											2024-08-05 20:32:24 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |         put_u32_unaligned(buffer, FX_5_OBJECT_TYPE_HULL_SHADER); | 
					
						
							| 
									
										
										
										
											2024-08-05 20:32:24 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |     else if (element_type->class == HLSL_CLASS_DOMAIN_SHADER) | 
					
						
							| 
									
										
										
										
											2024-08-05 20:32:24 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |         put_u32_unaligned(buffer, FX_5_OBJECT_TYPE_DOMAIN_SHADER); | 
					
						
							| 
									
										
										
										
											2024-08-05 20:32:24 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-02-27 13:34:52 -06:00
										 |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-09-02 01:59:20 +02:00
										 |  |  |         FIXME("Type %u is not supported.\n", element_type->class); | 
					
						
							| 
									
										
										
										
											2024-02-27 13:34:52 -06:00
										 |  |  |         set_status(fx, VKD3D_ERROR_NOT_IMPLEMENTED); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-30 23:06:11 +02:00
										 |  |  |     vkd3d_free(field_offsets); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     return offset; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | static void write_fx_4_technique(struct hlsl_ir_var *var, struct fx_write_context *fx) | 
					
						
							| 
									
										
										
										
											2023-11-11 01:14:03 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							|  |  |  |     uint32_t name_offset, count = 0; | 
					
						
							|  |  |  |     struct hlsl_ir_var *pass; | 
					
						
							|  |  |  |     uint32_t count_offset; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |     name_offset = write_string(var->name, fx); | 
					
						
							| 
									
										
										
										
											2023-11-11 01:14:03 +01:00
										 |  |  |     put_u32(buffer, name_offset); | 
					
						
							|  |  |  |     count_offset = put_u32(buffer, 0); | 
					
						
							| 
									
										
										
										
											2024-06-15 23:44:08 +02:00
										 |  |  |     write_fx_4_annotations(var->annotations, fx); | 
					
						
							| 
									
										
										
										
											2023-11-11 01:14:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-15 23:44:08 +02:00
										 |  |  |     count = 0; | 
					
						
							| 
									
										
										
										
											2023-11-11 01:14:03 +01:00
										 |  |  |     LIST_FOR_EACH_ENTRY(pass, &var->scope->vars, struct hlsl_ir_var, scope_entry) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         write_pass(pass, fx); | 
					
						
							|  |  |  |         ++count; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-11 01:14:03 +01:00
										 |  |  |     set_u32(buffer, count_offset, count); | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_techniques(struct hlsl_scope *scope, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct hlsl_ir_var *var; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LIST_FOR_EACH_ENTRY(var, &scope->vars, struct hlsl_ir_var, scope_entry) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-01-16 12:09:41 +01:00
										 |  |  |         if (technique_matches_version(var, fx)) | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |             fx->ops->write_technique(var, fx); | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  |             ++fx->technique_count; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     set_status(fx, fx->unstructured.status); | 
					
						
							|  |  |  |     set_status(fx, fx->structured.status); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 18:18:40 +01:00
										 |  |  | static void write_group(struct hlsl_ir_var *var, struct fx_write_context *fx) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							| 
									
										
										
										
											2024-01-25 18:18:40 +01:00
										 |  |  |     uint32_t name_offset = write_string(var ? var->name : NULL, fx); | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  |     uint32_t count_offset, count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     put_u32(buffer, name_offset); | 
					
						
							|  |  |  |     count_offset = put_u32(buffer, 0); /* Technique count */ | 
					
						
							| 
									
										
										
										
											2024-06-15 23:44:08 +02:00
										 |  |  |     write_fx_4_annotations(var ? var->annotations : NULL, fx); | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     count = fx->technique_count; | 
					
						
							| 
									
										
										
										
											2024-01-25 18:18:40 +01:00
										 |  |  |     write_techniques(var ? var->scope : fx->ctx->globals, fx); | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  |     set_u32(buffer, count_offset, fx->technique_count - count); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ++fx->group_count; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 18:18:40 +01:00
										 |  |  | static void write_groups(struct fx_write_context *fx) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-01-25 18:18:40 +01:00
										 |  |  |     struct hlsl_scope *scope = fx->ctx->globals; | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  |     bool needs_default_group = false; | 
					
						
							|  |  |  |     struct hlsl_ir_var *var; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LIST_FOR_EACH_ENTRY(var, &scope->vars, struct hlsl_ir_var, scope_entry) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (technique_matches_version(var, fx)) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             needs_default_group = true; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (needs_default_group) | 
					
						
							| 
									
										
										
										
											2024-01-25 18:18:40 +01:00
										 |  |  |         write_group(NULL, fx); | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  |     LIST_FOR_EACH_ENTRY(var, &scope->vars, struct hlsl_ir_var, scope_entry) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         const struct hlsl_type *type = var->data_type; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-05 20:35:22 -06:00
										 |  |  |         if (type->class == HLSL_CLASS_EFFECT_GROUP) | 
					
						
							| 
									
										
										
										
											2024-01-25 18:18:40 +01:00
										 |  |  |             write_group(var, fx); | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | static uint32_t write_fx_2_string(const char *string, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->unstructured; | 
					
						
							|  |  |  |     const char *s = string ? string : ""; | 
					
						
							| 
									
										
										
										
											2024-03-11 18:56:50 +01:00
										 |  |  |     static const char tail[3]; | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |     uint32_t size, offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     size = strlen(s) + 1; | 
					
						
							|  |  |  |     offset = put_u32(buffer, size); | 
					
						
							|  |  |  |     bytecode_put_bytes(buffer, s, size); | 
					
						
							| 
									
										
										
										
											2024-03-11 18:56:50 +01:00
										 |  |  |     size %= 4; | 
					
						
							|  |  |  |     if (size) | 
					
						
							|  |  |  |         bytecode_put_bytes_unaligned(buffer, tail, 4 - size); | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |     return offset; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-17 14:06:05 +02:00
										 |  |  | static uint32_t get_fx_2_type_class(const struct hlsl_type *type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (type->class == HLSL_CLASS_MATRIX) | 
					
						
							|  |  |  |         return D3DXPC_MATRIX_ROWS; | 
					
						
							|  |  |  |     return hlsl_sm1_class(type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-26 19:40:00 +02:00
										 |  |  | struct fx_2_write_type_context | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-05-26 19:40:00 +02:00
										 |  |  |     uint32_t *names; | 
					
						
							|  |  |  |     uint32_t *semantics; | 
					
						
							|  |  |  |     uint32_t count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     uint32_t offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool is_combined_sampler; | 
					
						
							|  |  |  |     struct fx_write_context *fx; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void count_type_iter(const struct hlsl_type *type, const char *name, | 
					
						
							|  |  |  |         const struct hlsl_semantic *semantic, void *context) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct fx_2_write_type_context *ctx = context; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ++ctx->count; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_fx_2_type_iter(const struct hlsl_type *type, const char *name, | 
					
						
							|  |  |  |         const struct hlsl_semantic *semantic, void *context) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct fx_2_write_type_context *ctx = context; | 
					
						
							|  |  |  |     struct fx_write_context *fx = ctx->fx; | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer; | 
					
						
							|  |  |  |     uint32_t offset, elements_count = 0; | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* Resolve arrays to element type and number of elements. */ | 
					
						
							|  |  |  |     if (type->class == HLSL_CLASS_ARRAY) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         elements_count = hlsl_get_multiarray_size(type); | 
					
						
							|  |  |  |         type = hlsl_get_multiarray_element_type(type); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-26 19:40:00 +02:00
										 |  |  |     buffer = &fx->unstructured; | 
					
						
							|  |  |  |     offset = put_u32(buffer, hlsl_sm1_base_type(type, ctx->is_combined_sampler)); | 
					
						
							| 
									
										
										
										
											2024-08-17 14:06:05 +02:00
										 |  |  |     put_u32(buffer, get_fx_2_type_class(type)); | 
					
						
							| 
									
										
										
										
											2025-05-26 19:40:00 +02:00
										 |  |  |     *ctx->names++ = put_u32(buffer, 0); | 
					
						
							|  |  |  |     *ctx->semantics++ = put_u32(buffer, 0); | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |     put_u32(buffer, elements_count); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (type->class) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         case HLSL_CLASS_VECTOR: | 
					
						
							| 
									
										
										
										
											2024-10-30 11:51:05 -05:00
										 |  |  |             put_u32(buffer, type->e.numeric.dimx); | 
					
						
							|  |  |  |             put_u32(buffer, type->e.numeric.dimy); | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  |         case HLSL_CLASS_SCALAR: | 
					
						
							|  |  |  |         case HLSL_CLASS_MATRIX: | 
					
						
							| 
									
										
										
										
											2024-10-30 11:51:05 -05:00
										 |  |  |             put_u32(buffer, type->e.numeric.dimy); | 
					
						
							|  |  |  |             put_u32(buffer, type->e.numeric.dimx); | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  |         case HLSL_CLASS_STRUCT: | 
					
						
							|  |  |  |             put_u32(buffer, type->e.record.field_count); | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2024-07-11 14:02:31 +02:00
										 |  |  |         case HLSL_CLASS_VERTEX_SHADER: | 
					
						
							|  |  |  |         case HLSL_CLASS_PIXEL_SHADER: | 
					
						
							|  |  |  |             fx->shader_count += elements_count; | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |         default: | 
					
						
							|  |  |  |             ; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-26 19:40:00 +02:00
										 |  |  |     /* Save the offset of the top level type. */ | 
					
						
							|  |  |  |     if (!ctx->offset) | 
					
						
							|  |  |  |         ctx->offset = offset; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_fx_2_type_strings_iter(const struct hlsl_type *type, const char *name, | 
					
						
							|  |  |  |         const struct hlsl_semantic *semantic, void *context) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct fx_2_write_type_context *ctx = context; | 
					
						
							|  |  |  |     struct fx_write_context *fx = ctx->fx; | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     buffer = &fx->unstructured; | 
					
						
							|  |  |  |     set_u32(buffer, *ctx->names++, write_string(name, fx)); | 
					
						
							|  |  |  |     set_u32(buffer, *ctx->semantics++, semantic->raw_name ? write_string(semantic->raw_name, fx) : 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void foreach_type(const struct hlsl_type *type, const char *name, const struct hlsl_semantic *semantic, | 
					
						
							|  |  |  |         void (*iter_func)(const struct hlsl_type *type, const char *name, const struct hlsl_semantic *semantic, void *context), | 
					
						
							|  |  |  |         void *context) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     iter_func(type, name, semantic, context); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     type = hlsl_get_multiarray_element_type(type); | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |     if (type->class == HLSL_CLASS_STRUCT) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-05-26 19:40:00 +02:00
										 |  |  |         for (size_t i = 0; i < type->e.record.field_count; ++i) | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |         { | 
					
						
							|  |  |  |             const struct hlsl_struct_field *field = &type->e.record.fields[i]; | 
					
						
							| 
									
										
										
										
											2025-05-26 19:40:00 +02:00
										 |  |  |             foreach_type(field->type, field->name, &field->semantic, iter_func, context); | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-05-26 19:40:00 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-26 19:40:00 +02:00
										 |  |  | static uint32_t write_fx_2_parameter(const struct hlsl_ir_var *var, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct fx_2_write_type_context ctx = { .fx = fx, .is_combined_sampler = var->is_combined_sampler }; | 
					
						
							|  |  |  |     uint32_t *offsets; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Parameter type information has to be stored in a contiguous segment, so
 | 
					
						
							|  |  |  |      * that any structure fields come right after each other. To achieve that | 
					
						
							|  |  |  |      * the variable length string data is written after the type data. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Calculate the number of string entries needed for this type. */ | 
					
						
							|  |  |  |     foreach_type(var->data_type, var->name, &var->semantic, count_type_iter, &ctx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!(offsets = calloc(ctx.count, 2 * sizeof(*offsets)))) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Writing type information also sets string offsets. */ | 
					
						
							|  |  |  |     ctx.names = offsets; | 
					
						
							|  |  |  |     ctx.semantics = &offsets[ctx.count]; | 
					
						
							|  |  |  |     foreach_type(var->data_type, var->name, &var->semantic, write_fx_2_type_iter, &ctx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Now the final pass to write the string data. */ | 
					
						
							|  |  |  |     ctx.names = offsets; | 
					
						
							|  |  |  |     ctx.semantics = &offsets[ctx.count]; | 
					
						
							|  |  |  |     foreach_type(var->data_type, var->name, &var->semantic, write_fx_2_type_strings_iter, &ctx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     free(offsets); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ctx.offset; | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | static void write_fx_2_technique(struct hlsl_ir_var *var, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  |     uint32_t name_offset, pass_count_offset, annotation_count_offset, count = 0; | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							| 
									
										
										
										
											2023-11-15 23:31:01 +01:00
										 |  |  |     struct hlsl_ir_var *pass; | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     name_offset = write_string(var->name, fx); | 
					
						
							|  |  |  |     put_u32(buffer, name_offset); | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  |     annotation_count_offset = put_u32(buffer, 0); | 
					
						
							|  |  |  |     pass_count_offset = put_u32(buffer, 0); | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  |     write_fx_2_annotations(var, annotation_count_offset, fx); | 
					
						
							| 
									
										
										
										
											2023-11-15 23:31:01 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     LIST_FOR_EACH_ENTRY(pass, &var->scope->vars, struct hlsl_ir_var, scope_entry) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         write_pass(pass, fx); | 
					
						
							|  |  |  |         ++count; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  |     set_u32(buffer, pass_count_offset, count); | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-23 10:16:30 +02:00
										 |  |  | /* Effects represent bool values as 1/0, as opposed to ~0u/0 as used by
 | 
					
						
							|  |  |  |  * Direct3D shader model 4+. */ | 
					
						
							|  |  |  | static uint32_t get_fx_default_numeric_value(const struct hlsl_type *type, uint32_t value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (type->e.numeric.type == HLSL_TYPE_BOOL) | 
					
						
							|  |  |  |         return !!value; | 
					
						
							|  |  |  |     return value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-09 20:20:01 +02:00
										 |  |  | static uint32_t write_fx_2_default_value(struct hlsl_type *value_type, struct hlsl_default_value *value, | 
					
						
							|  |  |  |             struct fx_write_context *fx) | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-07-09 20:20:01 +02:00
										 |  |  |     const struct hlsl_type *type = hlsl_get_multiarray_element_type(value_type); | 
					
						
							|  |  |  |     uint32_t elements_count = hlsl_get_multiarray_size(value_type), i, j; | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->unstructured; | 
					
						
							|  |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							|  |  |  |     uint32_t offset = buffer->size; | 
					
						
							|  |  |  |     unsigned int comp_count; | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-09 20:20:01 +02:00
										 |  |  |     if (!value) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     comp_count = hlsl_type_component_count(type); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < elements_count; ++i) | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-07-09 20:20:01 +02:00
										 |  |  |         switch (type->class) | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2024-07-09 20:20:01 +02:00
										 |  |  |             case HLSL_CLASS_SCALAR: | 
					
						
							|  |  |  |             case HLSL_CLASS_VECTOR: | 
					
						
							|  |  |  |             case HLSL_CLASS_MATRIX: | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 switch (type->e.numeric.type) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     case HLSL_TYPE_FLOAT: | 
					
						
							|  |  |  |                     case HLSL_TYPE_HALF: | 
					
						
							|  |  |  |                     case HLSL_TYPE_INT: | 
					
						
							|  |  |  |                     case HLSL_TYPE_UINT: | 
					
						
							|  |  |  |                     case HLSL_TYPE_BOOL: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         for (j = 0; j < comp_count; ++j) | 
					
						
							|  |  |  |                         { | 
					
						
							| 
									
										
										
										
											2025-05-23 10:16:30 +02:00
										 |  |  |                             put_u32(buffer, get_fx_default_numeric_value(type, value->number.u)); | 
					
						
							| 
									
										
										
										
											2024-07-09 20:20:01 +02:00
										 |  |  |                             value++; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         break; | 
					
						
							|  |  |  |                     default: | 
					
						
							|  |  |  |                         hlsl_fixme(ctx, &ctx->location, "Writing default values for numeric type %u is not implemented.", | 
					
						
							|  |  |  |                                 type->e.numeric.type); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             case HLSL_CLASS_STRUCT: | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 struct hlsl_struct_field *fields = type->e.record.fields; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 for (j = 0; j < type->e.record.field_count; ++j) | 
					
						
							|  |  |  |                 { | 
					
						
							| 
									
										
										
										
											2025-05-23 14:34:22 +02:00
										 |  |  |                     write_fx_2_default_value(fields[j].type, value, fx); | 
					
						
							|  |  |  |                     value += hlsl_type_component_count(fields[j].type); | 
					
						
							| 
									
										
										
										
											2024-07-09 20:20:01 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 hlsl_fixme(ctx, &ctx->location, "Writing default values for class %u is not implemented.", type->class); | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-09 20:20:01 +02:00
										 |  |  |     return offset; | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  | static uint32_t write_fx_2_object_initializer(const struct hlsl_ir_var *var, struct fx_write_context *fx) | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |     const struct hlsl_type *type = hlsl_get_multiarray_element_type(var->data_type); | 
					
						
							|  |  |  |     unsigned int i, elements_count = hlsl_get_multiarray_size(var->data_type); | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->objects; | 
					
						
							|  |  |  |     uint32_t offset = fx->unstructured.size, id, size; | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |     const void *data; | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |     for (i = 0; i < elements_count; ++i) | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |         if (type->class == HLSL_CLASS_SAMPLER) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             hlsl_fixme(ctx, &var->loc, "Writing fx_2_0 sampler objects initializers is not implemented."); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             switch (type->class) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 case HLSL_CLASS_STRING: | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     const char *string = var->default_values[i].string ? var->default_values[i].string : ""; | 
					
						
							|  |  |  |                     size = strlen(string) + 1; | 
					
						
							|  |  |  |                     data = string; | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 case HLSL_CLASS_TEXTURE: | 
					
						
							|  |  |  |                     size = 0; | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 case HLSL_CLASS_PIXEL_SHADER: | 
					
						
							|  |  |  |                 case HLSL_CLASS_VERTEX_SHADER: | 
					
						
							|  |  |  |                     size = 0; | 
					
						
							|  |  |  |                     hlsl_fixme(ctx, &var->loc, "Writing fx_2_0 shader objects initializers is not implemented."); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 default: | 
					
						
							|  |  |  |                     vkd3d_unreachable(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             id = fx->object_variable_count++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             put_u32(&fx->unstructured, id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             put_u32(buffer, id); | 
					
						
							|  |  |  |             put_u32(buffer, size); | 
					
						
							|  |  |  |             if (size) | 
					
						
							| 
									
										
										
										
											2025-03-18 10:08:14 +01:00
										 |  |  |             { | 
					
						
							|  |  |  |                 static const uint32_t pad; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |                 bytecode_put_bytes(buffer, data, size); | 
					
						
							| 
									
										
										
										
											2025-03-18 10:08:14 +01:00
										 |  |  |                 if (size % 4) | 
					
						
							|  |  |  |                     bytecode_put_bytes_unaligned(buffer, &pad, 4 - (size % 4)); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |     return offset; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uint32_t write_fx_2_initial_value(const struct hlsl_ir_var *var, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const struct hlsl_type *type = hlsl_get_multiarray_element_type(var->data_type); | 
					
						
							|  |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							|  |  |  |     uint32_t offset; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-09 19:01:34 -05:00
										 |  |  |     /* Note that struct fields must all be numeric;
 | 
					
						
							|  |  |  |      * this was validated in check_invalid_object_fields(). */ | 
					
						
							|  |  |  |     switch (type->class) | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-04-09 19:01:34 -05:00
										 |  |  |         case HLSL_CLASS_SCALAR: | 
					
						
							|  |  |  |         case HLSL_CLASS_VECTOR: | 
					
						
							|  |  |  |         case HLSL_CLASS_MATRIX: | 
					
						
							|  |  |  |         case HLSL_CLASS_STRUCT: | 
					
						
							| 
									
										
										
										
											2024-07-09 20:20:01 +02:00
										 |  |  |             offset = write_fx_2_default_value(var->data_type, var->default_values, fx); | 
					
						
							| 
									
										
										
										
											2024-04-09 19:01:34 -05:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  |         case HLSL_CLASS_SAMPLER: | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |         case HLSL_CLASS_TEXTURE: | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  |         case HLSL_CLASS_STRING: | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |         case HLSL_CLASS_PIXEL_SHADER: | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  |         case HLSL_CLASS_VERTEX_SHADER: | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |             offset = write_fx_2_object_initializer(var, fx); | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-09 19:01:34 -05:00
										 |  |  |         default: | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |             offset = 0; | 
					
						
							|  |  |  |             hlsl_fixme(ctx, &var->loc, "Writing initializer not implemented for parameter class %#x.", type->class); | 
					
						
							| 
									
										
										
										
											2024-04-09 19:01:34 -05:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return offset; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-09 16:40:56 -05:00
										 |  |  | static bool is_type_supported_fx_2(struct hlsl_ctx *ctx, const struct hlsl_type *type, | 
					
						
							|  |  |  |         const struct vkd3d_shader_location *loc) | 
					
						
							| 
									
										
										
										
											2024-03-19 10:46:31 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-04-09 16:40:56 -05:00
										 |  |  |     switch (type->class) | 
					
						
							| 
									
										
										
										
											2024-03-19 10:46:31 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-04-09 16:40:56 -05:00
										 |  |  |         case HLSL_CLASS_STRUCT: | 
					
						
							|  |  |  |             /* Note that the fields must all be numeric; this was validated in
 | 
					
						
							|  |  |  |              * check_invalid_object_fields(). */ | 
					
						
							| 
									
										
										
										
											2024-03-19 10:46:31 +01:00
										 |  |  |             return true; | 
					
						
							| 
									
										
										
										
											2024-04-09 16:40:56 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         case HLSL_CLASS_SCALAR: | 
					
						
							|  |  |  |         case HLSL_CLASS_VECTOR: | 
					
						
							|  |  |  |         case HLSL_CLASS_MATRIX: | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case HLSL_CLASS_ARRAY: | 
					
						
							|  |  |  |             return is_type_supported_fx_2(ctx, type->e.array.type, loc); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-05 20:25:57 -06:00
										 |  |  |         case HLSL_CLASS_TEXTURE: | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |         case HLSL_CLASS_SAMPLER: | 
					
						
							| 
									
										
										
										
											2024-02-05 20:25:57 -06:00
										 |  |  |             switch (type->sampler_dim) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 case HLSL_SAMPLER_DIM_1D: | 
					
						
							|  |  |  |                 case HLSL_SAMPLER_DIM_2D: | 
					
						
							|  |  |  |                 case HLSL_SAMPLER_DIM_3D: | 
					
						
							|  |  |  |                 case HLSL_SAMPLER_DIM_CUBE: | 
					
						
							|  |  |  |                 case HLSL_SAMPLER_DIM_GENERIC: | 
					
						
							|  |  |  |                     return true; | 
					
						
							|  |  |  |                 default: | 
					
						
							|  |  |  |                     return false; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-05 20:04:02 -06:00
										 |  |  |         case HLSL_CLASS_STRING: | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |             return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case HLSL_CLASS_PIXEL_SHADER: | 
					
						
							| 
									
										
										
										
											2024-02-06 17:33:26 -06:00
										 |  |  |         case HLSL_CLASS_VERTEX_SHADER: | 
					
						
							| 
									
										
										
										
											2024-02-05 20:04:02 -06:00
										 |  |  |             hlsl_fixme(ctx, loc, "Write fx 2.0 parameter class %#x.", type->class); | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  |         case HLSL_CLASS_DEPTH_STENCIL_STATE: | 
					
						
							| 
									
										
										
										
											2024-02-27 13:31:20 -06:00
										 |  |  |         case HLSL_CLASS_DEPTH_STENCIL_VIEW: | 
					
						
							| 
									
										
										
										
											2024-02-05 20:32:37 -06:00
										 |  |  |         case HLSL_CLASS_UAV: | 
					
						
							| 
									
										
										
										
											2024-04-24 11:12:08 +02:00
										 |  |  |         case HLSL_CLASS_RASTERIZER_STATE: | 
					
						
							| 
									
										
										
										
											2024-02-27 13:34:52 -06:00
										 |  |  |         case HLSL_CLASS_RENDER_TARGET_VIEW: | 
					
						
							| 
									
										
										
										
											2024-04-09 16:42:00 -05:00
										 |  |  |         case HLSL_CLASS_VOID: | 
					
						
							| 
									
										
										
										
											2024-08-05 18:41:23 +02:00
										 |  |  |         case HLSL_CLASS_COMPUTE_SHADER: | 
					
						
							|  |  |  |         case HLSL_CLASS_DOMAIN_SHADER: | 
					
						
							|  |  |  |         case HLSL_CLASS_HULL_SHADER: | 
					
						
							| 
									
										
										
										
											2024-08-06 16:41:46 +02:00
										 |  |  |         case HLSL_CLASS_GEOMETRY_SHADER: | 
					
						
							| 
									
										
										
										
											2024-08-07 12:49:04 +02:00
										 |  |  |         case HLSL_CLASS_BLEND_STATE: | 
					
						
							| 
									
										
										
										
											2024-04-09 16:42:00 -05:00
										 |  |  |             return false; | 
					
						
							| 
									
										
										
										
											2024-02-05 20:35:22 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |         case HLSL_CLASS_EFFECT_GROUP: | 
					
						
							| 
									
										
											  
											
												vkd3d-shader/hlsl: Introduce the "error" type.
Currently, if an expression successfully parses according to the bison grammar,
but for one reason or another cannot generate a meaningful IR instruction, we
abort parsing with YYABORT. This includes, for example, an undefined variable or
function, invalid swizzle or field reference, or a constructor with a complex or
non-numeric data type.
Aborting parsing is unfortunate, however, because it means that any further
errors in the program cannot be caught by the programmer, increasing the number
of times they will need to fix errors and recompile.
The idea of this patch is that any such expression will instead generate an IR
node whose data type is of HLSL_CLASS_ERROR. Any further expression which would
consume an "error" typed instruction will instead immediately return an
expression of type "error" (probably the same one) instead of aborting or doing
any other type-checking.
Currently these "error" instructions should not pass the parsing stage, since
hlsl_compile_shader() will immediately notice that compilation has failed and
skip any optimization, lowering, or bytecode-writing.
A further direction to take this is to pre-allocate one "error" expression
immediately when creating the HLSL parser, and return that expression when we
fail to allocate an hlsl_ir_node of any type. This means we do not need to
handle allocation errors when constructing nodes, saving us quite a lot of error
handling (which is not only tedious but currently often broken, if nothing else
by virtue of neglecting cleanup of local variables).
											
										 
											2024-08-29 12:48:23 -05:00
										 |  |  |         case HLSL_CLASS_ERROR: | 
					
						
							| 
									
										
										
										
											2024-02-06 17:08:01 -06:00
										 |  |  |         case HLSL_CLASS_PASS: | 
					
						
							| 
									
										
										
										
											2024-02-06 17:15:48 -06:00
										 |  |  |         case HLSL_CLASS_TECHNIQUE: | 
					
						
							| 
									
										
										
										
											2024-05-28 00:31:51 +02:00
										 |  |  |         case HLSL_CLASS_CONSTANT_BUFFER: | 
					
						
							| 
									
										
										
										
											2024-07-23 15:30:27 +02:00
										 |  |  |         case HLSL_CLASS_NULL: | 
					
						
							| 
									
										
										
										
											2024-11-08 16:15:32 +01:00
										 |  |  |         case HLSL_CLASS_STREAM_OUTPUT: | 
					
						
							| 
									
										
										
										
											2024-02-05 20:35:22 -06:00
										 |  |  |             /* This cannot appear as an extern variable. */ | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2024-03-19 10:46:31 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-09 16:40:56 -05:00
										 |  |  |     vkd3d_unreachable(); | 
					
						
							| 
									
										
										
										
											2024-03-19 10:46:31 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  | static void write_fx_2_parameters(struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  |     uint32_t desc_offset, value_offset, flags, annotation_count_offset; | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							| 
									
										
										
										
											2024-03-14 13:49:14 +01:00
										 |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |     struct hlsl_ir_var *var; | 
					
						
							| 
									
										
										
										
											2024-03-19 01:09:58 +01:00
										 |  |  |     enum fx_2_parameter_flags | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         IS_SHARED = 0x1, | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-14 13:49:14 +01:00
										 |  |  |     LIST_FOR_EACH_ENTRY(var, &ctx->extern_vars, struct hlsl_ir_var, extern_entry) | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-04-09 16:40:56 -05:00
										 |  |  |         if (!is_type_supported_fx_2(ctx, var->data_type, &var->loc)) | 
					
						
							| 
									
										
										
										
											2024-03-19 10:46:31 +01:00
										 |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-26 19:40:00 +02:00
										 |  |  |         desc_offset = write_fx_2_parameter(var, fx); | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |         value_offset = write_fx_2_initial_value(var, fx); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-19 01:09:58 +01:00
										 |  |  |         flags = 0; | 
					
						
							|  |  |  |         if (var->storage_modifiers & HLSL_STORAGE_SHARED) | 
					
						
							|  |  |  |             flags |= IS_SHARED; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  |         put_u32(buffer, desc_offset); | 
					
						
							|  |  |  |         put_u32(buffer, value_offset); | 
					
						
							|  |  |  |         put_u32(buffer, flags); | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  |         annotation_count_offset = put_u32(buffer, 0); | 
					
						
							|  |  |  |         write_fx_2_annotations(var, annotation_count_offset, fx); | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         ++fx->parameter_count; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  | static void write_fx_2_annotation(struct hlsl_ir_var *var, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							|  |  |  |     uint32_t desc_offset, value_offset; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-26 19:40:00 +02:00
										 |  |  |     desc_offset = write_fx_2_parameter(var, fx); | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  |     value_offset = write_fx_2_initial_value(var, fx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     put_u32(buffer, desc_offset); | 
					
						
							|  |  |  |     put_u32(buffer, value_offset); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 23:35:08 +02:00
										 |  |  | static void write_fx_2_state_assignment(const struct hlsl_ir_var *var, | 
					
						
							|  |  |  |         struct hlsl_state_block_entry *entry, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     hlsl_fixme(ctx, &var->loc, "Writing fx_2_0 state assignments is not implemented."); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | static const struct fx_write_context_ops fx_2_ops = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     .write_string = write_fx_2_string, | 
					
						
							|  |  |  |     .write_technique = write_fx_2_technique, | 
					
						
							| 
									
										
										
										
											2023-11-15 23:31:01 +01:00
										 |  |  |     .write_pass = write_fx_2_pass, | 
					
						
							| 
									
										
										
										
											2024-08-17 22:23:46 +02:00
										 |  |  |     .write_annotation = write_fx_2_annotation, | 
					
						
							| 
									
										
										
										
											2025-06-16 23:35:08 +02:00
										 |  |  |     .write_state_assignment = write_fx_2_state_assignment, | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int hlsl_fx_2_write(struct hlsl_ctx *ctx, struct vkd3d_shader_code *out) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-07-11 14:02:31 +02:00
										 |  |  |     uint32_t offset, size, technique_count, shader_count, parameter_count, object_count; | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |     struct vkd3d_bytecode_buffer buffer = { 0 }; | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *structured; | 
					
						
							|  |  |  |     struct fx_write_context fx; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fx_write_context_init(ctx, &fx_2_ops, &fx); | 
					
						
							| 
									
										
										
										
											2024-03-13 11:37:17 +01:00
										 |  |  |     fx.object_variable_count = 1; | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |     structured = &fx.structured; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* First entry is always zeroed and skipped. */ | 
					
						
							|  |  |  |     put_u32(&fx.unstructured, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     put_u32(&buffer, 0xfeff0901); /* Version. */ | 
					
						
							|  |  |  |     offset = put_u32(&buffer, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |     parameter_count = put_u32(structured, 0); /* Parameter count */ | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |     technique_count = put_u32(structured, 0); | 
					
						
							| 
									
										
										
										
											2024-07-11 14:02:31 +02:00
										 |  |  |     shader_count = put_u32(structured, 0); | 
					
						
							| 
									
										
										
										
											2024-03-13 11:37:17 +01:00
										 |  |  |     object_count = put_u32(structured, 0); | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-13 12:52:11 +01:00
										 |  |  |     write_fx_2_parameters(&fx); | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |     write_techniques(ctx->globals, &fx); | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |     put_u32(structured, fx.object_variable_count - 1); | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |     put_u32(structured, 0); /* Resource count */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |     bytecode_put_bytes(structured, fx.objects.data, fx.objects.size); | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |     /* TODO: resources */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |     set_u32(structured, parameter_count, fx.parameter_count); | 
					
						
							|  |  |  |     set_u32(structured, object_count, fx.object_variable_count); | 
					
						
							|  |  |  |     set_u32(structured, technique_count, fx.technique_count); | 
					
						
							|  |  |  |     set_u32(structured, shader_count, fx.shader_count); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |     size = align(fx.unstructured.size, 4); | 
					
						
							|  |  |  |     set_u32(&buffer, offset, size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bytecode_put_bytes(&buffer, fx.unstructured.data, fx.unstructured.size); | 
					
						
							|  |  |  |     bytecode_put_bytes(&buffer, fx.structured.data, fx.structured.size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_free(fx.unstructured.data); | 
					
						
							|  |  |  |     vkd3d_free(fx.structured.data); | 
					
						
							| 
									
										
										
										
											2024-08-18 23:45:38 +02:00
										 |  |  |     vkd3d_free(fx.objects.data); | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-23 02:11:38 +01:00
										 |  |  |     if (!fx.technique_count) | 
					
						
							|  |  |  |         hlsl_error(ctx, &ctx->location, VKD3D_SHADER_ERROR_HLSL_MISSING_TECHNIQUE, "No techniques found."); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (fx.status < 0) | 
					
						
							|  |  |  |         ctx->result = fx.status; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!ctx->result) | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         out->code = buffer.data; | 
					
						
							|  |  |  |         out->size = buffer.size; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return fx_write_context_cleanup(&fx); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 23:35:08 +02:00
										 |  |  | static void write_fx_4_state_assignment(const struct hlsl_ir_var *var, | 
					
						
							|  |  |  |         struct hlsl_state_block_entry *entry, struct fx_write_context *fx); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | static const struct fx_write_context_ops fx_4_ops = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     .write_string = write_fx_4_string, | 
					
						
							|  |  |  |     .write_technique = write_fx_4_technique, | 
					
						
							| 
									
										
										
										
											2023-11-15 23:31:01 +01:00
										 |  |  |     .write_pass = write_fx_4_pass, | 
					
						
							| 
									
										
										
										
											2024-06-15 23:44:08 +02:00
										 |  |  |     .write_annotation = write_fx_4_annotation, | 
					
						
							| 
									
										
										
										
											2025-06-16 23:35:08 +02:00
										 |  |  |     .write_state_assignment = write_fx_4_state_assignment, | 
					
						
							| 
									
										
										
										
											2024-02-25 01:08:46 +01:00
										 |  |  |     .are_child_effects_supported = true, | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-14 00:59:57 +02:00
										 |  |  | static uint32_t write_fx_4_default_value(struct hlsl_type *value_type, struct hlsl_default_value *value, | 
					
						
							|  |  |  |             struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const struct hlsl_type *type = hlsl_get_multiarray_element_type(value_type); | 
					
						
							|  |  |  |     uint32_t elements_count = hlsl_get_multiarray_size(value_type), i, j; | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->unstructured; | 
					
						
							|  |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							|  |  |  |     uint32_t offset = buffer->size; | 
					
						
							|  |  |  |     unsigned int comp_count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!value) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     comp_count = hlsl_type_component_count(type); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < elements_count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         switch (type->class) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             case HLSL_CLASS_SCALAR: | 
					
						
							|  |  |  |             case HLSL_CLASS_VECTOR: | 
					
						
							|  |  |  |             case HLSL_CLASS_MATRIX: | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 switch (type->e.numeric.type) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     case HLSL_TYPE_FLOAT: | 
					
						
							| 
									
										
										
										
											2024-09-26 14:52:11 +02:00
										 |  |  |                     case HLSL_TYPE_HALF: | 
					
						
							| 
									
										
										
										
											2024-06-14 00:59:57 +02:00
										 |  |  |                     case HLSL_TYPE_INT: | 
					
						
							|  |  |  |                     case HLSL_TYPE_UINT: | 
					
						
							|  |  |  |                     case HLSL_TYPE_BOOL: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         for (j = 0; j < comp_count; ++j) | 
					
						
							|  |  |  |                         { | 
					
						
							| 
									
										
										
										
											2025-05-23 10:16:30 +02:00
										 |  |  |                             put_u32_unaligned(buffer, get_fx_default_numeric_value(type, value->number.u)); | 
					
						
							| 
									
										
										
										
											2024-06-14 00:59:57 +02:00
										 |  |  |                             value++; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         break; | 
					
						
							|  |  |  |                     default: | 
					
						
							|  |  |  |                         hlsl_fixme(ctx, &ctx->location, "Writing default values for numeric type %u is not implemented.", | 
					
						
							|  |  |  |                                 type->e.numeric.type); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             case HLSL_CLASS_STRUCT: | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 struct hlsl_struct_field *fields = type->e.record.fields; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 for (j = 0; j < type->e.record.field_count; ++j) | 
					
						
							|  |  |  |                 { | 
					
						
							| 
									
										
										
										
											2025-05-23 14:34:22 +02:00
										 |  |  |                     write_fx_4_default_value(fields[j].type, value, fx); | 
					
						
							|  |  |  |                     value += hlsl_type_component_count(fields[j].type); | 
					
						
							| 
									
										
										
										
											2024-06-14 00:59:57 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 hlsl_fixme(ctx, &ctx->location, "Writing default values for class %u is not implemented.", type->class); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return offset; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-13 23:33:16 +02:00
										 |  |  | static void write_fx_4_string_initializer(struct hlsl_ir_var *var, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t elements_count = hlsl_get_multiarray_size(var->data_type), i; | 
					
						
							|  |  |  |     const struct hlsl_default_value *value = var->default_values; | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							|  |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							|  |  |  |     uint32_t offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!value) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         hlsl_error(ctx, &var->loc, VKD3D_SHADER_ERROR_HLSL_INVALID_SYNTAX, "String objects have to be initialized."); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < elements_count; ++i, ++value) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         offset = write_fx_4_string(value->string, fx); | 
					
						
							|  |  |  |         put_u32(buffer, offset); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  | static void write_fx_4_numeric_variable(struct hlsl_ir_var *var, bool shared, struct fx_write_context *fx) | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  |     uint32_t name_offset, type_offset, value_offset; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     uint32_t semantic_offset, flags = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-02 00:15:48 +02:00
										 |  |  |     if (var->has_explicit_bind_point) | 
					
						
							| 
									
										
										
										
											2025-05-30 10:20:26 +02:00
										 |  |  |         flags |= FX_4_HAS_EXPLICIT_BIND_POINT; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     type_offset = write_type(var->data_type, fx); | 
					
						
							|  |  |  |     name_offset = write_string(var->name, fx); | 
					
						
							| 
									
										
										
										
											2024-05-31 12:26:45 +02:00
										 |  |  |     semantic_offset = write_string(var->semantic.raw_name, fx); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     put_u32(buffer, name_offset); | 
					
						
							|  |  |  |     put_u32(buffer, type_offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     semantic_offset = put_u32(buffer, semantic_offset); /* Semantic */ | 
					
						
							| 
									
										
										
										
											2024-06-02 00:23:12 +02:00
										 |  |  |     put_u32(buffer, var->buffer_offset * 4); /* Offset in the constant buffer, in bytes. */ | 
					
						
							| 
									
										
										
										
											2024-06-14 00:59:57 +02:00
										 |  |  |     value_offset = put_u32(buffer, 0); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     put_u32(buffer, flags); /* Flags */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  |     if (shared) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx->shared_numeric_variable_count++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-06-14 00:59:57 +02:00
										 |  |  |         uint32_t offset = write_fx_4_default_value(var->data_type, var->default_values, fx); | 
					
						
							|  |  |  |         set_u32(buffer, value_offset, offset); | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-15 23:44:08 +02:00
										 |  |  |         write_fx_4_annotations(var->annotations, fx); | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         fx->numeric_variable_count++; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-15 23:44:08 +02:00
										 |  |  | static void write_fx_4_annotation(struct hlsl_ir_var *var, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const struct hlsl_type *type = hlsl_get_multiarray_element_type(var->data_type); | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							|  |  |  |     uint32_t name_offset, type_offset, offset; | 
					
						
							|  |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     name_offset = write_string(var->name, fx); | 
					
						
							|  |  |  |     type_offset = write_type(var->data_type, fx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     put_u32(buffer, name_offset); | 
					
						
							|  |  |  |     put_u32(buffer, type_offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (hlsl_is_numeric_type(type)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         offset = write_fx_4_default_value(var->data_type, var->default_values, fx); | 
					
						
							|  |  |  |         put_u32(buffer, offset); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-08-13 23:59:41 +02:00
										 |  |  |     else if (type->class == HLSL_CLASS_STRING) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         write_fx_4_string_initializer(var, fx); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-06-15 23:44:08 +02:00
										 |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         hlsl_fixme(ctx, &var->loc, "Writing annotations for type class %u is not implemented.", type->class); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | static bool get_fx_4_state_enum_value(const struct rhs_named_value *pairs, | 
					
						
							|  |  |  |         const char *name, unsigned int *value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     while (pairs->name) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (!ascii_strcasecmp(pairs->name, name)) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             *value = pairs->value; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         pairs++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uint32_t write_fx_4_state_numeric_value(struct hlsl_ir_constant *value, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->unstructured; | 
					
						
							|  |  |  |     struct hlsl_type *data_type = value->node.data_type; | 
					
						
							|  |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							|  |  |  |     uint32_t i, type, offset; | 
					
						
							|  |  |  |     unsigned int count = hlsl_type_component_count(data_type); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     offset = put_u32_unaligned(buffer, count); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-10-23 16:19:33 -05:00
										 |  |  |         switch (data_type->e.numeric.type) | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2024-10-23 16:19:33 -05:00
										 |  |  |             case HLSL_TYPE_FLOAT: | 
					
						
							|  |  |  |             case HLSL_TYPE_INT: | 
					
						
							|  |  |  |             case HLSL_TYPE_UINT: | 
					
						
							|  |  |  |             case HLSL_TYPE_BOOL: | 
					
						
							|  |  |  |                 type = fx_4_numeric_base_types[data_type->e.numeric.type]; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 type = 0; | 
					
						
							|  |  |  |                 hlsl_fixme(ctx, &ctx->location, "Unsupported numeric state value type %u.", data_type->e.numeric.type); | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         put_u32_unaligned(buffer, type); | 
					
						
							| 
									
										
										
										
											2025-07-08 19:20:27 +02:00
										 |  |  |         put_u32_unaligned(buffer, get_fx_default_numeric_value(data_type, value->value.u[i].u)); | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return offset; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_fx_4_state_assignment(const struct hlsl_ir_var *var, struct hlsl_state_block_entry *entry, | 
					
						
							|  |  |  |         struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-09-22 11:09:22 +02:00
										 |  |  |     uint32_t value_offset = 0, assignment_type = 0, rhs_offset, type_offset, offset; | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *unstructured = &fx->unstructured; | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							|  |  |  |     struct hlsl_ir_node *value = entry->args->node; | 
					
						
							| 
									
										
										
										
											2024-09-22 11:09:22 +02:00
										 |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							|  |  |  |     struct hlsl_ir_var *index_var; | 
					
						
							|  |  |  |     struct hlsl_ir_constant *c; | 
					
						
							|  |  |  |     struct hlsl_ir_load *load; | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     put_u32(buffer, entry->name_id); | 
					
						
							| 
									
										
										
										
											2024-08-07 18:35:51 +02:00
										 |  |  |     put_u32(buffer, entry->lhs_index); | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |     type_offset = put_u32(buffer, 0); | 
					
						
							|  |  |  |     rhs_offset = put_u32(buffer, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (value->type) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         case HLSL_IR_CONSTANT: | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2024-09-22 11:09:22 +02:00
										 |  |  |             c = hlsl_ir_constant(value); | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             value_offset = write_fx_4_state_numeric_value(c, fx); | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |             assignment_type = FX_4_ASSIGNMENT_CONSTANT; | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |         case HLSL_IR_LOAD: | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2024-09-22 11:09:22 +02:00
										 |  |  |             load = hlsl_ir_load(value); | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-22 11:09:22 +02:00
										 |  |  |             if (load->src.path_len) | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |                 hlsl_fixme(ctx, &var->loc, "Indexed access in RHS values is not implemented."); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-22 11:09:22 +02:00
										 |  |  |             value_offset = write_fx_4_string(load->src.var->name, fx); | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |             assignment_type = FX_4_ASSIGNMENT_VARIABLE; | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-09-22 11:09:22 +02:00
										 |  |  |         case HLSL_IR_INDEX: | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             struct hlsl_ir_index *index = hlsl_ir_index(value); | 
					
						
							|  |  |  |             struct hlsl_ir_node *val = index->val.node; | 
					
						
							|  |  |  |             struct hlsl_ir_node *idx = index->idx.node; | 
					
						
							|  |  |  |             struct hlsl_type *type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (val->type != HLSL_IR_LOAD) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 hlsl_fixme(ctx, &var->loc, "Unexpected indexed RHS value type."); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             load = hlsl_ir_load(val); | 
					
						
							|  |  |  |             value_offset = write_fx_4_string(load->src.var->name, fx); | 
					
						
							|  |  |  |             type = load->src.var->data_type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             switch (idx->type) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 case HLSL_IR_CONSTANT: | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     c = hlsl_ir_constant(idx); | 
					
						
							|  |  |  |                     value_offset = put_u32(unstructured, value_offset); | 
					
						
							|  |  |  |                     put_u32(unstructured, c->value.u[0].u); | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |                     assignment_type = FX_4_ASSIGNMENT_ARRAY_CONSTANT_INDEX; | 
					
						
							| 
									
										
										
										
											2024-09-22 11:09:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     if (c->value.u[0].u >= type->e.array.elements_count) | 
					
						
							|  |  |  |                         hlsl_error(ctx, &var->loc, VKD3D_SHADER_ERROR_HLSL_OFFSET_OUT_OF_BOUNDS, | 
					
						
							|  |  |  |                                 "Array index %u exceeds array size %u.", c->value.u[0].u, type->e.array.elements_count); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 case HLSL_IR_LOAD: | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     load = hlsl_ir_load(idx); | 
					
						
							|  |  |  |                     index_var = load->src.var; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     /* Special case for uint index variables, for anything more complex use an expression. */ | 
					
						
							|  |  |  |                     if (hlsl_types_are_equal(index_var->data_type, hlsl_get_scalar_type(ctx, HLSL_TYPE_UINT)) | 
					
						
							|  |  |  |                             && !load->src.path_len) | 
					
						
							|  |  |  |                     { | 
					
						
							|  |  |  |                         offset = write_fx_4_string(index_var->name, fx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         value_offset = put_u32(unstructured, value_offset); | 
					
						
							|  |  |  |                         put_u32(unstructured, offset); | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |                         assignment_type = FX_4_ASSIGNMENT_ARRAY_VARIABLE_INDEX; | 
					
						
							| 
									
										
										
										
											2024-09-22 11:09:22 +02:00
										 |  |  |                         break; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 /* fall through */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 default: | 
					
						
							|  |  |  |                     hlsl_fixme(ctx, &var->loc, "Complex array index expressions in RHS values are not implemented."); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |         default: | 
					
						
							|  |  |  |             hlsl_fixme(ctx, &var->loc, "Unsupported assignment type for state %s.", entry->name); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     set_u32(buffer, type_offset, assignment_type); | 
					
						
							|  |  |  |     set_u32(buffer, rhs_offset, value_offset); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-07 10:57:46 +02:00
										 |  |  | static bool state_block_contains_state(const struct hlsl_state_block_entry *entry, unsigned int start_index, | 
					
						
							|  |  |  |         struct hlsl_state_block *block) | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     unsigned int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-07 10:57:46 +02:00
										 |  |  |     for (i = start_index; i < block->count; ++i) | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-08-07 10:57:46 +02:00
										 |  |  |         const struct hlsl_state_block_entry *cur = block->entries[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (cur->is_function_call) | 
					
						
							| 
									
										
										
										
											2024-03-19 16:06:12 -03:00
										 |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-07 10:57:46 +02:00
										 |  |  |         if (ascii_strcasecmp(cur->name, entry->name)) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (cur->lhs_has_index != entry->lhs_has_index) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (cur->lhs_has_index && cur->lhs_index != entry->lhs_index) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct replace_state_context | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const struct rhs_named_value *values; | 
					
						
							|  |  |  |     struct hlsl_ir_var *var; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-14 16:32:15 +02:00
										 |  |  | static bool lower_null_constant(struct hlsl_ctx *ctx, struct hlsl_ir_node *instr, void *context) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct hlsl_ir_node *c; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (instr->type != HLSL_IR_CONSTANT) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     if (instr->data_type->class != HLSL_CLASS_NULL) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!(c = hlsl_new_uint_constant(ctx, 0, &instr->loc))) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     list_add_before(&instr->entry, &c->entry); | 
					
						
							|  |  |  |     hlsl_replace_node(instr, c); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | static bool replace_state_block_constant(struct hlsl_ctx *ctx, struct hlsl_ir_node *instr, void *context) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct replace_state_context *replace_context = context; | 
					
						
							|  |  |  |     struct hlsl_ir_stateblock_constant *state_constant; | 
					
						
							|  |  |  |     struct hlsl_ir_node *c; | 
					
						
							|  |  |  |     unsigned int value; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!replace_context->values) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     if (instr->type != HLSL_IR_STATEBLOCK_CONSTANT) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     state_constant = hlsl_ir_stateblock_constant(instr); | 
					
						
							|  |  |  |     if (!get_fx_4_state_enum_value(replace_context->values, state_constant->name, &value)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         hlsl_error(ctx, &replace_context->var->loc, VKD3D_SHADER_ERROR_HLSL_INVALID_SYNTAX, | 
					
						
							|  |  |  |                 "Unrecognized state constant %s.", state_constant->name); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!(c = hlsl_new_uint_constant(ctx, value, &replace_context->var->loc))) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     list_add_before(&state_constant->node.entry, &c->entry); | 
					
						
							|  |  |  |     hlsl_replace_node(&state_constant->node, c); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  | static inline bool is_object_fx_type(enum state_property_component_type type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (type) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-08-07 18:10:55 +02:00
										 |  |  |         case FX_DEPTHSTENCIL: | 
					
						
							|  |  |  |         case FX_RASTERIZER: | 
					
						
							|  |  |  |         case FX_DOMAINSHADER: | 
					
						
							|  |  |  |         case FX_HULLSHADER: | 
					
						
							|  |  |  |         case FX_COMPUTESHADER: | 
					
						
							| 
									
										
										
										
											2024-08-06 14:20:20 +02:00
										 |  |  |         case FX_TEXTURE: | 
					
						
							| 
									
										
										
										
											2024-08-07 18:35:51 +02:00
										 |  |  |         case FX_RENDERTARGETVIEW: | 
					
						
							|  |  |  |         case FX_DEPTHSTENCILVIEW: | 
					
						
							| 
									
										
										
										
											2024-08-14 13:46:34 +02:00
										 |  |  |         case FX_BLEND: | 
					
						
							| 
									
										
										
										
											2024-08-14 14:32:46 +02:00
										 |  |  |         case FX_VERTEXSHADER: | 
					
						
							|  |  |  |         case FX_PIXELSHADER: | 
					
						
							| 
									
										
										
										
											2025-03-26 20:18:41 +01:00
										 |  |  |         case FX_GEOMETRYSHADER: | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |             return true; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline enum hlsl_type_class hlsl_type_class_from_fx_type(enum state_property_component_type type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (type) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-08-07 18:10:55 +02:00
										 |  |  |         case FX_DEPTHSTENCIL: | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |             return HLSL_CLASS_DEPTH_STENCIL_STATE; | 
					
						
							| 
									
										
										
										
											2024-08-07 18:10:55 +02:00
										 |  |  |         case FX_RASTERIZER: | 
					
						
							| 
									
										
										
										
											2024-08-05 17:33:21 +02:00
										 |  |  |             return HLSL_CLASS_RASTERIZER_STATE; | 
					
						
							| 
									
										
										
										
											2024-08-07 18:10:55 +02:00
										 |  |  |         case FX_DOMAINSHADER: | 
					
						
							| 
									
										
										
										
											2024-08-05 20:32:24 +02:00
										 |  |  |             return HLSL_CLASS_DOMAIN_SHADER; | 
					
						
							| 
									
										
										
										
											2024-08-07 18:10:55 +02:00
										 |  |  |         case FX_HULLSHADER: | 
					
						
							| 
									
										
										
										
											2024-08-05 20:32:24 +02:00
										 |  |  |             return HLSL_CLASS_HULL_SHADER; | 
					
						
							| 
									
										
										
										
											2024-08-07 18:10:55 +02:00
										 |  |  |         case FX_COMPUTESHADER: | 
					
						
							| 
									
										
										
										
											2024-08-05 20:32:24 +02:00
										 |  |  |             return HLSL_CLASS_COMPUTE_SHADER; | 
					
						
							| 
									
										
										
										
											2024-08-06 14:20:20 +02:00
										 |  |  |         case FX_TEXTURE: | 
					
						
							|  |  |  |             return HLSL_CLASS_TEXTURE; | 
					
						
							| 
									
										
										
										
											2024-08-07 18:35:51 +02:00
										 |  |  |         case FX_RENDERTARGETVIEW: | 
					
						
							|  |  |  |             return HLSL_CLASS_RENDER_TARGET_VIEW; | 
					
						
							|  |  |  |         case FX_DEPTHSTENCILVIEW: | 
					
						
							|  |  |  |             return HLSL_CLASS_DEPTH_STENCIL_VIEW; | 
					
						
							| 
									
										
										
										
											2024-08-14 13:46:34 +02:00
										 |  |  |         case FX_BLEND: | 
					
						
							|  |  |  |             return HLSL_CLASS_BLEND_STATE; | 
					
						
							| 
									
										
										
										
											2024-08-14 14:32:46 +02:00
										 |  |  |         case FX_VERTEXSHADER: | 
					
						
							|  |  |  |             return HLSL_CLASS_VERTEX_SHADER; | 
					
						
							|  |  |  |         case FX_PIXELSHADER: | 
					
						
							|  |  |  |             return HLSL_CLASS_PIXEL_SHADER; | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |         default: | 
					
						
							|  |  |  |             vkd3d_unreachable(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  | static inline enum hlsl_base_type hlsl_type_from_fx_type(enum state_property_component_type type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (type) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         case FX_BOOL: | 
					
						
							|  |  |  |             return HLSL_TYPE_BOOL; | 
					
						
							|  |  |  |         case FX_FLOAT: | 
					
						
							|  |  |  |             return HLSL_TYPE_FLOAT; | 
					
						
							|  |  |  |         case FX_UINT: | 
					
						
							|  |  |  |         case FX_UINT8: | 
					
						
							|  |  |  |             return HLSL_TYPE_UINT; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             vkd3d_unreachable(); | 
					
						
							|  |  |  |      } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-12 18:50:37 +01:00
										 |  |  | static inline bool hlsl_type_state_compatible(struct hlsl_type *lhs, enum hlsl_base_type rhs) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!hlsl_is_numeric_type(lhs)) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     switch (lhs->e.numeric.type) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         case HLSL_TYPE_INT: | 
					
						
							|  |  |  |         case HLSL_TYPE_UINT: | 
					
						
							|  |  |  |             return rhs == HLSL_TYPE_INT || rhs == HLSL_TYPE_UINT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             return lhs->e.numeric.type == rhs; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_unreachable(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | static const struct rhs_named_value filter_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "MIN_MAG_MIP_POINT", 0x00 }, | 
					
						
							|  |  |  |     { "MIN_MAG_POINT_MIP_LINEAR", 0x01 }, | 
					
						
							|  |  |  |     { "MIN_POINT_MAG_LINEAR_MIP_POINT", 0x04 }, | 
					
						
							|  |  |  |     { "MIN_POINT_MAG_MIP_LINEAR", 0x05 }, | 
					
						
							|  |  |  |     { "MIN_LINEAR_MAG_MIP_POINT", 0x10 }, | 
					
						
							|  |  |  |     { "MIN_LINEAR_MAG_POINT_MIP_LINEAR", 0x11 }, | 
					
						
							|  |  |  |     { "MIN_MAG_LINEAR_MIP_POINT", 0x14 }, | 
					
						
							|  |  |  |     { "MIN_MAG_MIP_LINEAR", 0x15 }, | 
					
						
							|  |  |  |     { "ANISOTROPIC", 0x55 }, | 
					
						
							|  |  |  |     { "COMPARISON_MIN_MAG_MIP_POINT", 0x80 }, | 
					
						
							|  |  |  |     { "COMPARISON_MIN_MAG_POINT_MIP_LINEAR", 0x81 }, | 
					
						
							|  |  |  |     { "COMPARISON_MIN_POINT_MAG_LINEAR_MIP_POINT", 0x84 }, | 
					
						
							|  |  |  |     { "COMPARISON_MIN_POINT_MAG_MIP_LINEAR", 0x85 }, | 
					
						
							|  |  |  |     { "COMPARISON_MIN_LINEAR_MAG_MIP_POINT", 0x90 }, | 
					
						
							|  |  |  |     { "COMPARISON_MIN_LINEAR_MAG_POINT_MIP_LINEAR", 0x91 }, | 
					
						
							|  |  |  |     { "COMPARISON_MIN_MAG_LINEAR_MIP_POINT", 0x94 }, | 
					
						
							|  |  |  |     { "COMPARISON_MIN_MAG_MIP_LINEAR", 0x95 }, | 
					
						
							|  |  |  |     { "COMPARISON_ANISOTROPIC", 0xd5 }, | 
					
						
							|  |  |  |     { NULL }, | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | static const struct rhs_named_value address_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "WRAP", 1 }, | 
					
						
							|  |  |  |     { "MIRROR", 2 }, | 
					
						
							|  |  |  |     { "CLAMP", 3 }, | 
					
						
							|  |  |  |     { "BORDER", 4 }, | 
					
						
							|  |  |  |     { "MIRROR_ONCE", 5 }, | 
					
						
							|  |  |  |     { NULL }, | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | static const struct rhs_named_value compare_func_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "NEVER",         1 }, | 
					
						
							|  |  |  |     { "LESS",          2 }, | 
					
						
							|  |  |  |     { "EQUAL",         3 }, | 
					
						
							|  |  |  |     { "LESS_EQUAL",    4 }, | 
					
						
							|  |  |  |     { "GREATER",       5 }, | 
					
						
							|  |  |  |     { "NOT_EQUAL",     6 }, | 
					
						
							|  |  |  |     { "GREATER_EQUAL", 7 }, | 
					
						
							|  |  |  |     { "ALWAYS",        8 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | static const struct rhs_named_value depth_write_mask_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "ZERO", 0 }, | 
					
						
							|  |  |  |     { "ALL",  1 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | static const struct rhs_named_value comparison_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "NEVER", 1 }, | 
					
						
							|  |  |  |     { "LESS",  2 }, | 
					
						
							|  |  |  |     { "EQUAL", 3 }, | 
					
						
							|  |  |  |     { "LESS_EQUAL", 4 }, | 
					
						
							|  |  |  |     { "GREATER", 5 }, | 
					
						
							|  |  |  |     { "NOT_EQUAL", 6 }, | 
					
						
							|  |  |  |     { "GREATER_EQUAL", 7 }, | 
					
						
							|  |  |  |     { "ALWAYS", 8 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | static const struct rhs_named_value stencil_op_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "KEEP", 1 }, | 
					
						
							|  |  |  |     { "ZERO", 2 }, | 
					
						
							|  |  |  |     { "REPLACE", 3 }, | 
					
						
							|  |  |  |     { "INCR_SAT", 4 }, | 
					
						
							|  |  |  |     { "DECR_SAT", 5 }, | 
					
						
							|  |  |  |     { "INVERT", 6 }, | 
					
						
							|  |  |  |     { "INCR", 7 }, | 
					
						
							|  |  |  |     { "DECR", 8 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2024-06-10 13:18:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | static const struct rhs_named_value fill_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "WIREFRAME", 2 }, | 
					
						
							|  |  |  |     { "SOLID", 3 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2024-06-10 13:18:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | static const struct rhs_named_value cull_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "NONE", 1 }, | 
					
						
							|  |  |  |     { "FRONT", 2 }, | 
					
						
							|  |  |  |     { "BACK", 3 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2024-08-07 15:48:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | static const struct rhs_named_value blend_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "ZERO", 1 }, | 
					
						
							|  |  |  |     { "ONE", 2 }, | 
					
						
							|  |  |  |     { "SRC_COLOR", 3 }, | 
					
						
							|  |  |  |     { "INV_SRC_COLOR", 4 }, | 
					
						
							|  |  |  |     { "SRC_ALPHA", 5 }, | 
					
						
							|  |  |  |     { "INV_SRC_ALPHA", 6 }, | 
					
						
							|  |  |  |     { "DEST_ALPHA", 7 }, | 
					
						
							|  |  |  |     { "INV_DEST_ALPHA", 8 }, | 
					
						
							|  |  |  |     { "DEST_COLOR", 9 }, | 
					
						
							|  |  |  |     { "INV_DEST_COLOR", 10 }, | 
					
						
							|  |  |  |     { "SRC_ALPHA_SAT", 11 }, | 
					
						
							|  |  |  |     { "BLEND_FACTOR", 14 }, | 
					
						
							|  |  |  |     { "INV_BLEND_FACTOR", 15 }, | 
					
						
							|  |  |  |     { "SRC1_COLOR", 16 }, | 
					
						
							|  |  |  |     { "INV_SRC1_COLOR", 17 }, | 
					
						
							|  |  |  |     { "SRC1_ALPHA", 18 }, | 
					
						
							|  |  |  |     { "INV_SRC1_ALPHA", 19 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2024-08-07 15:48:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | static const struct rhs_named_value blendop_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "ADD", 1 }, | 
					
						
							|  |  |  |     { "SUBTRACT", 2 }, | 
					
						
							|  |  |  |     { "REV_SUBTRACT", 3 }, | 
					
						
							|  |  |  |     { "MIN", 4 }, | 
					
						
							|  |  |  |     { "MAX", 5 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2024-08-09 00:27:01 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | static const struct rhs_named_value bool_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "FALSE", 0 }, | 
					
						
							|  |  |  |     { "TRUE", 1 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2024-08-14 17:07:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | static const struct rhs_named_value null_values[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     { "NULL", 0 }, | 
					
						
							|  |  |  |     { NULL } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2024-08-07 15:48:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 08:36:17 +02:00
										 |  |  | static const struct fx_state fx_4_states[] = | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     { "RasterizerState",       HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_RASTERIZER,       1, 1, 0 }, | 
					
						
							|  |  |  |     { "DepthStencilState",     HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_DEPTHSTENCIL,     1, 1, 1 }, | 
					
						
							|  |  |  |     { "BlendState",            HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_BLEND,            1, 1, 2 }, | 
					
						
							|  |  |  |     { "RenderTargetView",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_RENDERTARGETVIEW, 1, 8, 3 }, | 
					
						
							|  |  |  |     { "DepthStencilView",      HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_DEPTHSTENCILVIEW, 1, 1, 4 }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "VertexShader",          HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_VERTEXSHADER,     1, 1, 6 }, | 
					
						
							|  |  |  |     { "PixelShader",           HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_PIXELSHADER,      1, 1, 7 }, | 
					
						
							| 
									
										
										
										
											2024-11-22 02:17:25 +01:00
										 |  |  |     { "GeometryShader",        HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_GEOMETRYSHADER,   1, 1, 8 }, | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |     { "DS_StencilRef",         HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 9 }, | 
					
						
							|  |  |  |     { "AB_BlendFactor",        HLSL_CLASS_PASS, HLSL_CLASS_VECTOR, FX_FLOAT, 4, 1, 10 }, | 
					
						
							|  |  |  |     { "AB_SampleMask",         HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 11 }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "FillMode",              HLSL_CLASS_RASTERIZER_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 12, fill_values }, | 
					
						
							|  |  |  |     { "CullMode",              HLSL_CLASS_RASTERIZER_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 13, cull_values }, | 
					
						
							|  |  |  |     { "FrontCounterClockwise", HLSL_CLASS_RASTERIZER_STATE, HLSL_CLASS_SCALAR, FX_BOOL,  1, 1, 14, bool_values }, | 
					
						
							|  |  |  |     { "DepthBias",             HLSL_CLASS_RASTERIZER_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 15 }, | 
					
						
							|  |  |  |     { "DepthBiasClamp",        HLSL_CLASS_RASTERIZER_STATE, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 16 }, | 
					
						
							|  |  |  |     { "SlopeScaledDepthBias",  HLSL_CLASS_RASTERIZER_STATE, HLSL_CLASS_SCALAR, FX_FLOAT, 1, 1, 17 }, | 
					
						
							|  |  |  |     { "DepthClipEnable",       HLSL_CLASS_RASTERIZER_STATE, HLSL_CLASS_SCALAR, FX_BOOL,  1, 1, 18, bool_values }, | 
					
						
							|  |  |  |     { "ScissorEnable",         HLSL_CLASS_RASTERIZER_STATE, HLSL_CLASS_SCALAR, FX_BOOL,  1, 1, 19, bool_values }, | 
					
						
							|  |  |  |     { "MultisampleEnable",     HLSL_CLASS_RASTERIZER_STATE, HLSL_CLASS_SCALAR, FX_BOOL,  1, 1, 20, bool_values }, | 
					
						
							|  |  |  |     { "AntializedLineEnable",  HLSL_CLASS_RASTERIZER_STATE, HLSL_CLASS_SCALAR, FX_BOOL,  1, 1, 21, bool_values }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "DepthEnable",               HLSL_CLASS_DEPTH_STENCIL_STATE, HLSL_CLASS_SCALAR, FX_BOOL,  1, 1, 22, bool_values }, | 
					
						
							|  |  |  |     { "DepthWriteMask",            HLSL_CLASS_DEPTH_STENCIL_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 23, depth_write_mask_values }, | 
					
						
							|  |  |  |     { "DepthFunc",                 HLSL_CLASS_DEPTH_STENCIL_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 24, comparison_values }, | 
					
						
							|  |  |  |     { "StencilEnable",             HLSL_CLASS_DEPTH_STENCIL_STATE, HLSL_CLASS_SCALAR, FX_BOOL,  1, 1, 25, bool_values }, | 
					
						
							|  |  |  |     { "StencilReadMask",           HLSL_CLASS_DEPTH_STENCIL_STATE, HLSL_CLASS_SCALAR, FX_UINT8, 1, 1, 26 }, | 
					
						
							|  |  |  |     { "StencilWriteMask",          HLSL_CLASS_DEPTH_STENCIL_STATE, HLSL_CLASS_SCALAR, FX_UINT8, 1, 1, 27 }, | 
					
						
							|  |  |  |     { "FrontFaceStencilFail",      HLSL_CLASS_DEPTH_STENCIL_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 28, stencil_op_values }, | 
					
						
							|  |  |  |     { "FrontFaceStencilDepthFail", HLSL_CLASS_DEPTH_STENCIL_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 29, stencil_op_values }, | 
					
						
							|  |  |  |     { "FrontFaceStencilPass",      HLSL_CLASS_DEPTH_STENCIL_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 30, stencil_op_values }, | 
					
						
							|  |  |  |     { "FrontFaceStencilFunc",      HLSL_CLASS_DEPTH_STENCIL_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 31, comparison_values }, | 
					
						
							|  |  |  |     { "BackFaceStencilFail",       HLSL_CLASS_DEPTH_STENCIL_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 32, stencil_op_values }, | 
					
						
							|  |  |  |     { "BackFaceStencilDepthFail",  HLSL_CLASS_DEPTH_STENCIL_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 33, stencil_op_values }, | 
					
						
							|  |  |  |     { "BackFaceStencilPass",       HLSL_CLASS_DEPTH_STENCIL_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 34, stencil_op_values }, | 
					
						
							|  |  |  |     { "BackFaceStencilFunc",       HLSL_CLASS_DEPTH_STENCIL_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 35, comparison_values }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "AlphaToCoverageEnable", HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_BOOL,  1, 1, 36, bool_values }, | 
					
						
							|  |  |  |     { "BlendEnable",           HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_BOOL,  1, 8, 37, bool_values }, | 
					
						
							|  |  |  |     { "SrcBlend",              HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 38, blend_values }, | 
					
						
							|  |  |  |     { "DestBlend",             HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 39, blend_values }, | 
					
						
							|  |  |  |     { "BlendOp",               HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 40, blendop_values }, | 
					
						
							|  |  |  |     { "SrcBlendAlpha",         HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 41, blend_values }, | 
					
						
							|  |  |  |     { "DestBlendAlpha",        HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 42, blend_values }, | 
					
						
							|  |  |  |     { "BlendOpAlpha",          HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 1, 43, blendop_values }, | 
					
						
							|  |  |  |     { "RenderTargetWriteMask", HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_UINT8, 1, 8, 44 }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "Filter",         HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR,  FX_UINT,    1, 1, 45, filter_values }, | 
					
						
							|  |  |  |     { "AddressU",       HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR,  FX_UINT,    1, 1, 46, address_values }, | 
					
						
							|  |  |  |     { "AddressV",       HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR,  FX_UINT,    1, 1, 47, address_values }, | 
					
						
							|  |  |  |     { "AddressW",       HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR,  FX_UINT,    1, 1, 48, address_values }, | 
					
						
							|  |  |  |     { "MipLODBias",     HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR,  FX_FLOAT,   1, 1, 49 }, | 
					
						
							|  |  |  |     { "MaxAnisotropy",  HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR,  FX_UINT,    1, 1, 50 }, | 
					
						
							|  |  |  |     { "ComparisonFunc", HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR,  FX_UINT,    1, 1, 51, compare_func_values }, | 
					
						
							|  |  |  |     { "BorderColor",    HLSL_CLASS_SAMPLER, HLSL_CLASS_VECTOR,  FX_FLOAT,   4, 1, 52 }, | 
					
						
							|  |  |  |     { "MinLOD",         HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR,  FX_FLOAT,   1, 1, 53 }, | 
					
						
							|  |  |  |     { "MaxLOD",         HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR,  FX_FLOAT,   1, 1, 54 }, | 
					
						
							|  |  |  |     { "Texture",        HLSL_CLASS_SAMPLER, HLSL_CLASS_SCALAR,  FX_TEXTURE, 1, 1, 55, null_values }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     { "HullShader",     HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_HULLSHADER,    1, 1, 56 }, | 
					
						
							|  |  |  |     { "DomainShader",   HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_DOMAINSHADER,  1, 1, 57 }, | 
					
						
							|  |  |  |     { "ComputeShader",  HLSL_CLASS_PASS, HLSL_CLASS_SCALAR, FX_COMPUTESHADER, 1, 1, 58 }, | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2024-08-07 15:48:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 08:36:17 +02:00
										 |  |  | static const struct fx_state fx_5_blend_states[] = | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-05-06 01:07:03 +02:00
										 |  |  |     { "AlphaToCoverageEnable", HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_BOOL,  1, 1, 36, bool_values }, | 
					
						
							|  |  |  |     { "BlendEnable",           HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_BOOL,  1, 8, 37, bool_values }, | 
					
						
							|  |  |  |     { "SrcBlend",              HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 8, 38, blend_values }, | 
					
						
							|  |  |  |     { "DestBlend",             HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 8, 39, blend_values }, | 
					
						
							|  |  |  |     { "BlendOp",               HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 8, 40, blendop_values }, | 
					
						
							|  |  |  |     { "SrcBlendAlpha",         HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 8, 41, blend_values }, | 
					
						
							|  |  |  |     { "DestBlendAlpha",        HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 8, 42, blend_values }, | 
					
						
							|  |  |  |     { "BlendOpAlpha",          HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_UINT,  1, 8, 43, blendop_values }, | 
					
						
							|  |  |  |     { "RenderTargetWriteMask", HLSL_CLASS_BLEND_STATE, HLSL_CLASS_SCALAR, FX_UINT8, 1, 8, 44 }, | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2024-08-07 15:48:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 08:36:17 +02:00
										 |  |  | struct fx_state_table | 
					
						
							| 
									
										
										
										
											2025-05-06 01:07:03 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-06-16 08:36:17 +02:00
										 |  |  |     const struct fx_state *ptr; | 
					
						
							| 
									
										
										
										
											2025-05-06 01:07:03 +02:00
										 |  |  |     unsigned int count; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2024-08-07 15:48:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 10:52:43 +02:00
										 |  |  | static struct fx_state_table fx_get_state_table(enum hlsl_type_class type_class, | 
					
						
							| 
									
										
										
										
											2025-05-06 01:07:03 +02:00
										 |  |  |         unsigned int major, unsigned int minor) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-06-16 08:36:17 +02:00
										 |  |  |     struct fx_state_table table; | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 10:52:43 +02:00
										 |  |  |     if (major == 2) | 
					
						
							| 
									
										
										
										
											2024-08-07 15:48:57 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-06-16 10:52:43 +02:00
										 |  |  |         if (type_class == HLSL_CLASS_PASS) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             table.ptr = fx_2_pass_states; | 
					
						
							|  |  |  |             table.count = ARRAY_SIZE(fx_2_pass_states); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             table.ptr = fx_2_sampler_states; | 
					
						
							|  |  |  |             table.count = ARRAY_SIZE(fx_2_sampler_states); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-08-07 15:48:57 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-06-16 10:52:43 +02:00
										 |  |  |         if (type_class == HLSL_CLASS_BLEND_STATE && (major == 5 || (major == 4 && minor == 1))) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             table.ptr = fx_5_blend_states; | 
					
						
							|  |  |  |             table.count = ARRAY_SIZE(fx_5_blend_states); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             table.ptr = fx_4_states; | 
					
						
							|  |  |  |             table.count = ARRAY_SIZE(fx_4_states); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-08-07 15:48:57 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-06 01:07:03 +02:00
										 |  |  |     return table; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 10:52:43 +02:00
										 |  |  | static void resolve_fx_state_block_values(struct hlsl_ir_var *var, | 
					
						
							| 
									
										
										
										
											2025-05-06 01:07:03 +02:00
										 |  |  |         struct hlsl_state_block_entry *entry, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const struct hlsl_type *type = hlsl_get_multiarray_element_type(var->data_type); | 
					
						
							|  |  |  |     struct replace_state_context replace_context; | 
					
						
							| 
									
										
										
										
											2025-06-16 08:36:17 +02:00
										 |  |  |     const struct fx_state *state = NULL; | 
					
						
							| 
									
										
										
										
											2025-05-06 01:07:03 +02:00
										 |  |  |     struct hlsl_type *state_type = NULL; | 
					
						
							|  |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							|  |  |  |     enum hlsl_base_type base_type; | 
					
						
							| 
									
										
										
										
											2025-06-16 08:36:17 +02:00
										 |  |  |     struct fx_state_table table; | 
					
						
							| 
									
										
										
										
											2025-05-06 01:07:03 +02:00
										 |  |  |     struct hlsl_ir_node *node; | 
					
						
							|  |  |  |     unsigned int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 10:52:43 +02:00
										 |  |  |     table = fx_get_state_table(type->class, ctx->profile->major_version, ctx->profile->minor_version); | 
					
						
							| 
									
										
										
										
											2025-05-06 01:07:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-07 15:48:57 +02:00
										 |  |  |     for (i = 0; i < table.count; ++i) | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-08-07 15:48:57 +02:00
										 |  |  |         if (type->class == table.ptr[i].container | 
					
						
							|  |  |  |                 && !ascii_strcasecmp(entry->name, table.ptr[i].name)) | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2024-08-07 15:48:57 +02:00
										 |  |  |             state = &table.ptr[i]; | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!state) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         hlsl_error(ctx, &var->loc, VKD3D_SHADER_ERROR_HLSL_INVALID_SYNTAX, "Unrecognized state name %s.", entry->name); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (entry->args_count != 1) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         hlsl_error(ctx, &var->loc, VKD3D_SHADER_ERROR_HLSL_INVALID_SYNTAX, "Unrecognized initializer for the state %s.", | 
					
						
							|  |  |  |                 entry->name); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-07 18:35:51 +02:00
										 |  |  |     if (entry->lhs_has_index && state->array_size == 1) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         hlsl_error(ctx, &var->loc, VKD3D_SHADER_ERROR_HLSL_INVALID_SYNTAX, "Can't use array-style access for non-array state %s.", | 
					
						
							|  |  |  |                 entry->name); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!entry->lhs_has_index && state->array_size > 1) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         hlsl_error(ctx, &var->loc, VKD3D_SHADER_ERROR_HLSL_INVALID_SYNTAX, "Expected array index for array state %s.", | 
					
						
							|  |  |  |                 entry->name); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (entry->lhs_has_index && (state->array_size <= entry->lhs_index)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         hlsl_error(ctx, &var->loc, VKD3D_SHADER_ERROR_HLSL_INVALID_SYNTAX, "Invalid element index %u for the state %s[%u].", | 
					
						
							|  |  |  |                 entry->lhs_index, state->name, state->array_size); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |     entry->name_id = state->id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     replace_context.values = state->values; | 
					
						
							|  |  |  |     replace_context.var = var; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-22 02:05:18 +02:00
										 |  |  |     /* Turn named constants to actual constants. */ | 
					
						
							| 
									
										
										
										
											2024-08-14 16:32:15 +02:00
										 |  |  |     hlsl_transform_ir(ctx, lower_null_constant, entry->instrs, NULL); | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |     hlsl_transform_ir(ctx, replace_state_block_constant, entry->instrs, &replace_context); | 
					
						
							| 
									
										
										
										
											2024-08-07 14:56:15 +02:00
										 |  |  |     hlsl_run_const_passes(ctx, entry->instrs); | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-22 02:05:18 +02:00
										 |  |  |     /* Now cast and run folding again. */ | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |     if (is_object_fx_type(state->type)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         node = entry->args->node; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         switch (node->type) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             case HLSL_IR_LOAD: | 
					
						
							| 
									
										
										
										
											2024-08-14 16:32:15 +02:00
										 |  |  |             { | 
					
						
							|  |  |  |                 struct hlsl_ir_load *load = hlsl_ir_load(node); | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 if (load->src.path_len) | 
					
						
							|  |  |  |                     hlsl_fixme(ctx, &ctx->location, "Arrays are not supported for RHS."); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (load->src.var->data_type->class != hlsl_type_class_from_fx_type(state->type)) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     hlsl_error(ctx, &ctx->location, VKD3D_SHADER_ERROR_HLSL_INVALID_SYNTAX, "Type mismatch for the %s state value", | 
					
						
							|  |  |  |                             entry->name); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2024-08-14 16:32:15 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |             case HLSL_IR_CONSTANT: | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 struct hlsl_ir_constant *c = hlsl_ir_constant(node); | 
					
						
							|  |  |  |                 struct hlsl_type *data_type = c->node.data_type; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-26 20:58:42 +01:00
										 |  |  |                 if (data_type->class == HLSL_CLASS_SCALAR | 
					
						
							|  |  |  |                         && (data_type->e.numeric.type == HLSL_TYPE_INT || data_type->e.numeric.type == HLSL_TYPE_UINT)) | 
					
						
							| 
									
										
										
										
											2024-08-14 16:32:15 +02:00
										 |  |  |                 { | 
					
						
							|  |  |  |                     if (c->value.u[0].u != 0) | 
					
						
							|  |  |  |                         hlsl_error(ctx, &ctx->location, VKD3D_SHADER_ERROR_HLSL_INVALID_SYNTAX, | 
					
						
							|  |  |  |                                 "Only 0 integer constants are allowed for object-typed fields."); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     hlsl_error(ctx, &ctx->location, VKD3D_SHADER_ERROR_HLSL_INVALID_SYNTAX, | 
					
						
							|  |  |  |                             "Unexpected constant used for object-typed field."); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |             default: | 
					
						
							|  |  |  |                 hlsl_fixme(ctx, &ctx->location, "Unhandled node type for object-typed field."); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  |     base_type = hlsl_type_from_fx_type(state->type); | 
					
						
							| 
									
										
										
										
											2024-04-22 02:05:18 +02:00
										 |  |  |     switch (state->class) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         case HLSL_CLASS_VECTOR: | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  |             state_type = hlsl_get_vector_type(ctx, base_type, state->dimx); | 
					
						
							| 
									
										
										
										
											2024-04-22 02:05:18 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |         case HLSL_CLASS_SCALAR: | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  |             state_type = hlsl_get_scalar_type(ctx, base_type); | 
					
						
							| 
									
										
										
										
											2024-04-22 02:05:18 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |         case HLSL_CLASS_TEXTURE: | 
					
						
							|  |  |  |             hlsl_fixme(ctx, &ctx->location, "Object type fields are not supported."); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             ; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-22 02:05:18 +02:00
										 |  |  |     if (state_type) | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-04-22 02:05:18 +02:00
										 |  |  |         node = entry->args->node; | 
					
						
							| 
									
										
										
										
											2025-03-12 18:50:37 +01:00
										 |  |  |         if (state->type == FX_UINT8 || !hlsl_type_state_compatible(node->data_type, base_type)) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             struct hlsl_ir_node *cast; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!(cast = hlsl_new_cast(ctx, node, state_type, &var->loc))) | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |             list_add_after(&node->entry, &cast->entry); | 
					
						
							|  |  |  |             node = cast; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-04-22 02:05:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  |         /* FX_UINT8 values are using 32-bits in the binary. Mask higher 24 bits for those. */ | 
					
						
							|  |  |  |         if (state->type == FX_UINT8) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             struct hlsl_ir_node *mask; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!(mask = hlsl_new_uint_constant(ctx, 0xff, &var->loc))) | 
					
						
							|  |  |  |                 return; | 
					
						
							| 
									
										
										
										
											2025-03-12 18:50:37 +01:00
										 |  |  |             list_add_after(&node->entry, &mask->entry); | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-12 18:50:37 +01:00
										 |  |  |             if (!(node = hlsl_new_binary_expr(ctx, HLSL_OP2_BIT_AND, node, mask))) | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  |                 return; | 
					
						
							| 
									
										
										
										
											2025-03-12 18:50:37 +01:00
										 |  |  |             list_add_after(&mask->entry, &node->entry); | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-12 18:50:37 +01:00
										 |  |  |         if (node != entry->args->node) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             hlsl_src_remove(entry->args); | 
					
						
							|  |  |  |             hlsl_src_from_node(entry->args, node); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-04-22 02:05:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-07 14:56:15 +02:00
										 |  |  |         hlsl_run_const_passes(ctx, entry->instrs); | 
					
						
							| 
									
										
										
										
											2024-04-22 02:05:18 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-11 14:21:25 +02:00
										 |  |  | static bool decompose_fx_4_state_add_entries(struct hlsl_state_block *block, unsigned int entry_index, | 
					
						
							|  |  |  |         unsigned int count) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!vkd3d_array_reserve((void **)&block->entries, &block->capacity, block->count + count, sizeof(*block->entries))) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (entry_index != block->count - 1) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         memmove(&block->entries[entry_index + count + 1], &block->entries[entry_index + 1], | 
					
						
							|  |  |  |                 (block->count - entry_index - 1) * sizeof(*block->entries)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     block->count += count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static unsigned int decompose_fx_4_state_function_call(struct hlsl_ir_var *var, struct hlsl_state_block *block, | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |         unsigned int entry_index, struct fx_write_context *fx) | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |     struct hlsl_state_block_entry *entry = block->entries[entry_index]; | 
					
						
							|  |  |  |     const struct state_block_function_info *info; | 
					
						
							|  |  |  |     struct function_component components[9]; | 
					
						
							|  |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							|  |  |  |     unsigned int i; | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |     if (!entry->is_function_call) | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!(info = get_state_block_function_info(entry->name))) | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-05 20:32:24 +02:00
										 |  |  |     if (info->min_profile > ctx->profile->major_version) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         hlsl_error(ctx, &var->loc, VKD3D_SHADER_ERROR_HLSL_INVALID_STATE_BLOCK_ENTRY, | 
					
						
							|  |  |  |                 "State %s is not supported for this profile.", entry->name); | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |     /* For single argument case simply replace the name. */ | 
					
						
							|  |  |  |     if (info->min_args == info->max_args && info->min_args == 1) | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |         vkd3d_free(entry->name); | 
					
						
							|  |  |  |         entry->name = hlsl_strdup(ctx, info->components[0].name); | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-11 14:21:25 +02:00
										 |  |  |     if (!decompose_fx_4_state_add_entries(block, entry_index, entry->args_count - 1)) | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |     get_state_block_function_components(info, components, entry->args_count); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < entry->args_count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         const struct function_component *comp = &components[i]; | 
					
						
							|  |  |  |         unsigned int arg_index = (i + 1) % entry->args_count; | 
					
						
							|  |  |  |         block->entries[entry_index + i] = clone_stateblock_entry(ctx, entry, comp->name, | 
					
						
							| 
									
										
										
										
											2024-08-26 16:43:20 -04:00
										 |  |  |                 comp->lhs_has_index, comp->lhs_index, true, arg_index); | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     hlsl_free_state_block_entry(entry); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-05 19:27:32 +02:00
										 |  |  |     return entry->args_count; | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-11 14:21:25 +02:00
										 |  |  | /* For some states assignment sets all of the elements. This behaviour is limited to certain states of BlendState
 | 
					
						
							| 
									
										
										
										
											2024-09-01 23:46:09 +02:00
										 |  |  |    object, and only when fx_4_1 or fx_5_0 profile is used. */ | 
					
						
							| 
									
										
										
										
											2024-08-11 14:21:25 +02:00
										 |  |  | static unsigned int decompose_fx_4_state_block_expand_array(struct hlsl_ir_var *var, struct hlsl_state_block *block, | 
					
						
							|  |  |  |         unsigned int entry_index, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-03-03 19:17:23 +01:00
										 |  |  |     static const char *const states[] = | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         "SrcBlend", | 
					
						
							|  |  |  |         "DestBlend", | 
					
						
							|  |  |  |         "BlendOp", | 
					
						
							|  |  |  |         "SrcBlendAlpha", | 
					
						
							|  |  |  |         "DestBlendAlpha", | 
					
						
							|  |  |  |         "BlendOpAlpha", | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2024-08-11 14:21:25 +02:00
										 |  |  |     const struct hlsl_type *type = hlsl_get_multiarray_element_type(var->data_type); | 
					
						
							|  |  |  |     struct hlsl_state_block_entry *entry = block->entries[entry_index]; | 
					
						
							|  |  |  |     static const unsigned int array_size = 8; | 
					
						
							|  |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							|  |  |  |     bool found = false; | 
					
						
							|  |  |  |     unsigned int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (type->class != HLSL_CLASS_BLEND_STATE) | 
					
						
							|  |  |  |         return 1; | 
					
						
							| 
									
										
										
										
											2024-09-01 23:46:09 +02:00
										 |  |  |     if (hlsl_version_lt(ctx, 4, 1)) | 
					
						
							| 
									
										
										
										
											2024-08-11 14:21:25 +02:00
										 |  |  |         return 1; | 
					
						
							|  |  |  |     if (entry->lhs_has_index) | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < ARRAY_SIZE(states); ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (!ascii_strcasecmp(entry->name, states[i])) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             found = true; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!found) | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!decompose_fx_4_state_add_entries(block, entry_index, array_size - 1)) | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     block->entries[entry_index]->lhs_has_index = true; | 
					
						
							|  |  |  |     for (i = 1; i < array_size; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         block->entries[entry_index + i] = clone_stateblock_entry(ctx, entry, | 
					
						
							| 
									
										
										
										
											2024-08-26 16:43:20 -04:00
										 |  |  |                 entry->name, true, i, true, 0); | 
					
						
							| 
									
										
										
										
											2024-08-11 14:21:25 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return array_size; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static unsigned int decompose_fx_4_state_block(struct hlsl_ir_var *var, struct hlsl_state_block *block, | 
					
						
							|  |  |  |         unsigned int entry_index, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct hlsl_state_block_entry *entry = block->entries[entry_index]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (entry->is_function_call) | 
					
						
							|  |  |  |         return decompose_fx_4_state_function_call(var, block, entry_index, fx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return decompose_fx_4_state_block_expand_array(var, block, entry_index, fx); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 16:40:37 +02:00
										 |  |  | static void fx_4_decompose_state_blocks(struct hlsl_ir_var *var, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     unsigned int block_count = hlsl_get_multiarray_size(var->data_type); | 
					
						
							|  |  |  |     struct hlsl_state_block *block; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!var->state_blocks) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (unsigned int i = 0; i < block_count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         block = var->state_blocks[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (unsigned int j = 0; j < block->count;) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             j += decompose_fx_4_state_block(var, block, j, fx); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 23:35:08 +02:00
										 |  |  | static uint32_t write_state_block(struct hlsl_ir_var *var, unsigned int block_index, | 
					
						
							| 
									
										
										
										
											2025-06-16 23:26:35 +02:00
										 |  |  |         struct fx_write_context *fx) | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     struct hlsl_state_block *block; | 
					
						
							|  |  |  |     uint32_t i, count = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (var->state_blocks) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         block = var->state_blocks[block_index]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (i = 0; i < block->count; ++i) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             struct hlsl_state_block_entry *entry = block->entries[i]; | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |             /* Skip if property is reassigned later. This will use the last assignment. */ | 
					
						
							| 
									
										
										
										
											2024-08-07 10:57:46 +02:00
										 |  |  |             if (state_block_contains_state(entry, i + 1, block)) | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |             /* Resolve special constant names and property names. */ | 
					
						
							| 
									
										
										
										
											2025-06-16 10:52:43 +02:00
										 |  |  |             resolve_fx_state_block_values(var, entry, fx); | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 23:35:08 +02:00
										 |  |  |             write_state_assignment(var, entry, fx); | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |             ++count; | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 23:26:35 +02:00
										 |  |  |     return count; | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_fx_4_state_object_initializer(struct hlsl_ir_var *var, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t elements_count = hlsl_get_multiarray_size(var->data_type), i; | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							| 
									
										
										
										
											2025-06-16 23:26:35 +02:00
										 |  |  |     uint32_t count_offset, count; | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 16:40:37 +02:00
										 |  |  |     fx_4_decompose_state_blocks(var, fx); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-20 11:10:06 +02:00
										 |  |  |     for (i = 0; i < elements_count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         count_offset = put_u32(buffer, 0); | 
					
						
							| 
									
										
										
										
											2025-06-16 23:35:08 +02:00
										 |  |  |         count = write_state_block(var, i, fx); | 
					
						
							| 
									
										
										
										
											2025-06-16 23:26:35 +02:00
										 |  |  |         set_u32(buffer, count_offset, count); | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-05 20:32:24 +02:00
										 |  |  | static void write_fx_4_shader_initializer(struct hlsl_ir_var *var, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							|  |  |  |     uint32_t elements_count = hlsl_get_multiarray_size(var->data_type); | 
					
						
							|  |  |  |     unsigned int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* FIXME: write shader blobs, once parser support works. */ | 
					
						
							|  |  |  |     for (i = 0; i < elements_count; ++i) | 
					
						
							|  |  |  |         put_u32(buffer, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_fx_5_shader_initializer(struct hlsl_ir_var *var, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							|  |  |  |     uint32_t elements_count = hlsl_get_multiarray_size(var->data_type); | 
					
						
							|  |  |  |     unsigned int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* FIXME: write shader blobs, once parser support works. */ | 
					
						
							|  |  |  |     for (i = 0; i < elements_count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         put_u32(buffer, 0); /* Blob offset */ | 
					
						
							|  |  |  |         put_u32(buffer, 0); /* SODecl[0] offset */ | 
					
						
							|  |  |  |         put_u32(buffer, 0); /* SODecl[1] offset */ | 
					
						
							|  |  |  |         put_u32(buffer, 0); /* SODecl[2] offset */ | 
					
						
							|  |  |  |         put_u32(buffer, 0); /* SODecl[3] offset */ | 
					
						
							|  |  |  |         put_u32(buffer, 0); /* SODecl count */ | 
					
						
							|  |  |  |         put_u32(buffer, 0); /* Rasterizer stream */ | 
					
						
							|  |  |  |         put_u32(buffer, 0); /* Interface bindings count */ | 
					
						
							|  |  |  |         put_u32(buffer, 0); /* Interface initializer offset */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-09 22:42:23 +01:00
										 |  |  | static void write_fx_4_object_variable(struct hlsl_ir_var *var, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-02-24 00:39:37 +01:00
										 |  |  |     const struct hlsl_type *type = hlsl_get_multiarray_element_type(var->data_type); | 
					
						
							|  |  |  |     uint32_t elements_count = hlsl_get_multiarray_size(var->data_type); | 
					
						
							| 
									
										
										
										
											2024-02-09 22:42:23 +01:00
										 |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							|  |  |  |     uint32_t semantic_offset, bind_point = ~0u; | 
					
						
							| 
									
										
										
										
											2024-08-05 20:32:24 +02:00
										 |  |  |     uint32_t name_offset, type_offset; | 
					
						
							| 
									
										
										
										
											2024-02-24 00:39:37 +01:00
										 |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							| 
									
										
										
										
											2024-02-09 22:42:23 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (var->reg_reservation.reg_type) | 
					
						
							|  |  |  |         bind_point = var->reg_reservation.reg_index; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     type_offset = write_type(var->data_type, fx); | 
					
						
							|  |  |  |     name_offset = write_string(var->name, fx); | 
					
						
							| 
									
										
										
										
											2024-05-31 12:26:45 +02:00
										 |  |  |     semantic_offset = write_string(var->semantic.raw_name, fx); | 
					
						
							| 
									
										
										
										
											2024-02-09 22:42:23 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     put_u32(buffer, name_offset); | 
					
						
							|  |  |  |     put_u32(buffer, type_offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     semantic_offset = put_u32(buffer, semantic_offset); /* Semantic */ | 
					
						
							|  |  |  |     put_u32(buffer, bind_point); /* Explicit bind point */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-25 01:08:46 +01:00
										 |  |  |     if (fx->child_effect && var->storage_modifiers & HLSL_STORAGE_SHARED) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         ++fx->shared_object_count; | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-24 00:39:37 +01:00
										 |  |  |     /* Initializer */ | 
					
						
							| 
									
										
										
										
											2024-02-05 20:25:57 -06:00
										 |  |  |     switch (type->class) | 
					
						
							| 
									
										
										
										
											2024-02-24 00:39:37 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-02-27 13:34:52 -06:00
										 |  |  |         case HLSL_CLASS_RENDER_TARGET_VIEW: | 
					
						
							| 
									
										
										
										
											2024-04-20 16:19:54 +02:00
										 |  |  |             fx->rtv_count += elements_count; | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2024-02-05 20:25:57 -06:00
										 |  |  |         case HLSL_CLASS_TEXTURE: | 
					
						
							| 
									
										
										
										
											2024-04-20 16:28:34 +02:00
										 |  |  |             fx->texture_count += elements_count; | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2024-02-05 20:32:37 -06:00
										 |  |  |         case HLSL_CLASS_UAV: | 
					
						
							| 
									
										
										
										
											2024-04-19 11:40:14 +02:00
										 |  |  |             fx->uav_count += elements_count; | 
					
						
							| 
									
										
										
										
											2024-02-24 00:39:37 +01:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2024-02-05 20:25:57 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 19:41:15 -06:00
										 |  |  |         case HLSL_CLASS_PIXEL_SHADER: | 
					
						
							| 
									
										
										
										
											2024-02-06 17:33:26 -06:00
										 |  |  |         case HLSL_CLASS_VERTEX_SHADER: | 
					
						
							| 
									
										
										
										
											2024-08-05 20:32:24 +02:00
										 |  |  |             write_fx_4_shader_initializer(var, fx); | 
					
						
							|  |  |  |             fx->shader_count += elements_count; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case HLSL_CLASS_HULL_SHADER: | 
					
						
							|  |  |  |         case HLSL_CLASS_COMPUTE_SHADER: | 
					
						
							|  |  |  |         case HLSL_CLASS_DOMAIN_SHADER: | 
					
						
							|  |  |  |             write_fx_5_shader_initializer(var, fx); | 
					
						
							| 
									
										
										
										
											2024-04-20 16:30:05 +02:00
										 |  |  |             fx->shader_count += elements_count; | 
					
						
							| 
									
										
										
										
											2024-02-06 17:33:26 -06:00
										 |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-06 23:29:15 +02:00
										 |  |  |         case HLSL_CLASS_DEPTH_STENCIL_VIEW: | 
					
						
							|  |  |  |             fx->dsv_count += elements_count; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  |         case HLSL_CLASS_DEPTH_STENCIL_STATE: | 
					
						
							|  |  |  |             write_fx_4_state_object_initializer(var, fx); | 
					
						
							|  |  |  |             fx->depth_stencil_state_count += elements_count; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |         case HLSL_CLASS_SAMPLER: | 
					
						
							|  |  |  |             write_fx_4_state_object_initializer(var, fx); | 
					
						
							|  |  |  |             fx->sampler_state_count += elements_count; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-10 13:18:43 +02:00
										 |  |  |         case HLSL_CLASS_RASTERIZER_STATE: | 
					
						
							|  |  |  |             write_fx_4_state_object_initializer(var, fx); | 
					
						
							|  |  |  |             fx->rasterizer_state_count += elements_count; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-07 12:59:24 +02:00
										 |  |  |         case HLSL_CLASS_BLEND_STATE: | 
					
						
							|  |  |  |             write_fx_4_state_object_initializer(var, fx); | 
					
						
							|  |  |  |             fx->blend_state_count += elements_count; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-13 23:33:16 +02:00
										 |  |  |         case HLSL_CLASS_STRING: | 
					
						
							|  |  |  |             write_fx_4_string_initializer(var, fx); | 
					
						
							|  |  |  |             fx->string_count += elements_count; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-24 00:39:37 +01:00
										 |  |  |         default: | 
					
						
							| 
									
										
										
										
											2024-08-05 20:32:24 +02:00
										 |  |  |             hlsl_fixme(ctx, &ctx->location, "Writing initializer for object class %u is not implemented.", | 
					
						
							|  |  |  |                     type->class); | 
					
						
							| 
									
										
										
										
											2024-02-24 00:39:37 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-15 23:44:08 +02:00
										 |  |  |     write_fx_4_annotations(var->annotations, fx); | 
					
						
							| 
									
										
										
										
											2024-02-24 00:39:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ++fx->object_variable_count; | 
					
						
							| 
									
										
										
										
											2024-02-09 22:42:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | static void write_fx_4_buffer(struct hlsl_buffer *b, struct fx_write_context *fx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     enum fx_4_buffer_flags | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         IS_TBUFFER = 0x1, | 
					
						
							|  |  |  |         IS_SINGLE  = 0x2, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer *buffer = &fx->structured; | 
					
						
							|  |  |  |     uint32_t count = 0, bind_point = ~0u, flags = 0, size; | 
					
						
							|  |  |  |     uint32_t name_offset, size_offset; | 
					
						
							|  |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							|  |  |  |     struct hlsl_ir_var *var; | 
					
						
							|  |  |  |     uint32_t count_offset; | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  |     bool shared; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     shared = fx->child_effect && b->modifiers & HLSL_STORAGE_SHARED; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (b->reservation.reg_type) | 
					
						
							|  |  |  |         bind_point = b->reservation.reg_index; | 
					
						
							|  |  |  |     if (b->type == HLSL_BUFFER_TEXTURE) | 
					
						
							|  |  |  |         flags |= IS_TBUFFER; | 
					
						
							| 
									
										
										
										
											2024-03-10 20:01:22 +01:00
										 |  |  |     if (ctx->profile->major_version == 5 && b->modifiers & HLSL_MODIFIER_SINGLE) | 
					
						
							|  |  |  |         flags |= IS_SINGLE; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     name_offset = write_string(b->name, fx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     put_u32(buffer, name_offset); /* Name */ | 
					
						
							|  |  |  |     size_offset = put_u32(buffer, 0); /* Data size */ | 
					
						
							|  |  |  |     put_u32(buffer, flags); /* Flags */ | 
					
						
							|  |  |  |     count_offset = put_u32(buffer, 0); | 
					
						
							|  |  |  |     put_u32(buffer, bind_point); /* Bind point */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  |     if (shared) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         ++fx->shared_buffer_count; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-06-15 23:44:08 +02:00
										 |  |  |         write_fx_4_annotations(b->annotations, fx); | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  |         ++fx->buffer_count; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     count = 0; | 
					
						
							|  |  |  |     size = 0; | 
					
						
							|  |  |  |     LIST_FOR_EACH_ENTRY(var, &ctx->extern_vars, struct hlsl_ir_var, extern_entry) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-05-29 22:25:13 +02:00
										 |  |  |         uint32_t unpacked_size; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-31 01:24:51 +02:00
										 |  |  |         if (!is_numeric_fx_4_type(var->data_type)) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |         if (var->buffer != b) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  |         write_fx_4_numeric_variable(var, shared, fx); | 
					
						
							| 
									
										
										
										
											2025-05-29 22:25:13 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         unpacked_size = var->data_type->reg_size[HLSL_REGSET_NUMERIC] * sizeof(float); | 
					
						
							|  |  |  |         size = max(size, unpacked_size + var->buffer_offset * 4); | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |         ++count; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     set_u32(buffer, count_offset, count); | 
					
						
							|  |  |  |     set_u32(buffer, size_offset, align(size, 16)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  | static void write_buffers(struct fx_write_context *fx, bool shared) | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     struct hlsl_buffer *buffer; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-30 13:59:36 +02:00
										 |  |  |     if (shared && !fx->child_effect) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     LIST_FOR_EACH_ENTRY(buffer, &fx->ctx->buffers, struct hlsl_buffer, entry) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-03-28 11:39:04 +01:00
										 |  |  |         if (!buffer->size && !fx->include_empty_buffers) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         if (!strcmp(buffer->name, "$Params")) | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  |         if (fx->child_effect && (shared != !!(buffer->modifiers & HLSL_STORAGE_SHARED))) | 
					
						
							|  |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         write_fx_4_buffer(buffer, fx); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-19 04:05:00 +02:00
										 |  |  | static bool is_supported_object_variable(const struct hlsl_ctx *ctx, const struct hlsl_ir_var *var) | 
					
						
							| 
									
										
										
										
											2024-02-09 22:42:23 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     const struct hlsl_type *type = hlsl_get_multiarray_element_type(var->data_type); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-05 20:13:17 -06:00
										 |  |  |     switch (type->class) | 
					
						
							| 
									
										
										
										
											2024-02-09 22:42:23 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  |         case HLSL_CLASS_DEPTH_STENCIL_STATE: | 
					
						
							| 
									
										
										
										
											2024-05-06 23:29:15 +02:00
										 |  |  |         case HLSL_CLASS_DEPTH_STENCIL_VIEW: | 
					
						
							| 
									
										
										
										
											2024-02-06 19:41:15 -06:00
										 |  |  |         case HLSL_CLASS_PIXEL_SHADER: | 
					
						
							| 
									
										
										
										
											2024-04-24 11:12:08 +02:00
										 |  |  |         case HLSL_CLASS_RASTERIZER_STATE: | 
					
						
							| 
									
										
										
										
											2024-02-27 13:34:52 -06:00
										 |  |  |         case HLSL_CLASS_RENDER_TARGET_VIEW: | 
					
						
							| 
									
										
										
										
											2024-02-05 20:13:17 -06:00
										 |  |  |         case HLSL_CLASS_SAMPLER: | 
					
						
							| 
									
										
										
										
											2024-02-05 20:25:57 -06:00
										 |  |  |         case HLSL_CLASS_TEXTURE: | 
					
						
							| 
									
										
										
										
											2024-08-07 12:59:24 +02:00
										 |  |  |         case HLSL_CLASS_BLEND_STATE: | 
					
						
							| 
									
										
										
										
											2024-08-13 23:33:16 +02:00
										 |  |  |         case HLSL_CLASS_VERTEX_SHADER: | 
					
						
							|  |  |  |         case HLSL_CLASS_STRING: | 
					
						
							| 
									
										
										
										
											2024-04-19 04:05:00 +02:00
										 |  |  |             return true; | 
					
						
							| 
									
										
										
										
											2024-08-05 20:32:24 +02:00
										 |  |  |         case HLSL_CLASS_COMPUTE_SHADER: | 
					
						
							|  |  |  |         case HLSL_CLASS_DOMAIN_SHADER: | 
					
						
							|  |  |  |         case HLSL_CLASS_HULL_SHADER: | 
					
						
							|  |  |  |             if (ctx->profile->major_version < 5) | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |             return true; | 
					
						
							| 
									
										
										
										
											2024-02-05 20:32:37 -06:00
										 |  |  |         case HLSL_CLASS_UAV: | 
					
						
							| 
									
										
										
										
											2024-04-19 04:05:00 +02:00
										 |  |  |             if (ctx->profile->major_version < 5) | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |             if (type->e.resource.rasteriser_ordered) | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |             return true; | 
					
						
							| 
									
										
										
										
											2024-02-05 20:13:17 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-09 22:42:23 +01:00
										 |  |  |         default: | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-25 01:08:46 +01:00
										 |  |  | static void write_objects(struct fx_write_context *fx, bool shared) | 
					
						
							| 
									
										
										
										
											2024-02-09 22:42:23 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2024-04-19 04:05:00 +02:00
										 |  |  |     struct hlsl_ctx *ctx = fx->ctx; | 
					
						
							| 
									
										
										
										
											2024-02-09 22:42:23 +01:00
										 |  |  |     struct hlsl_ir_var *var; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-25 01:08:46 +01:00
										 |  |  |     if (shared && !fx->child_effect) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-19 04:05:00 +02:00
										 |  |  |     LIST_FOR_EACH_ENTRY(var, &ctx->extern_vars, struct hlsl_ir_var, extern_entry) | 
					
						
							| 
									
										
										
										
											2024-02-09 22:42:23 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-04-19 04:05:00 +02:00
										 |  |  |         if (!is_supported_object_variable(ctx, var)) | 
					
						
							| 
									
										
										
										
											2024-02-09 22:42:23 +01:00
										 |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-25 01:08:46 +01:00
										 |  |  |         if (fx->child_effect && (shared != !!(var->storage_modifiers & HLSL_STORAGE_SHARED))) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-09 22:42:23 +01:00
										 |  |  |         write_fx_4_object_variable(var, fx); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | static int hlsl_fx_4_write(struct hlsl_ctx *ctx, struct vkd3d_shader_code *out) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer buffer = { 0 }; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     struct fx_write_context fx; | 
					
						
							| 
									
										
										
										
											2024-02-07 00:38:59 +01:00
										 |  |  |     uint32_t size_offset; | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |     fx_write_context_init(ctx, &fx_4_ops, &fx); | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     put_u32(&fx.unstructured, 0); /* Empty string placeholder. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  |     write_buffers(&fx, false); | 
					
						
							| 
									
										
										
										
											2024-02-25 01:08:46 +01:00
										 |  |  |     write_objects(&fx, false); | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  |     write_buffers(&fx, true); | 
					
						
							| 
									
										
										
										
											2024-02-25 01:08:46 +01:00
										 |  |  |     write_objects(&fx, true); | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     write_techniques(ctx->globals, &fx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     put_u32(&buffer, ctx->profile->minor_version == 0 ? 0xfeff1001 : 0xfeff1011); /* Version. */ | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     put_u32(&buffer, fx.buffer_count); /* Buffer count. */ | 
					
						
							|  |  |  |     put_u32(&buffer, fx.numeric_variable_count); /* Numeric variable count. */ | 
					
						
							| 
									
										
										
										
											2024-02-09 22:42:23 +01:00
										 |  |  |     put_u32(&buffer, fx.object_variable_count); /* Object variable count. */ | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  |     put_u32(&buffer, fx.shared_buffer_count); | 
					
						
							|  |  |  |     put_u32(&buffer, fx.shared_numeric_variable_count); | 
					
						
							|  |  |  |     put_u32(&buffer, fx.shared_object_count); | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  |     put_u32(&buffer, fx.technique_count); | 
					
						
							|  |  |  |     size_offset = put_u32(&buffer, 0); /* Unstructured size. */ | 
					
						
							| 
									
										
										
										
											2024-08-13 23:33:16 +02:00
										 |  |  |     put_u32(&buffer, fx.string_count); | 
					
						
							| 
									
										
										
										
											2024-04-20 16:28:34 +02:00
										 |  |  |     put_u32(&buffer, fx.texture_count); | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  |     put_u32(&buffer, fx.depth_stencil_state_count); | 
					
						
							| 
									
										
										
										
											2024-08-07 12:59:24 +02:00
										 |  |  |     put_u32(&buffer, fx.blend_state_count); | 
					
						
							| 
									
										
										
										
											2024-06-10 13:18:43 +02:00
										 |  |  |     put_u32(&buffer, fx.rasterizer_state_count); | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |     put_u32(&buffer, fx.sampler_state_count); | 
					
						
							| 
									
										
										
										
											2024-04-20 16:19:54 +02:00
										 |  |  |     put_u32(&buffer, fx.rtv_count); | 
					
						
							| 
									
										
										
										
											2024-05-06 23:29:15 +02:00
										 |  |  |     put_u32(&buffer, fx.dsv_count); | 
					
						
							| 
									
										
										
										
											2024-04-20 16:30:05 +02:00
										 |  |  |     put_u32(&buffer, fx.shader_count); | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  |     put_u32(&buffer, 0); /* Inline shader count. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-07 00:38:59 +01:00
										 |  |  |     set_u32(&buffer, size_offset, fx.unstructured.size); | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     bytecode_put_bytes(&buffer, fx.unstructured.data, fx.unstructured.size); | 
					
						
							| 
									
										
										
										
											2024-02-07 00:38:59 +01:00
										 |  |  |     bytecode_put_bytes_unaligned(&buffer, fx.structured.data, fx.structured.size); | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_free(fx.unstructured.data); | 
					
						
							|  |  |  |     vkd3d_free(fx.structured.data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     set_status(&fx, buffer.status); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-23 02:11:38 +01:00
										 |  |  |     if (fx.status < 0) | 
					
						
							|  |  |  |         ctx->result = fx.status; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!ctx->result) | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         out->code = buffer.data; | 
					
						
							|  |  |  |         out->size = buffer.size; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-12 14:33:05 +01:00
										 |  |  |     return fx_write_context_cleanup(&fx); | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  | static int hlsl_fx_5_write(struct hlsl_ctx *ctx, struct vkd3d_shader_code *out) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct vkd3d_bytecode_buffer buffer = { 0 }; | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     struct fx_write_context fx; | 
					
						
							| 
									
										
										
										
											2024-02-07 00:38:59 +01:00
										 |  |  |     uint32_t size_offset; | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |     fx_write_context_init(ctx, &fx_4_ops, &fx); | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     put_u32(&fx.unstructured, 0); /* Empty string placeholder. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  |     write_buffers(&fx, false); | 
					
						
							| 
									
										
										
										
											2024-02-25 01:08:46 +01:00
										 |  |  |     write_objects(&fx, false); | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  |     /* TODO: interface variables */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-25 18:18:40 +01:00
										 |  |  |     write_groups(&fx); | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     put_u32(&buffer, 0xfeff2001); /* Version. */ | 
					
						
							| 
									
										
										
										
											2024-02-06 15:11:47 +01:00
										 |  |  |     put_u32(&buffer, fx.buffer_count); /* Buffer count. */ | 
					
						
							|  |  |  |     put_u32(&buffer, fx.numeric_variable_count); /* Numeric variable count. */ | 
					
						
							| 
									
										
										
										
											2024-02-09 22:42:23 +01:00
										 |  |  |     put_u32(&buffer, fx.object_variable_count); /* Object variable count. */ | 
					
						
							| 
									
										
										
										
											2024-04-21 19:06:55 +02:00
										 |  |  |     put_u32(&buffer, fx.shared_buffer_count); | 
					
						
							|  |  |  |     put_u32(&buffer, fx.shared_numeric_variable_count); | 
					
						
							|  |  |  |     put_u32(&buffer, fx.shared_object_count); | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  |     put_u32(&buffer, fx.technique_count); | 
					
						
							|  |  |  |     size_offset = put_u32(&buffer, 0); /* Unstructured size. */ | 
					
						
							| 
									
										
										
										
											2024-08-13 23:33:16 +02:00
										 |  |  |     put_u32(&buffer, fx.string_count); | 
					
						
							| 
									
										
										
										
											2024-04-20 16:28:34 +02:00
										 |  |  |     put_u32(&buffer, fx.texture_count); | 
					
						
							| 
									
										
										
										
											2024-06-10 13:05:13 +02:00
										 |  |  |     put_u32(&buffer, fx.depth_stencil_state_count); | 
					
						
							| 
									
										
										
										
											2024-08-07 12:59:24 +02:00
										 |  |  |     put_u32(&buffer, fx.blend_state_count); | 
					
						
							| 
									
										
										
										
											2024-06-10 13:18:43 +02:00
										 |  |  |     put_u32(&buffer, fx.rasterizer_state_count); | 
					
						
							| 
									
										
										
										
											2024-04-21 00:33:41 +02:00
										 |  |  |     put_u32(&buffer, fx.sampler_state_count); | 
					
						
							| 
									
										
										
										
											2024-04-20 16:19:54 +02:00
										 |  |  |     put_u32(&buffer, fx.rtv_count); | 
					
						
							| 
									
										
										
										
											2024-05-06 23:29:15 +02:00
										 |  |  |     put_u32(&buffer, fx.dsv_count); | 
					
						
							| 
									
										
										
										
											2024-04-20 16:30:05 +02:00
										 |  |  |     put_u32(&buffer, fx.shader_count); | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  |     put_u32(&buffer, 0); /* Inline shader count. */ | 
					
						
							|  |  |  |     put_u32(&buffer, fx.group_count); /* Group count. */ | 
					
						
							| 
									
										
										
										
											2024-04-19 11:40:14 +02:00
										 |  |  |     put_u32(&buffer, fx.uav_count); | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  |     put_u32(&buffer, 0); /* Interface variables count. */ | 
					
						
							|  |  |  |     put_u32(&buffer, 0); /* Interface variable element count. */ | 
					
						
							|  |  |  |     put_u32(&buffer, 0); /* Class instance elements count. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-07 00:38:59 +01:00
										 |  |  |     set_u32(&buffer, size_offset, fx.unstructured.size); | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     bytecode_put_bytes(&buffer, fx.unstructured.data, fx.unstructured.size); | 
					
						
							| 
									
										
										
										
											2024-02-07 00:38:59 +01:00
										 |  |  |     bytecode_put_bytes_unaligned(&buffer, fx.structured.data, fx.structured.size); | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_free(fx.unstructured.data); | 
					
						
							|  |  |  |     vkd3d_free(fx.structured.data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     set_status(&fx, buffer.status); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-23 02:11:38 +01:00
										 |  |  |     if (fx.status < 0) | 
					
						
							|  |  |  |         ctx->result = fx.status; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!ctx->result) | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         out->code = buffer.data; | 
					
						
							|  |  |  |         out->size = buffer.size; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-12 14:33:05 +01:00
										 |  |  |     return fx_write_context_cleanup(&fx); | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  | int hlsl_emit_effect_binary(struct hlsl_ctx *ctx, struct vkd3d_shader_code *out) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (ctx->profile->major_version == 2) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-01-16 12:17:06 +01:00
										 |  |  |         return hlsl_fx_2_write(ctx, out); | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     else if (ctx->profile->major_version == 4) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return hlsl_fx_4_write(ctx, out); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (ctx->profile->major_version == 5) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-01-12 14:26:39 +01:00
										 |  |  |         return hlsl_fx_5_write(ctx, out); | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         vkd3d_unreachable(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | struct fx_parser | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-06-12 00:38:10 +02:00
										 |  |  |     enum vkd3d_shader_source_type source_type; | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |     const uint8_t *ptr, *start, *end; | 
					
						
							|  |  |  |     struct vkd3d_shader_message_context *message_context; | 
					
						
							|  |  |  |     struct vkd3d_string_buffer buffer; | 
					
						
							| 
									
										
										
										
											2024-10-17 11:36:34 +02:00
										 |  |  |     unsigned int indent; | 
					
						
							| 
									
										
										
										
											2025-05-06 01:07:03 +02:00
										 |  |  |     struct | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         unsigned int major; | 
					
						
							|  |  |  |         unsigned int minor; | 
					
						
							|  |  |  |     } version; | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |     struct | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         const uint8_t *ptr; | 
					
						
							|  |  |  |         const uint8_t *end; | 
					
						
							|  |  |  |         uint32_t size; | 
					
						
							|  |  |  |     } unstructured; | 
					
						
							|  |  |  |     uint32_t buffer_count; | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  |     uint32_t object_count; | 
					
						
							| 
									
										
										
										
											2024-10-19 18:12:52 +02:00
										 |  |  |     uint32_t group_count; | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  |     struct | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t count; | 
					
						
							|  |  |  |         uint32_t *types; | 
					
						
							|  |  |  |     } objects; | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |     bool failed; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static uint32_t fx_parser_read_u32(struct fx_parser *parser) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ((parser->end - parser->ptr) < sizeof(uint32_t)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parser->failed = true; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ret = *(uint32_t *)parser->ptr; | 
					
						
							|  |  |  |     parser->ptr += sizeof(uint32_t); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fx_parser_read_u32s(struct fx_parser *parser, void *dst, size_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t *ptr = dst; | 
					
						
							|  |  |  |     size_t i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < size / sizeof(uint32_t); ++i) | 
					
						
							|  |  |  |         ptr[i] = fx_parser_read_u32(parser); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fx_parser_skip(struct fx_parser *parser, size_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if ((parser->end - parser->ptr) < size) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parser->ptr = parser->end; | 
					
						
							|  |  |  |         parser->failed = true; | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     parser->ptr += size; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void VKD3D_PRINTF_FUNC(3, 4) fx_parser_error(struct fx_parser *parser, enum vkd3d_shader_error error, | 
					
						
							|  |  |  |         const char *format, ...) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     va_list args; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     va_start(args, format); | 
					
						
							|  |  |  |     vkd3d_shader_verror(parser->message_context, NULL, error, format, args); | 
					
						
							|  |  |  |     va_end(args); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parser->failed = true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  | static const void *fx_parser_get_unstructured_ptr(struct fx_parser *parser, uint32_t offset, size_t size) | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     const uint8_t *ptr = parser->unstructured.ptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (offset >= parser->unstructured.size | 
					
						
							|  |  |  |             || size > parser->unstructured.size - offset) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parser->failed = true; | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |     return &ptr[offset]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  | static const void *fx_parser_get_ptr(struct fx_parser *parser, size_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (parser->end - parser->ptr < size) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parser->failed = true; | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return parser->ptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | static uint32_t fx_parser_read_unstructured(struct fx_parser *parser, void *dst, uint32_t offset, size_t size) | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     const uint8_t *ptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     memset(dst, 0, size); | 
					
						
							|  |  |  |     if (!(ptr = fx_parser_get_unstructured_ptr(parser, offset, size))) | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |         return offset; | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  |     memcpy(dst, ptr, size); | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |     return offset + size; | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  | static void parse_fx_start_indent(struct fx_parser *parser) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ++parser->indent; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void parse_fx_end_indent(struct fx_parser *parser) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     --parser->indent; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void parse_fx_print_indent(struct fx_parser *parser) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "%*s", 4 * parser->indent, ""); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  | static void fx_2_print_string_literal(struct fx_parser *parser, const char *prefix, | 
					
						
							|  |  |  |         const char *s, uint32_t max_len, const char *suffix) | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  |     uint32_t len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (s) | 
					
						
							|  |  |  |         len = strnlen(s, max_len); | 
					
						
							|  |  |  |     if (!s || len == max_len) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_error(parser, VKD3D_SHADER_ERROR_FX_INVALID_DATA, "Failed to parse a string entry."); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "%s", prefix); | 
					
						
							|  |  |  |     vkd3d_string_buffer_print_string_escaped(&parser->buffer, s, len); | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "%s", suffix); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fx_2_parse_string_literal(struct fx_parser *parser, uint32_t offset, | 
					
						
							|  |  |  |         bool unstructured, const char *prefix, const char *suffix) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t max_len; | 
					
						
							|  |  |  |     const char *s; | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  |     if (unstructured) | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  |         fx_parser_read_unstructured(parser, &max_len, offset, sizeof(max_len)); | 
					
						
							|  |  |  |         s = fx_parser_get_unstructured_ptr(parser, offset + 4, max_len); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         max_len = fx_parser_read_u32(parser); | 
					
						
							|  |  |  |         s = fx_parser_get_ptr(parser, max_len); | 
					
						
							|  |  |  |         fx_parser_skip(parser, align(max_len, 4)); | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  |     fx_2_print_string_literal(parser, prefix, s, max_len, suffix); | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-24 21:21:21 +01:00
										 |  |  | static unsigned int fx_get_fx_2_type_size(struct fx_parser *parser, uint32_t *offset) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t element_count, member_count, class, columns, rows; | 
					
						
							|  |  |  |     unsigned int size = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fx_parser_read_unstructured(parser, &class, *offset + 4, sizeof(class)); | 
					
						
							|  |  |  |     fx_parser_read_unstructured(parser, &element_count, *offset + 16, sizeof(element_count)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (class == D3DXPC_STRUCT) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         *offset = fx_parser_read_unstructured(parser, &member_count, *offset + 20, sizeof(member_count)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (uint32_t i = 0; i < member_count; ++i) | 
					
						
							|  |  |  |             size += fx_get_fx_2_type_size(parser, offset); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (class == D3DXPC_VECTOR) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_read_unstructured(parser, &columns, *offset + 20, sizeof(columns)); | 
					
						
							|  |  |  |         *offset = fx_parser_read_unstructured(parser, &rows, *offset + 24, sizeof(rows)); | 
					
						
							|  |  |  |         size = rows * columns * sizeof(float); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (class == D3DXPC_MATRIX_ROWS | 
					
						
							|  |  |  |             || class == D3DXPC_MATRIX_COLUMNS | 
					
						
							|  |  |  |             || class == D3DXPC_SCALAR) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_read_unstructured(parser, &rows, *offset + 20, sizeof(rows)); | 
					
						
							|  |  |  |         *offset = fx_parser_read_unstructured(parser, &columns, *offset + 24, sizeof(columns)); | 
					
						
							|  |  |  |         size = rows * columns * sizeof(float); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         *offset += 20; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (element_count) | 
					
						
							|  |  |  |         size *= element_count; | 
					
						
							|  |  |  |     return size; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  | static const char *const fx_2_types[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     [D3DXPT_VOID]           = "void", | 
					
						
							|  |  |  |     [D3DXPT_BOOL]           = "bool", | 
					
						
							|  |  |  |     [D3DXPT_INT]            = "int", | 
					
						
							|  |  |  |     [D3DXPT_FLOAT]          = "float", | 
					
						
							|  |  |  |     [D3DXPT_STRING]         = "string", | 
					
						
							|  |  |  |     [D3DXPT_TEXTURE]        = "texture", | 
					
						
							|  |  |  |     [D3DXPT_TEXTURE1D]      = "texture1D", | 
					
						
							|  |  |  |     [D3DXPT_TEXTURE2D]      = "texture2D", | 
					
						
							|  |  |  |     [D3DXPT_TEXTURE3D]      = "texture3D", | 
					
						
							|  |  |  |     [D3DXPT_TEXTURECUBE]    = "textureCUBE", | 
					
						
							|  |  |  |     [D3DXPT_SAMPLER]        = "sampler", | 
					
						
							|  |  |  |     [D3DXPT_SAMPLER1D]      = "sampler1D", | 
					
						
							|  |  |  |     [D3DXPT_SAMPLER2D]      = "sampler2D", | 
					
						
							|  |  |  |     [D3DXPT_SAMPLER3D]      = "sampler3D", | 
					
						
							|  |  |  |     [D3DXPT_SAMPLERCUBE]    = "samplerCUBE", | 
					
						
							|  |  |  |     [D3DXPT_PIXELSHADER]    = "PixelShader", | 
					
						
							|  |  |  |     [D3DXPT_VERTEXSHADER]   = "VertexShader", | 
					
						
							|  |  |  |     [D3DXPT_PIXELFRAGMENT]  = "<pixel-fragment>", | 
					
						
							|  |  |  |     [D3DXPT_VERTEXFRAGMENT] = "<vertex-fragment>", | 
					
						
							|  |  |  |     [D3DXPT_UNSUPPORTED]    = "<unsupported>", | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-05 19:51:34 +01:00
										 |  |  | static void fx_parse_fx_2_type(struct fx_parser *parser, uint32_t offset) | 
					
						
							| 
									
										
										
										
											2025-02-24 21:21:21 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     uint32_t type, class, rows, columns; | 
					
						
							|  |  |  |     const char *name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fx_parser_read_unstructured(parser, &type, offset, sizeof(type)); | 
					
						
							|  |  |  |     fx_parser_read_unstructured(parser, &class, offset + 4, sizeof(class)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (class == D3DXPC_STRUCT) | 
					
						
							|  |  |  |         name = "struct"; | 
					
						
							|  |  |  |     else | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  |         name = type < ARRAY_SIZE(fx_2_types) ? fx_2_types[type] : "<unknown>"; | 
					
						
							| 
									
										
										
										
											2025-02-24 21:21:21 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "%s", name); | 
					
						
							|  |  |  |     if (class == D3DXPC_VECTOR) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_read_unstructured(parser, &columns, offset + 20, sizeof(columns)); | 
					
						
							|  |  |  |         fx_parser_read_unstructured(parser, &rows, offset + 24, sizeof(rows)); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "%u", columns); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (class == D3DXPC_MATRIX_ROWS || class == D3DXPC_MATRIX_COLUMNS) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_read_unstructured(parser, &rows, offset + 20, sizeof(rows)); | 
					
						
							|  |  |  |         fx_parser_read_unstructured(parser, &columns, offset + 24, sizeof(columns)); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "%ux%u", rows, columns); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  | static void parse_fx_2_object_value(struct fx_parser *parser, uint32_t element_count, | 
					
						
							|  |  |  |         uint32_t type, uint32_t offset) | 
					
						
							| 
									
										
										
										
											2025-03-07 12:45:24 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     uint32_t id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     element_count = max(element_count, 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (uint32_t i = 0; i < element_count; ++i, offset += 4) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_read_unstructured(parser, &id, offset, sizeof(id)); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "<object id %u>", id); | 
					
						
							|  |  |  |         if (element_count > 1) | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, ", "); | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  |         if (id < parser->objects.count) | 
					
						
							|  |  |  |             parser->objects.types[id] = type; | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             fx_parser_error(parser, VKD3D_SHADER_ERROR_FX_INVALID_DATA, | 
					
						
							|  |  |  |                     "Initializer object id exceeds the number of objects in the effect."); | 
					
						
							| 
									
										
										
										
											2025-03-07 12:45:24 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-07 12:45:24 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-24 21:21:21 +01:00
										 |  |  | static void parse_fx_2_numeric_value(struct fx_parser *parser, uint32_t offset, | 
					
						
							|  |  |  |         unsigned int size, uint32_t base_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     unsigned int i, comp_count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     comp_count = size / sizeof(uint32_t); | 
					
						
							|  |  |  |     if (comp_count > 1) | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "{"); | 
					
						
							|  |  |  |     for (i = 0; i < comp_count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         union hlsl_constant_value_component value; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         fx_parser_read_unstructured(parser, &value, offset, sizeof(uint32_t)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (base_type == D3DXPT_INT) | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "%d", value.i); | 
					
						
							|  |  |  |         else if (base_type == D3DXPT_BOOL) | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "%s", value.u ? "true" : "false" ); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             vkd3d_string_buffer_print_f32(&parser->buffer, value.f); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (i < comp_count - 1) | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, ", "); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         offset += sizeof(uint32_t); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (comp_count > 1) | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "}"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-05 19:51:34 +01:00
										 |  |  | static void fx_parse_fx_2_parameter(struct fx_parser *parser, uint32_t offset) | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-02-24 21:21:21 +01:00
										 |  |  |     struct fx_2_var | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t type; | 
					
						
							|  |  |  |         uint32_t class; | 
					
						
							|  |  |  |         uint32_t name; | 
					
						
							|  |  |  |         uint32_t semantic; | 
					
						
							|  |  |  |         uint32_t element_count; | 
					
						
							|  |  |  |     } var; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-05 19:51:34 +01:00
										 |  |  |     fx_parser_read_unstructured(parser, &var, offset, sizeof(var)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fx_parse_fx_2_type(parser, offset); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  |     fx_2_parse_string_literal(parser, var.name, true, " ", ""); | 
					
						
							| 
									
										
										
										
											2025-03-05 19:51:34 +01:00
										 |  |  |     if (var.element_count) | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "[%u]", var.element_count); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-07 12:45:24 +01:00
										 |  |  | static bool is_fx_2_sampler(uint32_t type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return type == D3DXPT_SAMPLER | 
					
						
							|  |  |  |             || type == D3DXPT_SAMPLER1D | 
					
						
							|  |  |  |             || type == D3DXPT_SAMPLER2D | 
					
						
							|  |  |  |             || type == D3DXPT_SAMPLER3D | 
					
						
							|  |  |  |             || type == D3DXPT_SAMPLERCUBE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-04 20:28:33 +02:00
										 |  |  | static void fx_parse_fx_2_assignment(struct fx_parser *parser, enum hlsl_type_class container, | 
					
						
							|  |  |  |         const struct fx_assignment *entry); | 
					
						
							| 
									
										
										
										
											2025-03-10 17:58:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void parse_fx_2_sampler(struct fx_parser *parser, uint32_t element_count, | 
					
						
							|  |  |  |         uint32_t offset) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct fx_assignment entry; | 
					
						
							|  |  |  |     uint32_t count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     element_count = max(element_count, 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "\n"); | 
					
						
							|  |  |  |     for (uint32_t i = 0; i < element_count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_read_unstructured(parser, &count, offset, sizeof(count)); | 
					
						
							|  |  |  |         offset += sizeof(count); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parse_fx_start_indent(parser); | 
					
						
							|  |  |  |         parse_fx_print_indent(parser); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "{\n"); | 
					
						
							|  |  |  |         parse_fx_start_indent(parser); | 
					
						
							|  |  |  |         for (uint32_t j = 0; j < count; ++j, offset += sizeof(entry)) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             fx_parser_read_unstructured(parser, &entry, offset, sizeof(entry)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             parse_fx_print_indent(parser); | 
					
						
							| 
									
										
										
										
											2025-06-04 20:28:33 +02:00
										 |  |  |             fx_parse_fx_2_assignment(parser, HLSL_CLASS_SAMPLER, &entry); | 
					
						
							| 
									
										
										
										
											2025-03-10 17:58:18 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         parse_fx_end_indent(parser); | 
					
						
							|  |  |  |         parse_fx_print_indent(parser); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "},\n"); | 
					
						
							|  |  |  |         parse_fx_end_indent(parser); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-05 19:51:34 +01:00
										 |  |  | static void fx_parse_fx_2_initial_value(struct fx_parser *parser, uint32_t param, uint32_t value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct fx_2_var | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t type; | 
					
						
							|  |  |  |         uint32_t class; | 
					
						
							|  |  |  |         uint32_t name; | 
					
						
							|  |  |  |         uint32_t semantic; | 
					
						
							|  |  |  |         uint32_t element_count; | 
					
						
							|  |  |  |     } var; | 
					
						
							|  |  |  |     unsigned int size; | 
					
						
							|  |  |  |     uint32_t offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!value) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fx_parser_read_unstructured(parser, &var, param, sizeof(var)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     offset = param; | 
					
						
							|  |  |  |     size = fx_get_fx_2_type_size(parser, &offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, " = "); | 
					
						
							|  |  |  |     if (var.element_count) | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "{ "); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-07 12:45:24 +01:00
										 |  |  |     if (var.class == D3DXPC_OBJECT) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (is_fx_2_sampler(var.type)) | 
					
						
							| 
									
										
										
										
											2025-03-10 17:58:18 +01:00
										 |  |  |             parse_fx_2_sampler(parser, var.element_count, value); | 
					
						
							| 
									
										
										
										
											2025-03-07 12:45:24 +01:00
										 |  |  |         else | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  |             parse_fx_2_object_value(parser, var.element_count, var.type, value); | 
					
						
							| 
									
										
										
										
											2025-03-07 12:45:24 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-03-05 19:51:34 +01:00
										 |  |  |     else | 
					
						
							| 
									
										
										
										
											2025-03-07 12:45:24 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-03-05 19:51:34 +01:00
										 |  |  |         parse_fx_2_numeric_value(parser, value, size, var.type); | 
					
						
							| 
									
										
										
										
											2025-03-07 12:45:24 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-03-05 19:51:34 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (var.element_count) | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, " }"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fx_parse_fx_2_annotations(struct fx_parser *parser, uint32_t count) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t param, value; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-24 21:21:21 +01:00
										 |  |  |     if (parser->failed || !count) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "\n"); | 
					
						
							|  |  |  |     parse_fx_print_indent(parser); | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "<\n"); | 
					
						
							|  |  |  |     parse_fx_start_indent(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (uint32_t i = 0; i < count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         param = fx_parser_read_u32(parser); | 
					
						
							|  |  |  |         value = fx_parser_read_u32(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parse_fx_print_indent(parser); | 
					
						
							| 
									
										
										
										
											2025-03-05 19:51:34 +01:00
										 |  |  |         fx_parse_fx_2_parameter(parser, param); | 
					
						
							|  |  |  |         fx_parse_fx_2_initial_value(parser, param, value); | 
					
						
							| 
									
										
										
										
											2025-02-24 21:21:21 +01:00
										 |  |  |         vkd3d_string_buffer_printf(&parser->buffer, ";\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parse_fx_end_indent(parser); | 
					
						
							|  |  |  |     parse_fx_print_indent(parser); | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, ">"); | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 08:36:17 +02:00
										 |  |  | static const struct fx_state *fx_2_get_state_by_id(enum hlsl_type_class container, uint32_t id) | 
					
						
							| 
									
										
										
										
											2025-06-04 20:28:33 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-06-16 10:52:43 +02:00
										 |  |  |     struct fx_state_table table = fx_get_state_table(container, 2, 0); | 
					
						
							| 
									
										
										
										
											2025-06-04 20:28:33 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* State identifiers are sequential, no gaps */ | 
					
						
							| 
									
										
										
										
											2025-06-16 10:52:43 +02:00
										 |  |  |     if (id >= table.ptr[0].id && id <= table.ptr[table.count - 1].id) | 
					
						
							|  |  |  |         return &table.ptr[id - table.ptr[0].id]; | 
					
						
							| 
									
										
										
										
											2025-06-04 20:28:33 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fx_parse_fx_2_assignment(struct fx_parser *parser, enum hlsl_type_class container, | 
					
						
							|  |  |  |         const struct fx_assignment *entry) | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     const struct rhs_named_value *named_value = NULL; | 
					
						
							| 
									
										
										
										
											2025-06-16 08:36:17 +02:00
										 |  |  |     const struct fx_state *state; | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-04 20:28:33 +02:00
										 |  |  |     if ((state = fx_2_get_state_by_id(container, entry->id))) | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "%s", state->name); | 
					
						
							|  |  |  |         if (state->array_size > 1) | 
					
						
							| 
									
										
										
										
											2025-03-13 22:21:36 +01:00
										 |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "[%u]", entry->lhs_index); | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-03-13 22:21:36 +01:00
										 |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "<unrecognized state %u>", entry->id); | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, " = "); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-02 00:16:21 +02:00
										 |  |  |     if (state && state->type == FX_UINT && state->values) | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         const struct rhs_named_value *ptr = state->values; | 
					
						
							|  |  |  |         uint32_t value; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-13 22:21:36 +01:00
										 |  |  |         fx_parser_read_unstructured(parser, &value, entry->value, sizeof(value)); | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         while (ptr->name) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (value == ptr->value) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 named_value = ptr; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             ++ptr; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (named_value) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "%s /* %u */", named_value->name, named_value->value); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-03-10 23:05:27 +01:00
										 |  |  |     else if (state) | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-06-02 00:51:01 +02:00
										 |  |  |         if (state->type == FX_UINT || state->type == FX_FLOAT || state->type == FX_BOOL) | 
					
						
							| 
									
										
										
										
											2025-03-10 23:05:27 +01:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2025-06-02 00:14:32 +02:00
										 |  |  |             uint32_t offset = entry->type, base_type; | 
					
						
							| 
									
										
										
										
											2025-03-10 23:05:27 +01:00
										 |  |  |             unsigned int size; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             size = fx_get_fx_2_type_size(parser, &offset); | 
					
						
							| 
									
										
										
										
											2025-06-02 00:14:32 +02:00
										 |  |  |             fx_parser_read_unstructured(parser, &base_type, entry->type, sizeof(base_type)); | 
					
						
							|  |  |  |             parse_fx_2_numeric_value(parser, entry->value, size, base_type); | 
					
						
							| 
									
										
										
										
											2025-03-10 23:05:27 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         else if (state->type == FX_VERTEXSHADER || state->type == FX_PIXELSHADER) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             uint32_t id; | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-10 23:05:27 +01:00
										 |  |  |             fx_parser_read_unstructured(parser, &id, entry->value, sizeof(id)); | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "<object id %u>", id); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "<ignored>"); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "<ignored>"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, ";\n"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  | static void fx_parse_fx_2_technique(struct fx_parser *parser) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct fx_technique | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t name; | 
					
						
							|  |  |  |         uint32_t annotation_count; | 
					
						
							|  |  |  |         uint32_t pass_count; | 
					
						
							|  |  |  |     } technique; | 
					
						
							|  |  |  |     struct fx_pass | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t name; | 
					
						
							|  |  |  |         uint32_t annotation_count; | 
					
						
							|  |  |  |         uint32_t assignment_count; | 
					
						
							|  |  |  |     } pass; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (parser->failed) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fx_parser_read_u32s(parser, &technique, sizeof(technique)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parse_fx_print_indent(parser); | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  |     fx_2_parse_string_literal(parser, technique.name, true, "technique ", ""); | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  |     fx_parse_fx_2_annotations(parser, technique.annotation_count); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "\n"); | 
					
						
							|  |  |  |     parse_fx_print_indent(parser); | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "{\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parse_fx_start_indent(parser); | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  |     for (uint32_t i = 0; i < technique.pass_count; ++i) | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         fx_parser_read_u32s(parser, &pass, sizeof(pass)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parse_fx_print_indent(parser); | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  |         fx_2_parse_string_literal(parser, pass.name, true, "pass ", ""); | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  |         fx_parse_fx_2_annotations(parser, pass.annotation_count); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "\n"); | 
					
						
							|  |  |  |         parse_fx_print_indent(parser); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "{\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parse_fx_start_indent(parser); | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  |         for (uint32_t j = 0; j < pass.assignment_count; ++j) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2025-03-13 22:21:36 +01:00
										 |  |  |             struct fx_assignment entry; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  |             parse_fx_print_indent(parser); | 
					
						
							| 
									
										
										
										
											2025-03-13 22:21:36 +01:00
										 |  |  |             fx_parser_read_u32s(parser, &entry, sizeof(entry)); | 
					
						
							| 
									
										
										
										
											2025-06-04 20:28:33 +02:00
										 |  |  |             fx_parse_fx_2_assignment(parser, HLSL_CLASS_PASS, &entry); | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  |         parse_fx_end_indent(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parse_fx_print_indent(parser); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "}\n\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parse_fx_end_indent(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parse_fx_print_indent(parser); | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "}\n\n"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-05 19:51:34 +01:00
										 |  |  | static void fx_2_parse_parameters(struct fx_parser *parser, uint32_t count) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct fx_2_parameter | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t type; | 
					
						
							|  |  |  |         uint32_t value; | 
					
						
							|  |  |  |         uint32_t flags; | 
					
						
							|  |  |  |         uint32_t annotation_count; | 
					
						
							|  |  |  |     } param; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (uint32_t i = 0; i < count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_read_u32s(parser, ¶m, sizeof(param)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         fx_parse_fx_2_parameter(parser, param.type); | 
					
						
							|  |  |  |         fx_parse_fx_2_annotations(parser, param.annotation_count); | 
					
						
							|  |  |  |         fx_parse_fx_2_initial_value(parser, param.type, param.value); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, ";\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (count) | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "\n"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-10 23:05:27 +01:00
										 |  |  | static void fx_parse_shader_blob(struct fx_parser *parser, enum vkd3d_shader_source_type source_type, | 
					
						
							|  |  |  |         const void *data, uint32_t data_size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct vkd3d_shader_compile_info info = { 0 }; | 
					
						
							|  |  |  |     struct vkd3d_shader_code output; | 
					
						
							|  |  |  |     const char *p, *q, *end; | 
					
						
							|  |  |  |     int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static const struct vkd3d_shader_compile_option options[] = | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-05-20 12:00:00 +02:00
										 |  |  |         {VKD3D_SHADER_COMPILE_OPTION_API_VERSION, VKD3D_SHADER_API_VERSION_1_16}, | 
					
						
							| 
									
										
										
										
											2025-03-10 23:05:27 +01:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     info.type = VKD3D_SHADER_STRUCTURE_TYPE_COMPILE_INFO; | 
					
						
							|  |  |  |     info.source.code = data; | 
					
						
							|  |  |  |     info.source.size = data_size; | 
					
						
							|  |  |  |     info.source_type = source_type; | 
					
						
							|  |  |  |     info.target_type = VKD3D_SHADER_TARGET_D3D_ASM; | 
					
						
							|  |  |  |     info.options = options; | 
					
						
							|  |  |  |     info.option_count = ARRAY_SIZE(options); | 
					
						
							|  |  |  |     info.log_level = VKD3D_SHADER_LOG_INFO; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ((ret = vkd3d_shader_compile(&info, &output, NULL)) < 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_error(parser, VKD3D_SHADER_ERROR_FX_INVALID_DATA, | 
					
						
							|  |  |  |                 "Failed to disassemble shader blob."); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     parse_fx_print_indent(parser); | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "asm {\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parse_fx_start_indent(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     end = (const char *)output.code + output.size; | 
					
						
							|  |  |  |     for (p = output.code; p < end; p = q) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (!(q = memchr(p, '\n', end - p))) | 
					
						
							|  |  |  |             q = end; | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             ++q; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parse_fx_print_indent(parser); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "%.*s", (int)(q - p), p); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parse_fx_end_indent(parser); | 
					
						
							|  |  |  |     parse_fx_print_indent(parser); | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "}"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_shader_free_shader_code(&output); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  | static void fx_parse_fx_2_data_blob(struct fx_parser *parser) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t id, size; | 
					
						
							| 
									
										
										
										
											2025-03-10 23:05:27 +01:00
										 |  |  |     const void *data; | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     id = fx_parser_read_u32(parser); | 
					
						
							|  |  |  |     size = fx_parser_read_u32(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parse_fx_print_indent(parser); | 
					
						
							|  |  |  |     if (id < parser->objects.count) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t type = parser->objects.types[id]; | 
					
						
							|  |  |  |         switch (type) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             case D3DXPT_STRING: | 
					
						
							|  |  |  |             case D3DXPT_TEXTURE: | 
					
						
							|  |  |  |             case D3DXPT_TEXTURE1D: | 
					
						
							|  |  |  |             case D3DXPT_TEXTURE2D: | 
					
						
							|  |  |  |             case D3DXPT_TEXTURE3D: | 
					
						
							|  |  |  |             case D3DXPT_TEXTURECUBE: | 
					
						
							|  |  |  |             case D3DXPT_PIXELSHADER: | 
					
						
							|  |  |  |             case D3DXPT_VERTEXSHADER: | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, "%s object %u size %u bytes%s\n", | 
					
						
							|  |  |  |                         fx_2_types[type], id, size, size ? ":" : ","); | 
					
						
							| 
									
										
										
										
											2025-03-10 23:05:27 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 if (size) | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  |                 { | 
					
						
							| 
									
										
										
										
											2025-03-10 23:05:27 +01:00
										 |  |  |                     data = fx_parser_get_ptr(parser, size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     if (type == D3DXPT_STRING) | 
					
						
							|  |  |  |                     { | 
					
						
							|  |  |  |                         parse_fx_start_indent(parser); | 
					
						
							|  |  |  |                         parse_fx_print_indent(parser); | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  |                         fx_2_print_string_literal(parser, "\"", (const char *)data, size, "\""); | 
					
						
							| 
									
										
										
										
											2025-03-10 23:05:27 +01:00
										 |  |  |                         parse_fx_end_indent(parser); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     else if (type == D3DXPT_PIXELSHADER || type == D3DXPT_VERTEXSHADER) | 
					
						
							|  |  |  |                     { | 
					
						
							|  |  |  |                         fx_parse_shader_blob(parser, VKD3D_SHADER_SOURCE_D3D_BYTECODE, data, size); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                     vkd3d_string_buffer_printf(&parser->buffer, "\n"); | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  |                 } | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, "<type%u> object %u size %u bytes\n", type, id, size); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "object %u - out-of-range id\n", id); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fx_parser_skip(parser, align(size, 4)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 23:06:58 +02:00
										 |  |  | static void fx_2_parse_fxlvm_expression(struct fx_parser *parser, const uint32_t *blob, uint32_t size); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  | static void fx_parse_fx_2_array_selector(struct fx_parser *parser) | 
					
						
							| 
									
										
										
										
											2025-03-11 15:52:30 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  |     uint32_t size, blob_size = 0; | 
					
						
							| 
									
										
										
										
											2025-03-11 15:52:30 +01:00
										 |  |  |     const void *blob = NULL; | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  |     const uint8_t *end; | 
					
						
							| 
									
										
										
										
											2025-03-11 15:52:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  |     size = fx_parser_read_u32(parser); | 
					
						
							|  |  |  |     end = parser->ptr + size; | 
					
						
							| 
									
										
										
										
											2025-03-11 15:52:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  |     fx_2_parse_string_literal(parser, 0, false, "array \"", "\"\n"); | 
					
						
							| 
									
										
										
										
											2025-03-11 15:52:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (parser->ptr <= end) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         blob_size = end - parser->ptr; | 
					
						
							|  |  |  |         blob = fx_parser_get_ptr(parser, blob_size); | 
					
						
							|  |  |  |         fx_parser_skip(parser, blob_size); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_error(parser, VKD3D_SHADER_ERROR_FX_INVALID_DATA, | 
					
						
							|  |  |  |                 "Malformed blob entry in the array selector."); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (blob) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parse_fx_print_indent(parser); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "selector blob size %u\n", blob_size); | 
					
						
							| 
									
										
										
										
											2025-06-07 23:06:58 +02:00
										 |  |  |         fx_2_parse_fxlvm_expression(parser, blob, blob_size); | 
					
						
							| 
									
										
										
										
											2025-03-11 15:52:30 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-11 00:11:01 +01:00
										 |  |  | static void fx_parse_fx_2_complex_state(struct fx_parser *parser) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t technique; | 
					
						
							|  |  |  |         uint32_t index; | 
					
						
							|  |  |  |         uint32_t element; | 
					
						
							|  |  |  |         uint32_t state; | 
					
						
							| 
									
										
										
										
											2025-03-11 15:52:30 +01:00
										 |  |  |         uint32_t assignment_type; | 
					
						
							| 
									
										
										
										
											2025-03-11 00:11:01 +01:00
										 |  |  |     } state; | 
					
						
							| 
									
										
										
										
											2025-06-07 23:43:50 +02:00
										 |  |  |     const uint32_t *data; | 
					
						
							| 
									
										
										
										
											2025-03-11 00:11:01 +01:00
										 |  |  |     uint32_t size; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fx_parser_read_u32s(parser, &state, sizeof(state)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (state.technique == ~0u) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "parameter %u[%u], state %u =\n", | 
					
						
							|  |  |  |                 state.index, state.element, state.state); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "technique %u, pass %u, state %u =\n", | 
					
						
							|  |  |  |                 state.technique, state.index, state.state); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parse_fx_print_indent(parser); | 
					
						
							| 
									
										
										
										
											2025-03-11 15:52:30 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (state.assignment_type == FX_2_ASSIGNMENT_PARAMETER) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  |         fx_2_parse_string_literal(parser, 0, false, "parameter \"", "\"\n"); | 
					
						
							| 
									
										
										
										
											2025-03-11 15:52:30 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     else if (state.assignment_type == FX_2_ASSIGNMENT_ARRAY_SELECTOR) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  |         fx_parse_fx_2_array_selector(parser); | 
					
						
							| 
									
										
										
										
											2025-03-11 15:52:30 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-06-07 23:11:09 +02:00
										 |  |  |     else if (state.assignment_type == FX_2_ASSIGNMENT_CODE_BLOB) | 
					
						
							| 
									
										
										
										
											2025-03-11 15:52:30 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-06-07 10:42:49 +02:00
										 |  |  |         size = fx_parser_read_u32(parser); | 
					
						
							| 
									
										
										
										
											2025-03-11 15:52:30 +01:00
										 |  |  |         data = fx_parser_get_ptr(parser, size); | 
					
						
							| 
									
										
										
										
											2025-06-07 23:43:50 +02:00
										 |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "blob size %u\n", size); | 
					
						
							|  |  |  |         fx_2_parse_fxlvm_expression(parser, data, size); | 
					
						
							| 
									
										
										
										
											2025-03-11 15:52:30 +01:00
										 |  |  |         fx_parser_skip(parser, align(size, 4)); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-06-07 23:11:09 +02:00
										 |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_error(parser, VKD3D_SHADER_ERROR_FX_INVALID_DATA, | 
					
						
							|  |  |  |                 "Unknown state assignment type %u.", state.assignment_type); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-03-11 00:11:01 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-05 16:34:33 +01:00
										 |  |  | static void fx_2_parse(struct fx_parser *parser) | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-03-11 00:11:01 +01:00
										 |  |  |     uint32_t i, size, parameter_count, technique_count, blob_count, state_count; | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     fx_parser_skip(parser, sizeof(uint32_t)); /* Version */ | 
					
						
							|  |  |  |     size = fx_parser_read_u32(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parser->unstructured.ptr = parser->ptr; | 
					
						
							|  |  |  |     parser->unstructured.end = parser->ptr + size; | 
					
						
							|  |  |  |     parser->unstructured.size = size; | 
					
						
							|  |  |  |     fx_parser_skip(parser, size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parameter_count = fx_parser_read_u32(parser); | 
					
						
							|  |  |  |     technique_count = fx_parser_read_u32(parser); | 
					
						
							|  |  |  |     fx_parser_read_u32(parser); /* Shader count */ | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  |     parser->objects.count = fx_parser_read_u32(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!(parser->objects.types = calloc(parser->objects.count, sizeof(*parser->objects.types)))) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_error(parser, VKD3D_SHADER_ERROR_FX_OUT_OF_MEMORY, "Out of memory."); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     fx_2_parse_parameters(parser, parameter_count); | 
					
						
							|  |  |  |     for (i = 0; i < technique_count; ++i) | 
					
						
							|  |  |  |         fx_parse_fx_2_technique(parser); | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     blob_count = fx_parser_read_u32(parser); | 
					
						
							| 
									
										
										
										
											2025-03-11 00:11:01 +01:00
										 |  |  |     state_count = fx_parser_read_u32(parser); | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "object data {\n"); | 
					
						
							|  |  |  |     parse_fx_start_indent(parser); | 
					
						
							|  |  |  |     for (i = 0; i < blob_count; ++i) | 
					
						
							|  |  |  |         fx_parse_fx_2_data_blob(parser); | 
					
						
							|  |  |  |     parse_fx_end_indent(parser); | 
					
						
							| 
									
										
										
										
											2025-03-11 00:11:01 +01:00
										 |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "}\n\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "state data {\n"); | 
					
						
							|  |  |  |     parse_fx_start_indent(parser); | 
					
						
							|  |  |  |     for (i = 0; i < state_count; ++i) | 
					
						
							|  |  |  |         fx_parse_fx_2_complex_state(parser); | 
					
						
							|  |  |  |     parse_fx_end_indent(parser); | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "}\n"); | 
					
						
							| 
									
										
										
										
											2025-02-23 11:41:49 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  | static const char *fx_4_get_string(struct fx_parser *parser, uint32_t offset) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const uint8_t *ptr = parser->unstructured.ptr; | 
					
						
							|  |  |  |     const uint8_t *end = parser->unstructured.end; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (offset >= parser->unstructured.size) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parser->failed = true; | 
					
						
							|  |  |  |         return "<invalid>"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ptr += offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (ptr < end && *ptr) | 
					
						
							|  |  |  |         ++ptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (*ptr) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parser->failed = true; | 
					
						
							|  |  |  |         return "<invalid>"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return (const char *)(parser->unstructured.ptr + offset); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-17 11:36:34 +02:00
										 |  |  | static void parse_fx_4_numeric_value(struct fx_parser *parser, uint32_t offset, | 
					
						
							|  |  |  |         const struct fx_4_binary_type *type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     unsigned int base_type, comp_count; | 
					
						
							|  |  |  |     size_t i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     base_type = (type->typeinfo >> FX_4_NUMERIC_BASE_TYPE_SHIFT) & 0xf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     comp_count = type->packed_size / sizeof(uint32_t); | 
					
						
							|  |  |  |     for (i = 0; i < comp_count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         union hlsl_constant_value_component value; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         fx_parser_read_unstructured(parser, &value, offset, sizeof(uint32_t)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (base_type == FX_4_NUMERIC_TYPE_FLOAT) | 
					
						
							| 
									
										
										
										
											2025-02-24 16:07:04 +01:00
										 |  |  |             vkd3d_string_buffer_print_f32(&parser->buffer, value.f); | 
					
						
							| 
									
										
										
										
											2024-10-17 11:36:34 +02:00
										 |  |  |         else if (base_type == FX_4_NUMERIC_TYPE_INT) | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "%d", value.i); | 
					
						
							|  |  |  |         else if (base_type == FX_4_NUMERIC_TYPE_UINT) | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "%u", value.u); | 
					
						
							|  |  |  |         else if (base_type == FX_4_NUMERIC_TYPE_BOOL) | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "%s", value.u ? "true" : "false" ); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "%#x", value.u); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (i < comp_count - 1) | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, ", "); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         offset += sizeof(uint32_t); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fx_4_parse_string_initializer(struct fx_parser *parser, uint32_t offset) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const char *str = fx_4_get_string(parser, offset); | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "\"%s\"", str); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fx_parse_fx_4_annotations(struct fx_parser *parser) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct fx_4_annotation | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t name; | 
					
						
							|  |  |  |         uint32_t type; | 
					
						
							|  |  |  |     } var; | 
					
						
							|  |  |  |     struct fx_4_binary_type type; | 
					
						
							|  |  |  |     const char *name, *type_name; | 
					
						
							|  |  |  |     uint32_t count, i, value; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |     if (parser->failed) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-17 11:36:34 +02:00
										 |  |  |     count = fx_parser_read_u32(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!count) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "\n"); | 
					
						
							|  |  |  |     parse_fx_print_indent(parser); | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "<\n"); | 
					
						
							|  |  |  |     parse_fx_start_indent(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_read_u32s(parser, &var, sizeof(var)); | 
					
						
							|  |  |  |         fx_parser_read_unstructured(parser, &type, var.type, sizeof(type)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         name = fx_4_get_string(parser, var.name); | 
					
						
							|  |  |  |         type_name = fx_4_get_string(parser, type.name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parse_fx_print_indent(parser); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "%s %s", type_name, name); | 
					
						
							|  |  |  |         if (type.element_count) | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "[%u]", type.element_count); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, " = "); | 
					
						
							|  |  |  |         if (type.element_count) | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "{ "); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (type.class == FX_4_TYPE_CLASS_NUMERIC) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             value = fx_parser_read_u32(parser); | 
					
						
							|  |  |  |             parse_fx_4_numeric_value(parser, value, &type); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else if (type.class == FX_4_TYPE_CLASS_OBJECT && type.typeinfo == FX_4_OBJECT_TYPE_STRING) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             uint32_t element_count = max(type.element_count, 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for (uint32_t j = 0; j < element_count; ++j) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 value = fx_parser_read_u32(parser); | 
					
						
							|  |  |  |                 fx_4_parse_string_initializer(parser, value); | 
					
						
							|  |  |  |                 if (j < element_count - 1) | 
					
						
							|  |  |  |                     vkd3d_string_buffer_printf(&parser->buffer, ", "); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             fx_parser_error(parser, VKD3D_SHADER_ERROR_FX_INVALID_DATA, | 
					
						
							| 
									
										
										
										
											2024-11-22 02:22:04 +01:00
										 |  |  |                     "Only numeric and string types are supported in annotations."); | 
					
						
							| 
									
										
										
										
											2024-10-17 11:36:34 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (type.element_count) | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, " }"); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, ";\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     parse_fx_end_indent(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parse_fx_print_indent(parser); | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, ">"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  | static void fx_parse_fx_4_numeric_variables(struct fx_parser *parser, uint32_t count) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  |     struct fx_4_numeric_variable | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         uint32_t name; | 
					
						
							|  |  |  |         uint32_t type; | 
					
						
							|  |  |  |         uint32_t semantic; | 
					
						
							|  |  |  |         uint32_t offset; | 
					
						
							|  |  |  |         uint32_t value; | 
					
						
							|  |  |  |         uint32_t flags; | 
					
						
							|  |  |  |     } var; | 
					
						
							|  |  |  |     const char *name, *semantic, *type_name; | 
					
						
							| 
									
										
										
										
											2024-10-17 11:36:34 +02:00
										 |  |  |     struct fx_4_binary_type type; | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  |     uint32_t i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_read_u32s(parser, &var, sizeof(var)); | 
					
						
							|  |  |  |         fx_parser_read_unstructured(parser, &type, var.type, sizeof(type)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         name = fx_4_get_string(parser, var.name); | 
					
						
							|  |  |  |         type_name = fx_4_get_string(parser, type.name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "    %s %s", type_name, name); | 
					
						
							|  |  |  |         if (type.element_count) | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "[%u]", type.element_count); | 
					
						
							| 
									
										
										
										
											2024-10-17 11:36:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  |         if (var.semantic) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             semantic = fx_4_get_string(parser, var.semantic); | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, " : %s", semantic); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2025-05-30 10:20:26 +02:00
										 |  |  |         if (var.flags & FX_4_HAS_EXPLICIT_BIND_POINT) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, " : packoffset(c%u.%c)", | 
					
						
							|  |  |  |                     var.offset / 16, "xyzw"[(var.offset % 16) / 4]); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-10-17 11:36:34 +02:00
										 |  |  |         fx_parse_fx_4_annotations(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  |         if (var.value) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, " = { "); | 
					
						
							| 
									
										
										
										
											2024-10-17 11:36:34 +02:00
										 |  |  |             parse_fx_4_numeric_value(parser, var.value, &type); | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  |             vkd3d_string_buffer_printf(&parser->buffer, " }"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, ";    // Offset: %u, size %u.\n", var.offset, type.unpacked_size); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  | static void fx_parse_buffers(struct fx_parser *parser) | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     struct fx_buffer | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t name; | 
					
						
							|  |  |  |         uint32_t size; | 
					
						
							|  |  |  |         uint32_t flags; | 
					
						
							|  |  |  |         uint32_t count; | 
					
						
							|  |  |  |         uint32_t bind_point; | 
					
						
							|  |  |  |     } buffer; | 
					
						
							|  |  |  |     const char *name; | 
					
						
							|  |  |  |     uint32_t i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  |     if (parser->failed) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |     for (i = 0; i < parser->buffer_count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_read_u32s(parser, &buffer, sizeof(buffer)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         name = fx_4_get_string(parser, buffer.name); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-17 11:36:34 +02:00
										 |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "cbuffer %s", name); | 
					
						
							| 
									
										
										
										
											2025-05-30 09:32:24 +02:00
										 |  |  |         if (buffer.bind_point != ~0u) | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, " : register(b%u)", buffer.bind_point); | 
					
						
							| 
									
										
										
										
											2024-10-17 11:36:34 +02:00
										 |  |  |         fx_parse_fx_4_annotations(parser); | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-17 11:36:34 +02:00
										 |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "\n{\n"); | 
					
						
							|  |  |  |         parse_fx_start_indent(parser); | 
					
						
							| 
									
										
										
										
											2024-10-10 21:47:00 +02:00
										 |  |  |         fx_parse_fx_4_numeric_variables(parser, buffer.count); | 
					
						
							| 
									
										
										
										
											2024-10-17 11:36:34 +02:00
										 |  |  |         parse_fx_end_indent(parser); | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "}\n\n"); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-22 02:17:25 +01:00
										 |  |  | static void fx_4_parse_shader_blob(struct fx_parser *parser, unsigned int object_type, const struct fx_5_shader *shader) | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     const void *data = NULL; | 
					
						
							| 
									
										
										
										
											2024-11-22 02:17:25 +01:00
										 |  |  |     uint32_t data_size; | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-22 09:46:56 +01:00
										 |  |  |     if (!shader->offset) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parse_fx_print_indent(parser); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "NULL"); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-22 02:17:25 +01:00
										 |  |  |     fx_parser_read_unstructured(parser, &data_size, shader->offset, sizeof(data_size)); | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |     if (data_size) | 
					
						
							| 
									
										
										
										
											2024-11-22 02:17:25 +01:00
										 |  |  |         data = fx_parser_get_unstructured_ptr(parser, shader->offset + 4, data_size); | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (!data) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-10 23:05:27 +01:00
										 |  |  |     fx_parse_shader_blob(parser, VKD3D_SHADER_SOURCE_DXBC_TPF, data, data_size); | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-22 02:17:25 +01:00
										 |  |  |     if (object_type == FX_4_OBJECT_TYPE_GEOMETRY_SHADER_SO && shader->sodecl[0]) | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "\n/* Stream output declaration: \"%s\" */", | 
					
						
							| 
									
										
										
										
											2024-11-22 02:17:25 +01:00
										 |  |  |                 fx_4_get_string(parser, shader->sodecl[0])); | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     else if (object_type == FX_5_OBJECT_TYPE_GEOMETRY_SHADER) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-11-22 02:17:25 +01:00
										 |  |  |         for (unsigned int i = 0; i < ARRAY_SIZE(shader->sodecl); ++i) | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2024-11-22 02:17:25 +01:00
										 |  |  |            if (shader->sodecl[i]) | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |                vkd3d_string_buffer_printf(&parser->buffer, "\n/* Stream output %u declaration: \"%s\" */", | 
					
						
							| 
									
										
										
										
											2024-11-22 02:17:25 +01:00
										 |  |  |                        i, fx_4_get_string(parser, shader->sodecl[i])); | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-11-22 02:17:25 +01:00
										 |  |  |         if (shader->sodecl_count) | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "\n/* Rasterized stream %u */", shader->rast_stream); | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-22 02:17:25 +01:00
										 |  |  | static void fx_4_parse_shader_initializer(struct fx_parser *parser, unsigned int object_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct fx_5_shader shader = { 0 }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (object_type) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         case FX_4_OBJECT_TYPE_PIXEL_SHADER: | 
					
						
							|  |  |  |         case FX_4_OBJECT_TYPE_VERTEX_SHADER: | 
					
						
							|  |  |  |         case FX_4_OBJECT_TYPE_GEOMETRY_SHADER: | 
					
						
							|  |  |  |             shader.offset = fx_parser_read_u32(parser); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case FX_4_OBJECT_TYPE_GEOMETRY_SHADER_SO: | 
					
						
							|  |  |  |             shader.offset    = fx_parser_read_u32(parser); | 
					
						
							|  |  |  |             shader.sodecl[0] = fx_parser_read_u32(parser); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case FX_5_OBJECT_TYPE_GEOMETRY_SHADER: | 
					
						
							|  |  |  |         case FX_5_OBJECT_TYPE_COMPUTE_SHADER: | 
					
						
							|  |  |  |         case FX_5_OBJECT_TYPE_HULL_SHADER: | 
					
						
							|  |  |  |         case FX_5_OBJECT_TYPE_DOMAIN_SHADER: | 
					
						
							|  |  |  |             fx_parser_read_u32s(parser, &shader, sizeof(shader)); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             parser->failed = true; | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fx_4_parse_shader_blob(parser, object_type, &shader); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-05 13:43:28 +01:00
										 |  |  | static bool fx_4_object_has_initializer(const struct fx_4_binary_type *type) | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     switch (type->typeinfo) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-11-05 13:43:28 +01:00
										 |  |  |         case FX_4_OBJECT_TYPE_STRING: | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |         case FX_4_OBJECT_TYPE_BLEND_STATE: | 
					
						
							|  |  |  |         case FX_4_OBJECT_TYPE_DEPTH_STENCIL_STATE: | 
					
						
							|  |  |  |         case FX_4_OBJECT_TYPE_RASTERIZER_STATE: | 
					
						
							|  |  |  |         case FX_4_OBJECT_TYPE_SAMPLER_STATE: | 
					
						
							| 
									
										
										
										
											2024-11-05 13:43:28 +01:00
										 |  |  |         case FX_4_OBJECT_TYPE_PIXEL_SHADER: | 
					
						
							|  |  |  |         case FX_4_OBJECT_TYPE_VERTEX_SHADER: | 
					
						
							|  |  |  |         case FX_4_OBJECT_TYPE_GEOMETRY_SHADER: | 
					
						
							|  |  |  |         case FX_4_OBJECT_TYPE_GEOMETRY_SHADER_SO: | 
					
						
							|  |  |  |         case FX_5_OBJECT_TYPE_GEOMETRY_SHADER: | 
					
						
							|  |  |  |         case FX_5_OBJECT_TYPE_COMPUTE_SHADER: | 
					
						
							|  |  |  |         case FX_5_OBJECT_TYPE_HULL_SHADER: | 
					
						
							|  |  |  |         case FX_5_OBJECT_TYPE_DOMAIN_SHADER: | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  |             return true; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | static int fx_4_state_id_compare(const void *a, const void *b) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-06-16 08:36:17 +02:00
										 |  |  |     const struct fx_state *state = b; | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |     int id = *(int *)a; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return id - state->id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | static const struct | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t opcode; | 
					
						
							|  |  |  |     const char *name; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2025-06-07 16:44:16 +02:00
										 |  |  | fxlc_opcodes[] = | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     { 0x100, "mov"   }, | 
					
						
							|  |  |  |     { 0x101, "neg"   }, | 
					
						
							|  |  |  |     { 0x103, "rcp"   }, | 
					
						
							|  |  |  |     { 0x104, "frc"   }, | 
					
						
							|  |  |  |     { 0x105, "exp"   }, | 
					
						
							|  |  |  |     { 0x106, "log"   }, | 
					
						
							|  |  |  |     { 0x107, "rsq"   }, | 
					
						
							|  |  |  |     { 0x108, "sin"   }, | 
					
						
							|  |  |  |     { 0x109, "cos"   }, | 
					
						
							|  |  |  |     { 0x10a, "asin"  }, | 
					
						
							|  |  |  |     { 0x10b, "acos"  }, | 
					
						
							|  |  |  |     { 0x10c, "atan"  }, | 
					
						
							|  |  |  |     { 0x112, "sqrt"  }, | 
					
						
							|  |  |  |     { 0x120, "ineg"  }, | 
					
						
							|  |  |  |     { 0x121, "not"   }, | 
					
						
							|  |  |  |     { 0x130, "itof"  }, | 
					
						
							|  |  |  |     { 0x131, "utof"  }, | 
					
						
							|  |  |  |     { 0x133, "ftou"  }, | 
					
						
							|  |  |  |     { 0x137, "ftob"  }, | 
					
						
							|  |  |  |     { 0x139, "floor" }, | 
					
						
							|  |  |  |     { 0x13a, "ceil"  }, | 
					
						
							|  |  |  |     { 0x200, "min"   }, | 
					
						
							|  |  |  |     { 0x201, "max"   }, | 
					
						
							| 
									
										
										
										
											2025-06-04 11:17:38 +02:00
										 |  |  |     { 0x202, "lt"    }, | 
					
						
							|  |  |  |     { 0x203, "ge"    }, | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |     { 0x204, "add"   }, | 
					
						
							|  |  |  |     { 0x205, "mul"   }, | 
					
						
							|  |  |  |     { 0x206, "atan2" }, | 
					
						
							|  |  |  |     { 0x208, "div"   }, | 
					
						
							|  |  |  |     { 0x210, "bilt"  }, | 
					
						
							|  |  |  |     { 0x211, "bige"  }, | 
					
						
							|  |  |  |     { 0x212, "bieq"  }, | 
					
						
							|  |  |  |     { 0x213, "bine"  }, | 
					
						
							|  |  |  |     { 0x214, "buge"  }, | 
					
						
							|  |  |  |     { 0x215, "bult"  }, | 
					
						
							|  |  |  |     { 0x216, "iadd"  }, | 
					
						
							|  |  |  |     { 0x219, "imul"  }, | 
					
						
							|  |  |  |     { 0x21a, "udiv"  }, | 
					
						
							|  |  |  |     { 0x21d, "imin"  }, | 
					
						
							|  |  |  |     { 0x21e, "imax"  }, | 
					
						
							|  |  |  |     { 0x21f, "umin"  }, | 
					
						
							|  |  |  |     { 0x220, "umax"  }, | 
					
						
							|  |  |  |     { 0x230, "and"   }, | 
					
						
							|  |  |  |     { 0x231, "or"    }, | 
					
						
							|  |  |  |     { 0x233, "xor"   }, | 
					
						
							|  |  |  |     { 0x234, "ishl"  }, | 
					
						
							|  |  |  |     { 0x235, "ishr"  }, | 
					
						
							|  |  |  |     { 0x236, "ushr"  }, | 
					
						
							|  |  |  |     { 0x301, "movc"  }, | 
					
						
							|  |  |  |     { 0x500, "dot"   }, | 
					
						
							| 
									
										
										
										
											2025-06-12 00:20:14 +02:00
										 |  |  |     { 0x502, "noise" }, | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |     { 0x70e, "d3ds_dotswiz" }, | 
					
						
							| 
									
										
										
										
											2025-06-12 00:20:14 +02:00
										 |  |  |     { 0x711, "d3ds_noiseswiz" }, | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 16:44:16 +02:00
										 |  |  | static const char *get_fxlc_opcode_name(uint32_t opcode) | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     size_t i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 16:44:16 +02:00
										 |  |  |     for (i = 0; i < ARRAY_SIZE(fxlc_opcodes); ++i) | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-06-07 16:44:16 +02:00
										 |  |  |         if (fxlc_opcodes[i].opcode == opcode) | 
					
						
							|  |  |  |             return fxlc_opcodes[i].name; | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return "<unrecognized>"; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct fx_4_ctab_entry | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t name; | 
					
						
							|  |  |  |     uint16_t register_set; | 
					
						
							|  |  |  |     uint16_t register_index; | 
					
						
							|  |  |  |     uint16_t register_count; | 
					
						
							|  |  |  |     uint16_t reserved; | 
					
						
							|  |  |  |     uint32_t typeinfo; | 
					
						
							|  |  |  |     uint32_t default_value; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 20:04:24 +02:00
										 |  |  | struct fxlc_arg | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t reg_type; | 
					
						
							|  |  |  |     uint32_t address; | 
					
						
							|  |  |  |     bool indexed; | 
					
						
							|  |  |  |     struct | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t reg_type; | 
					
						
							|  |  |  |         uint32_t address; | 
					
						
							|  |  |  |     } index; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | struct fxlvm_code | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-06-07 20:04:24 +02:00
										 |  |  |     const uint32_t *ptr, *end; | 
					
						
							|  |  |  |     bool failed; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 20:17:06 +02:00
										 |  |  |     union | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         const float *_4; | 
					
						
							| 
									
										
										
										
											2025-06-07 23:06:58 +02:00
										 |  |  |         const double *_8; | 
					
						
							| 
									
										
										
										
											2025-06-07 20:17:06 +02:00
										 |  |  |     } cli; | 
					
						
							|  |  |  |     uint32_t cli_count; | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const struct fx_4_ctab_entry *constants; | 
					
						
							|  |  |  |     uint32_t ctab_offset; | 
					
						
							|  |  |  |     uint32_t ctab_count; | 
					
						
							|  |  |  |     const char *ctab; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned int comp_count; | 
					
						
							|  |  |  |     bool scalar; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 20:04:24 +02:00
										 |  |  | static uint32_t fxlvm_read_u32(struct fxlvm_code *code) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (code->end == code->ptr) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         code->failed = true; | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return *code->ptr++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 23:06:58 +02:00
										 |  |  | static const uint32_t *find_d3dbc_section(const uint32_t *ptr, uint32_t count, uint32_t tag, uint32_t *size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!count) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Skip version tag */ | 
					
						
							|  |  |  |     ptr++; | 
					
						
							|  |  |  |     count--; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (count > 2 && (*ptr & 0xffff) == 0xfffe) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         unsigned int section_size; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         section_size = (*ptr >> 16); | 
					
						
							|  |  |  |         if (!section_size || section_size + 1 > count) | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         if (*(ptr + 1) == tag) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             *size = section_size; | 
					
						
							|  |  |  |             return ptr + 2; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         count -= section_size + 1; | 
					
						
							|  |  |  |         ptr += section_size + 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fx_parse_print_swizzle(struct fx_parser *parser, const struct fxlvm_code *code, unsigned int addr) | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     unsigned int comp_count = code->scalar ? 1 : code->comp_count; | 
					
						
							|  |  |  |     static const char comp[] = "xyzw"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (comp_count < 4) | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, ".%.*s", comp_count, &comp[addr % 4]); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-13 12:01:19 +02:00
										 |  |  | static void fx_print_fxlc_register(struct fx_parser *parser, uint32_t reg_type, | 
					
						
							|  |  |  |         uint32_t address, uint32_t index_type, uint32_t index_address, struct fxlvm_code *code) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     static const char *table_names[FX_FXLC_REG_MAX + 1] = | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         [FX_FXLC_REG_LITERAL] = "imm", | 
					
						
							|  |  |  |         [FX_FXLC_REG_CB] = "c", | 
					
						
							|  |  |  |         [FX_FXLC_REG_INPUT] = "i", | 
					
						
							|  |  |  |         [FX_FXLC_REG_OUTPUT] = "expr", | 
					
						
							|  |  |  |         [FX_FXLC_REG_TEMP] = "r", | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     uint32_t reg_index = address / 4; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (parser->source_type == VKD3D_SHADER_SOURCE_TX | 
					
						
							|  |  |  |             && (reg_type == FX_FXLC_REG_INPUT || reg_type == FX_FXLC_REG_OUTPUT)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (reg_type == FX_FXLC_REG_INPUT) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (reg_index == 0) | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, "vPos"); | 
					
						
							|  |  |  |             else if (reg_index == 1) | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, "vPSize"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "oC%u", reg_index); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "%s%u", table_names[reg_type], reg_index); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (index_type != FX_FXLC_REG_UNUSED) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "[%s%u.%c]", table_names[index_type], | 
					
						
							|  |  |  |                 index_address / 4, "xyzw"[index_address % 4]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     fx_parse_print_swizzle(parser, code, address); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 20:55:09 +02:00
										 |  |  | static void fx_parse_fxlc_constant_argument(struct fx_parser *parser, | 
					
						
							| 
									
										
										
										
											2025-06-07 20:04:24 +02:00
										 |  |  |         const struct fxlc_arg *arg, const struct fxlvm_code *code) | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-06-07 23:06:58 +02:00
										 |  |  |     uint32_t register_index = arg->address / 4; /* Address counts in components. */ | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 23:06:58 +02:00
										 |  |  |     if (code->ctab_count) | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-06-07 23:06:58 +02:00
										 |  |  |         uint32_t i, offset; | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 23:06:58 +02:00
										 |  |  |         for (i = 0; i < code->ctab_count; ++i) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             const struct fx_4_ctab_entry *c = &code->constants[i]; | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 23:06:58 +02:00
										 |  |  |             if (register_index < c->register_index || register_index - c->register_index >= c->register_count) | 
					
						
							|  |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 23:06:58 +02:00
										 |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "%s", &code->ctab[c->name]); | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 23:06:58 +02:00
										 |  |  |             /* Register offset within variable */ | 
					
						
							|  |  |  |             offset = arg->address - c->register_index * 4; | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 23:06:58 +02:00
										 |  |  |             if (offset / 4) | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, "[%u]", offset / 4); | 
					
						
							|  |  |  |             fx_parse_print_swizzle(parser, code, offset); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "(var-not-found)"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "c%u", register_index); | 
					
						
							|  |  |  |         fx_parse_print_swizzle(parser, code, arg->address); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 20:04:24 +02:00
										 |  |  | static void fx_parse_fxlc_argument(struct fx_parser *parser, struct fxlc_arg *arg, struct fxlvm_code *code) | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2025-06-07 20:04:24 +02:00
										 |  |  |     uint32_t flags; | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 20:04:24 +02:00
										 |  |  |     memset(arg, 0, sizeof(*arg)); | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 20:04:24 +02:00
										 |  |  |     flags = fxlvm_read_u32(code); | 
					
						
							|  |  |  |     if (flags) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         arg->indexed = true; | 
					
						
							|  |  |  |         arg->index.reg_type = fxlvm_read_u32(code); | 
					
						
							|  |  |  |         arg->index.address  = fxlvm_read_u32(code); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     arg->reg_type = fxlvm_read_u32(code); | 
					
						
							|  |  |  |     arg->address  = fxlvm_read_u32(code); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 20:53:20 +02:00
										 |  |  | static void fx_print_fxlc_literal(struct fx_parser *parser, uint32_t address, struct fxlvm_code *code) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-06-07 23:06:58 +02:00
										 |  |  |     if (parser->version.major >= 4) | 
					
						
							|  |  |  |         vkd3d_string_buffer_print_f32(&parser->buffer, code->cli._4[address]); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         vkd3d_string_buffer_print_f64(&parser->buffer, code->cli._8[address]); | 
					
						
							| 
									
										
										
										
											2025-06-07 20:53:20 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 20:04:24 +02:00
										 |  |  | static void fx_print_fxlc_argument(struct fx_parser *parser, const struct fxlc_arg *arg, struct fxlvm_code *code) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t count; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-09 16:50:31 +02:00
										 |  |  |     if (arg->reg_type > FX_FXLC_REG_MAX) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_error(parser, VKD3D_SHADER_ERROR_FX_INVALID_DATA, | 
					
						
							|  |  |  |                 "Unexpected register type %u.", arg->reg_type); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (arg->index.reg_type > FX_FXLC_REG_MAX) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_error(parser, VKD3D_SHADER_ERROR_FX_INVALID_DATA, | 
					
						
							|  |  |  |                 "Unexpected index register type %u.", arg->index.reg_type); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (arg->indexed) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-06-13 12:01:19 +02:00
										 |  |  |         fx_print_fxlc_register(parser, arg->reg_type, arg->address, arg->index.reg_type, | 
					
						
							|  |  |  |                 arg->index.address, code); | 
					
						
							| 
									
										
										
										
											2025-06-09 16:50:31 +02:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 20:04:24 +02:00
										 |  |  |     switch (arg->reg_type) | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-06-07 16:33:57 +02:00
										 |  |  |         case FX_FXLC_REG_LITERAL: | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |             count = code->scalar ? 1 : code->comp_count; | 
					
						
							| 
									
										
										
										
											2025-06-07 20:17:06 +02:00
										 |  |  |             if (arg->address >= code->cli_count || count > code->cli_count - arg->address) | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |             { | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, "(<out-of-bounds>)"); | 
					
						
							|  |  |  |                 parser->failed = true; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "("); | 
					
						
							| 
									
										
										
										
											2025-06-07 20:53:20 +02:00
										 |  |  |             fx_print_fxlc_literal(parser, arg->address, code); | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |             for (unsigned int i = 1; i < code->comp_count; ++i) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, ", "); | 
					
						
							| 
									
										
										
										
											2025-06-07 20:53:20 +02:00
										 |  |  |                 fx_print_fxlc_literal(parser, arg->address + (code->scalar ? 0 : i), code); | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, ")"); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 16:33:57 +02:00
										 |  |  |         case FX_FXLC_REG_CB: | 
					
						
							| 
									
										
										
										
											2025-06-07 20:55:09 +02:00
										 |  |  |             fx_parse_fxlc_constant_argument(parser, arg, code); | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-13 12:01:19 +02:00
										 |  |  |         case FX_FXLC_REG_INPUT: | 
					
						
							| 
									
										
										
										
											2025-06-07 16:33:57 +02:00
										 |  |  |         case FX_FXLC_REG_OUTPUT: | 
					
						
							|  |  |  |         case FX_FXLC_REG_TEMP: | 
					
						
							| 
									
										
										
										
											2025-06-13 12:01:19 +02:00
										 |  |  |             fx_print_fxlc_register(parser, arg->reg_type, arg->address, FX_FXLC_REG_UNUSED, 0, code); | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         default: | 
					
						
							| 
									
										
										
										
											2025-06-07 20:04:24 +02:00
										 |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "<unknown register %u>", arg->reg_type); | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 20:04:24 +02:00
										 |  |  | static void fx_parse_fxlvm_expression(struct fx_parser *parser, struct fxlvm_code *code) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct fxlc_arg args[9]; | 
					
						
							|  |  |  |     uint32_t ins_count; | 
					
						
							|  |  |  |     size_t i, j; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parse_fx_start_indent(parser); | 
					
						
							| 
									
										
										
										
											2025-06-12 00:38:10 +02:00
										 |  |  |     if (parser->source_type == VKD3D_SHADER_SOURCE_TX) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parse_fx_print_indent(parser); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "tx_1_0\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ins_count = fxlvm_read_u32(code); | 
					
						
							| 
									
										
										
										
											2025-06-07 20:04:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < ins_count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t instr, opcode, src_count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         instr = fxlvm_read_u32(code); | 
					
						
							|  |  |  |         src_count = fxlvm_read_u32(code); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (src_count >= ARRAY_SIZE(args)) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             fx_parser_error(parser, VKD3D_SHADER_ERROR_FX_INVALID_DATA, | 
					
						
							|  |  |  |                     "Unexpected instruction source count %u.", src_count); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         /* Sources entries are followed by the destination, first read them all.
 | 
					
						
							|  |  |  |            Output format is "opcode dst, src[0]...src[n]". */ | 
					
						
							|  |  |  |         for (j = 0; j < src_count; ++j) | 
					
						
							|  |  |  |             fx_parse_fxlc_argument(parser, &args[j], code); | 
					
						
							|  |  |  |         fx_parse_fxlc_argument(parser, &args[src_count], code); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         opcode = (instr >> FX_FXLC_OPCODE_SHIFT) & FX_FXLC_OPCODE_MASK; | 
					
						
							|  |  |  |         code->comp_count = instr & FX_FXLC_COMP_COUNT_MASK; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parse_fx_print_indent(parser); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "%s ", get_fxlc_opcode_name(opcode)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         code->scalar = false; | 
					
						
							|  |  |  |         fx_print_fxlc_argument(parser, &args[src_count], code); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, ", "); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (j = 0; j < src_count; ++j) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             /* Scalar modifier applies only to the first source. */ | 
					
						
							|  |  |  |             code->scalar = j == 0 && !!(instr & FX_FXLC_IS_SCALAR_MASK); | 
					
						
							|  |  |  |             fx_print_fxlc_argument(parser, &args[j], code); | 
					
						
							|  |  |  |             if (j < src_count - 1) | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, ", "); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parse_fx_end_indent(parser); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 23:06:58 +02:00
										 |  |  | static void fx_2_parse_fxlvm_expression(struct fx_parser *parser, const uint32_t *blob, uint32_t size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t count = size / sizeof(uint32_t); | 
					
						
							|  |  |  |     struct fxlvm_code code = { 0 }; | 
					
						
							|  |  |  |     uint32_t section_size; | 
					
						
							|  |  |  |     const uint32_t *data; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 23:43:50 +02:00
										 |  |  |     if (!blob) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 23:06:58 +02:00
										 |  |  |     /* Literal constants, using 64-bit floats. */ | 
					
						
							|  |  |  |     if ((data = find_d3dbc_section(blob, count, TAG_CLIT, §ion_size))) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         code.cli_count = *data++; | 
					
						
							|  |  |  |         code.cli._8 = (const double *)data; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* CTAB does not contain variable names */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Code blob */ | 
					
						
							|  |  |  |     code.ptr = find_d3dbc_section(blob, count, TAG_FXLC, &count); | 
					
						
							|  |  |  |     code.end = code.ptr + count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fx_parse_fxlvm_expression(parser, &code); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | static void fx_4_parse_fxlvm_expression(struct fx_parser *parser, uint32_t offset) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct vkd3d_shader_dxbc_section_desc *section, fxlc, cli4, ctab; | 
					
						
							|  |  |  |     struct vkd3d_shader_dxbc_desc dxbc_desc; | 
					
						
							| 
									
										
										
										
											2025-06-07 20:04:24 +02:00
										 |  |  |     struct fxlvm_code code = { 0 }; | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |     struct vkd3d_shader_code dxbc; | 
					
						
							| 
									
										
										
										
											2025-06-07 20:04:24 +02:00
										 |  |  |     uint32_t size; | 
					
						
							|  |  |  |     size_t i; | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     offset = fx_parser_read_unstructured(parser, &size, offset, sizeof(size)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     dxbc.size = size; | 
					
						
							|  |  |  |     dxbc.code = fx_parser_get_unstructured_ptr(parser, offset, size); | 
					
						
							|  |  |  |     if (!dxbc.code) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (vkd3d_shader_parse_dxbc(&dxbc, 0, &dxbc_desc, NULL) < 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parser->failed = true; | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     memset(&fxlc, 0, sizeof(fxlc)); | 
					
						
							|  |  |  |     memset(&cli4, 0, sizeof(cli4)); | 
					
						
							|  |  |  |     memset(&ctab, 0, sizeof(ctab)); | 
					
						
							|  |  |  |     for (i = 0; i < dxbc_desc.section_count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         section = &dxbc_desc.sections[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (section->tag == TAG_FXLC) | 
					
						
							|  |  |  |             fxlc = *section; | 
					
						
							|  |  |  |         else if (section->tag == TAG_CLI4) | 
					
						
							|  |  |  |             cli4 = *section; | 
					
						
							|  |  |  |         else if (section->tag == TAG_CTAB) | 
					
						
							|  |  |  |             ctab = *section; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_shader_free_dxbc(&dxbc_desc); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (cli4.data.code) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t cli4_offset = offset + (size_t)cli4.data.code - (size_t)dxbc.code; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 20:17:06 +02:00
										 |  |  |         fx_parser_read_unstructured(parser, &code.cli_count, cli4_offset, sizeof(code.cli_count)); | 
					
						
							|  |  |  |         code.cli._4 = fx_parser_get_unstructured_ptr(parser, cli4_offset + 4, code.cli_count * sizeof(float)); | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ctab.data.code) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t ctab_offset = offset + (size_t)ctab.data.code - (size_t)dxbc.code; | 
					
						
							|  |  |  |         uint32_t consts_offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         fx_parser_read_unstructured(parser, &code.ctab_count, ctab_offset + 12, sizeof(code.ctab_count)); | 
					
						
							|  |  |  |         fx_parser_read_unstructured(parser, &consts_offset, ctab_offset + 16, sizeof(consts_offset)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         code.ctab = ctab.data.code; | 
					
						
							|  |  |  |         code.constants = fx_parser_get_unstructured_ptr(parser, | 
					
						
							|  |  |  |                 ctab_offset + consts_offset, code.ctab_count * sizeof(*code.constants)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 20:04:24 +02:00
										 |  |  |     code.ptr = fxlc.data.code; | 
					
						
							|  |  |  |     code.end = (uint32_t *)((uint8_t *)fxlc.data.code + fxlc.data.size); | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-07 20:04:24 +02:00
										 |  |  |     fx_parse_fxlvm_expression(parser, &code); | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | static void fx_4_parse_state_object_initializer(struct fx_parser *parser, uint32_t count, | 
					
						
							|  |  |  |         enum hlsl_type_class type_class) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-03-06 11:31:30 +01:00
										 |  |  |     struct fx_assignment entry; | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |     struct | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t name; | 
					
						
							|  |  |  |         uint32_t index; | 
					
						
							|  |  |  |     } index; | 
					
						
							|  |  |  |     struct | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t type; | 
					
						
							|  |  |  |         union | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             uint32_t u; | 
					
						
							|  |  |  |             float f; | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  |     } value; | 
					
						
							| 
									
										
										
										
											2025-03-03 19:17:23 +01:00
										 |  |  |     static const char *const value_types[FX_COMPONENT_TYPE_COUNT] = | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         [FX_BOOL]  = "bool", | 
					
						
							|  |  |  |         [FX_FLOAT] = "float", | 
					
						
							|  |  |  |         [FX_UINT]  = "uint", | 
					
						
							|  |  |  |         [FX_UINT8] = "byte", | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     const struct rhs_named_value *named_value; | 
					
						
							| 
									
										
										
										
											2024-11-22 02:17:25 +01:00
										 |  |  |     struct fx_5_shader shader = { 0 }; | 
					
						
							| 
									
										
										
										
											2025-06-16 08:36:17 +02:00
										 |  |  |     struct fx_state_table table; | 
					
						
							| 
									
										
										
										
											2024-11-22 02:17:25 +01:00
										 |  |  |     unsigned int shader_type = 0; | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |     uint32_t i, j, comp_count; | 
					
						
							| 
									
										
										
										
											2025-06-16 08:36:17 +02:00
										 |  |  |     struct fx_state *state; | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 10:52:43 +02:00
										 |  |  |     table = fx_get_state_table(type_class, parser->version.major, parser->version.minor); | 
					
						
							| 
									
										
										
										
											2025-05-06 01:07:03 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |     for (i = 0; i < count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_read_u32s(parser, &entry, sizeof(entry)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-06 01:07:03 +02:00
										 |  |  |         if (!(state = bsearch(&entry.id, table.ptr, table.count, sizeof(*table.ptr), fx_4_state_id_compare))) | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2024-11-22 02:22:04 +01:00
										 |  |  |             fx_parser_error(parser, VKD3D_SHADER_ERROR_FX_INVALID_DATA, "Unrecognized state id %#x.", entry.id); | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (state->container != type_class) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             fx_parser_error(parser, VKD3D_SHADER_ERROR_FX_INVALID_DATA, | 
					
						
							|  |  |  |                     "State '%s' does not belong to object type class %#x.", state->name, type_class); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parse_fx_print_indent(parser); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "%s", state->name); | 
					
						
							|  |  |  |         if (state->array_size > 1) | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "[%u]", entry.lhs_index); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, " = "); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         switch (entry.type) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             case FX_4_ASSIGNMENT_CONSTANT: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (value_types[state->type]) | 
					
						
							|  |  |  |                     vkd3d_string_buffer_printf(&parser->buffer, "%s", value_types[state->type]); | 
					
						
							|  |  |  |                 if (state->dimx > 1) | 
					
						
							|  |  |  |                     vkd3d_string_buffer_printf(&parser->buffer, "%u", state->dimx); | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, "("); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 fx_parser_read_unstructured(parser, &comp_count, entry.value, sizeof(uint32_t)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 named_value = NULL; | 
					
						
							|  |  |  |                 if (comp_count == 1 && state->values && (state->type == FX_UINT || state->type == FX_BOOL)) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     const struct rhs_named_value *ptr = state->values; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     fx_parser_read_unstructured(parser, &value, entry.value + 4, sizeof(value)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     while (ptr->name) | 
					
						
							|  |  |  |                     { | 
					
						
							|  |  |  |                         if (value.u == ptr->value) | 
					
						
							|  |  |  |                         { | 
					
						
							|  |  |  |                             named_value = ptr; | 
					
						
							|  |  |  |                             break; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         ++ptr; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (named_value) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     vkd3d_string_buffer_printf(&parser->buffer, "%s /* %u */", named_value->name, named_value->value); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     uint32_t offset = entry.value + 4; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     for (j = 0; j < comp_count; ++j, offset += sizeof(value)) | 
					
						
							|  |  |  |                     { | 
					
						
							|  |  |  |                         fx_parser_read_unstructured(parser, &value, offset, sizeof(value)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         if (state->type == FX_UINT8) | 
					
						
							|  |  |  |                             vkd3d_string_buffer_printf(&parser->buffer, "0x%.2x", value.u); | 
					
						
							|  |  |  |                         else if (state->type == FX_UINT) | 
					
						
							|  |  |  |                             vkd3d_string_buffer_printf(&parser->buffer, "%u", value.u); | 
					
						
							| 
									
										
										
										
											2025-07-08 19:25:15 +02:00
										 |  |  |                         else if (state->type == FX_BOOL) | 
					
						
							|  |  |  |                             vkd3d_string_buffer_printf(&parser->buffer, "%s", value.u ? "true" : "false"); | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |                         else if (state->type == FX_FLOAT) | 
					
						
							|  |  |  |                             vkd3d_string_buffer_printf(&parser->buffer, "%g", value.f); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         if (comp_count > 1 && j < comp_count - 1) | 
					
						
							|  |  |  |                             vkd3d_string_buffer_printf(&parser->buffer, ", "); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, ")"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case FX_4_ASSIGNMENT_VARIABLE: | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, "%s", fx_4_get_string(parser, entry.value)); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case FX_4_ASSIGNMENT_ARRAY_CONSTANT_INDEX: | 
					
						
							|  |  |  |                 fx_parser_read_unstructured(parser, &index, entry.value, sizeof(index)); | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, "%s[%u]", fx_4_get_string(parser, index.name), index.index); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case FX_4_ASSIGNMENT_ARRAY_VARIABLE_INDEX: | 
					
						
							|  |  |  |                 fx_parser_read_unstructured(parser, &index, entry.value, sizeof(index)); | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, "%s[%s]", fx_4_get_string(parser, index.name), | 
					
						
							|  |  |  |                         fx_4_get_string(parser, index.index)); | 
					
						
							|  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2025-02-21 23:57:50 +01:00
										 |  |  |             case FX_4_ASSIGNMENT_INDEX_EXPRESSION: | 
					
						
							|  |  |  |                 fx_parser_read_unstructured(parser, &index, entry.value, sizeof(index)); | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, "%s[eval(\n", fx_4_get_string(parser, index.name)); | 
					
						
							|  |  |  |                 fx_4_parse_fxlvm_expression(parser, index.index); | 
					
						
							|  |  |  |                 parse_fx_print_indent(parser); | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, ")]"); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case FX_4_ASSIGNMENT_VALUE_EXPRESSION: | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, "eval(\n"); | 
					
						
							|  |  |  |                 fx_4_parse_fxlvm_expression(parser, entry.value); | 
					
						
							|  |  |  |                 parse_fx_print_indent(parser); | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, ")"); | 
					
						
							|  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2024-11-22 02:17:25 +01:00
										 |  |  |             case FX_4_ASSIGNMENT_INLINE_SHADER: | 
					
						
							|  |  |  |             case FX_5_ASSIGNMENT_INLINE_SHADER: | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 bool shader5 = entry.type == FX_5_ASSIGNMENT_INLINE_SHADER; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (shader5) | 
					
						
							|  |  |  |                     fx_parser_read_unstructured(parser, &shader, entry.value, sizeof(shader)); | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                     fx_parser_read_unstructured(parser, &shader, entry.value, 2 * sizeof(uint32_t)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (state->type == FX_PIXELSHADER) | 
					
						
							|  |  |  |                     shader_type = FX_4_OBJECT_TYPE_PIXEL_SHADER; | 
					
						
							|  |  |  |                 else if (state->type == FX_VERTEXSHADER) | 
					
						
							|  |  |  |                     shader_type = FX_4_OBJECT_TYPE_VERTEX_SHADER; | 
					
						
							|  |  |  |                 else if (state->type == FX_GEOMETRYSHADER) | 
					
						
							|  |  |  |                     shader_type = shader5 ? FX_5_OBJECT_TYPE_GEOMETRY_SHADER : FX_4_OBJECT_TYPE_GEOMETRY_SHADER_SO; | 
					
						
							|  |  |  |                 else if (state->type == FX_HULLSHADER) | 
					
						
							|  |  |  |                     shader_type = FX_5_OBJECT_TYPE_HULL_SHADER; | 
					
						
							|  |  |  |                 else if (state->type == FX_DOMAINSHADER) | 
					
						
							|  |  |  |                     shader_type = FX_5_OBJECT_TYPE_DOMAIN_SHADER; | 
					
						
							|  |  |  |                 else if (state->type == FX_COMPUTESHADER) | 
					
						
							|  |  |  |                     shader_type = FX_5_OBJECT_TYPE_COMPUTE_SHADER; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, "\n"); | 
					
						
							|  |  |  |                 parse_fx_start_indent(parser); | 
					
						
							|  |  |  |                 fx_4_parse_shader_blob(parser, shader_type, &shader); | 
					
						
							|  |  |  |                 parse_fx_end_indent(parser); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |             default: | 
					
						
							|  |  |  |                 fx_parser_error(parser, VKD3D_SHADER_ERROR_FX_NOT_IMPLEMENTED, | 
					
						
							| 
									
										
										
										
											2024-11-22 02:22:04 +01:00
										 |  |  |                         "Unsupported assignment type %u.", entry.type); | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, ";\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  | static void fx_4_parse_object_initializer(struct fx_parser *parser, const struct fx_4_binary_type *type) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |     static const enum hlsl_type_class type_classes[] = | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         [FX_4_OBJECT_TYPE_BLEND_STATE]         = HLSL_CLASS_BLEND_STATE, | 
					
						
							|  |  |  |         [FX_4_OBJECT_TYPE_DEPTH_STENCIL_STATE] = HLSL_CLASS_DEPTH_STENCIL_STATE, | 
					
						
							|  |  |  |         [FX_4_OBJECT_TYPE_RASTERIZER_STATE]    = HLSL_CLASS_RASTERIZER_STATE, | 
					
						
							|  |  |  |         [FX_4_OBJECT_TYPE_SAMPLER_STATE]       = HLSL_CLASS_SAMPLER, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     unsigned int i, element_count, count; | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  |     uint32_t value; | 
					
						
							| 
									
										
										
										
											2025-02-22 09:46:56 +01:00
										 |  |  |     bool is_array; | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-05 13:43:28 +01:00
										 |  |  |     if (!fx_4_object_has_initializer(type)) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  |     vkd3d_string_buffer_printf(&parser->buffer, " = {\n"); | 
					
						
							|  |  |  |     element_count = max(type->element_count, 1); | 
					
						
							| 
									
										
										
										
											2025-02-22 09:46:56 +01:00
										 |  |  |     is_array = element_count > 1; | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  |     for (i = 0; i < element_count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         switch (type->typeinfo) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             case FX_4_OBJECT_TYPE_STRING: | 
					
						
							|  |  |  |                 vkd3d_string_buffer_printf(&parser->buffer, "    "); | 
					
						
							|  |  |  |                 value = fx_parser_read_u32(parser); | 
					
						
							|  |  |  |                 fx_4_parse_string_initializer(parser, value); | 
					
						
							|  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |             case FX_4_OBJECT_TYPE_BLEND_STATE: | 
					
						
							|  |  |  |             case FX_4_OBJECT_TYPE_DEPTH_STENCIL_STATE: | 
					
						
							|  |  |  |             case FX_4_OBJECT_TYPE_RASTERIZER_STATE: | 
					
						
							|  |  |  |             case FX_4_OBJECT_TYPE_SAMPLER_STATE: | 
					
						
							|  |  |  |                 count = fx_parser_read_u32(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-22 09:46:56 +01:00
										 |  |  |                 if (is_array) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     parse_fx_start_indent(parser); | 
					
						
							|  |  |  |                     parse_fx_print_indent(parser); | 
					
						
							|  |  |  |                     vkd3d_string_buffer_printf(&parser->buffer, "{\n"); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |                 parse_fx_start_indent(parser); | 
					
						
							|  |  |  |                 fx_4_parse_state_object_initializer(parser, count, type_classes[type->typeinfo]); | 
					
						
							|  |  |  |                 parse_fx_end_indent(parser); | 
					
						
							| 
									
										
										
										
											2025-02-22 09:46:56 +01:00
										 |  |  |                 if (is_array) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     parse_fx_print_indent(parser); | 
					
						
							|  |  |  |                     vkd3d_string_buffer_printf(&parser->buffer, "}"); | 
					
						
							|  |  |  |                     parse_fx_end_indent(parser); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  |             case FX_4_OBJECT_TYPE_PIXEL_SHADER: | 
					
						
							|  |  |  |             case FX_4_OBJECT_TYPE_VERTEX_SHADER: | 
					
						
							|  |  |  |             case FX_4_OBJECT_TYPE_GEOMETRY_SHADER: | 
					
						
							|  |  |  |             case FX_4_OBJECT_TYPE_GEOMETRY_SHADER_SO: | 
					
						
							|  |  |  |             case FX_5_OBJECT_TYPE_GEOMETRY_SHADER: | 
					
						
							|  |  |  |             case FX_5_OBJECT_TYPE_COMPUTE_SHADER: | 
					
						
							|  |  |  |             case FX_5_OBJECT_TYPE_HULL_SHADER: | 
					
						
							|  |  |  |             case FX_5_OBJECT_TYPE_DOMAIN_SHADER: | 
					
						
							|  |  |  |                 parse_fx_start_indent(parser); | 
					
						
							|  |  |  |                 fx_4_parse_shader_initializer(parser, type->typeinfo); | 
					
						
							|  |  |  |                 parse_fx_end_indent(parser); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 fx_parser_error(parser, VKD3D_SHADER_ERROR_FX_NOT_IMPLEMENTED, | 
					
						
							|  |  |  |                         "Parsing object type %u is not implemented.", type->typeinfo); | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2025-02-22 09:46:56 +01:00
										 |  |  |         vkd3d_string_buffer_printf(&parser->buffer, is_array ? ",\n" : "\n"); | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "}"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  | static void fx_4_parse_objects(struct fx_parser *parser) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct fx_4_object_variable | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t name; | 
					
						
							|  |  |  |         uint32_t type; | 
					
						
							|  |  |  |         uint32_t semantic; | 
					
						
							|  |  |  |         uint32_t bind_point; | 
					
						
							|  |  |  |     } var; | 
					
						
							| 
									
										
										
										
											2024-10-17 11:36:34 +02:00
										 |  |  |     struct fx_4_binary_type type; | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  |     const char *name, *type_name; | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  |     uint32_t i; | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (parser->failed) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < parser->object_count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |         if (parser->failed) | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  |         fx_parser_read_u32s(parser, &var, sizeof(var)); | 
					
						
							|  |  |  |         fx_parser_read_unstructured(parser, &type, var.type, sizeof(type)); | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  |         name = fx_4_get_string(parser, var.name); | 
					
						
							|  |  |  |         type_name = fx_4_get_string(parser, type.name); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "%s %s", type_name, name); | 
					
						
							|  |  |  |         if (type.element_count) | 
					
						
							|  |  |  |             vkd3d_string_buffer_printf(&parser->buffer, "[%u]", type.element_count); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-05 13:43:28 +01:00
										 |  |  |         fx_4_parse_object_initializer(parser, &type); | 
					
						
							| 
									
										
										
										
											2024-10-28 12:14:54 +01:00
										 |  |  |         vkd3d_string_buffer_printf(&parser->buffer, ";\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-28 00:38:28 +01:00
										 |  |  |         fx_parse_fx_4_annotations(parser); | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-19 18:12:52 +02:00
										 |  |  | static void fx_parse_fx_4_technique(struct fx_parser *parser) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct fx_technique | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t name; | 
					
						
							|  |  |  |         uint32_t count; | 
					
						
							|  |  |  |     } technique; | 
					
						
							|  |  |  |     struct fx_pass | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t name; | 
					
						
							|  |  |  |         uint32_t count; | 
					
						
							|  |  |  |     } pass; | 
					
						
							|  |  |  |     const char *name; | 
					
						
							|  |  |  |     uint32_t i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (parser->failed) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fx_parser_read_u32s(parser, &technique, sizeof(technique)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     name = fx_4_get_string(parser, technique.name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parse_fx_print_indent(parser); | 
					
						
							| 
									
										
										
										
											2025-05-06 01:07:03 +02:00
										 |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "technique%u %s", parser->version.major == 4 ? 10 : 11, name); | 
					
						
							| 
									
										
										
										
											2024-10-19 18:12:52 +02:00
										 |  |  |     fx_parse_fx_4_annotations(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "\n"); | 
					
						
							|  |  |  |     parse_fx_print_indent(parser); | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "{\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parse_fx_start_indent(parser); | 
					
						
							|  |  |  |     for (i = 0; i < technique.count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_read_u32s(parser, &pass, sizeof(pass)); | 
					
						
							|  |  |  |         name = fx_4_get_string(parser, pass.name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parse_fx_print_indent(parser); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "pass %s", name); | 
					
						
							|  |  |  |         fx_parse_fx_4_annotations(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "\n"); | 
					
						
							|  |  |  |         parse_fx_print_indent(parser); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "{\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-12 14:31:37 +01:00
										 |  |  |         parse_fx_start_indent(parser); | 
					
						
							|  |  |  |         fx_4_parse_state_object_initializer(parser, pass.count, HLSL_CLASS_PASS); | 
					
						
							|  |  |  |         parse_fx_end_indent(parser); | 
					
						
							| 
									
										
										
										
											2024-10-19 18:12:52 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         parse_fx_print_indent(parser); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "}\n\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parse_fx_end_indent(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parse_fx_print_indent(parser); | 
					
						
							|  |  |  |     vkd3d_string_buffer_printf(&parser->buffer, "}\n\n"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fx_parse_groups(struct fx_parser *parser) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct fx_group | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t name; | 
					
						
							|  |  |  |         uint32_t count; | 
					
						
							|  |  |  |     } group; | 
					
						
							|  |  |  |     const char *name; | 
					
						
							|  |  |  |     uint32_t i, j; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (parser->failed) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (i = 0; i < parser->group_count; ++i) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_read_u32s(parser, &group, sizeof(group)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         name = fx_4_get_string(parser, group.name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "fxgroup %s", name); | 
					
						
							|  |  |  |         fx_parse_fx_4_annotations(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "\n{\n"); | 
					
						
							|  |  |  |         parse_fx_start_indent(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (j = 0; j < group.count; ++j) | 
					
						
							|  |  |  |             fx_parse_fx_4_technique(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parse_fx_end_indent(parser); | 
					
						
							|  |  |  |         vkd3d_string_buffer_printf(&parser->buffer, "}\n\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-05 16:34:33 +01:00
										 |  |  | static void fx_4_parse(struct fx_parser *parser) | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     struct fx_4_header | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t version; | 
					
						
							|  |  |  |         uint32_t buffer_count; | 
					
						
							|  |  |  |         uint32_t numeric_variable_count; | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  |         uint32_t object_count; | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |         uint32_t shared_buffer_count; | 
					
						
							|  |  |  |         uint32_t shared_numeric_variable_count; | 
					
						
							|  |  |  |         uint32_t shared_object_count; | 
					
						
							|  |  |  |         uint32_t technique_count; | 
					
						
							|  |  |  |         uint32_t unstructured_size; | 
					
						
							|  |  |  |         uint32_t string_count; | 
					
						
							|  |  |  |         uint32_t texture_count; | 
					
						
							|  |  |  |         uint32_t depth_stencil_state_count; | 
					
						
							|  |  |  |         uint32_t blend_state_count; | 
					
						
							|  |  |  |         uint32_t rasterizer_state_count; | 
					
						
							|  |  |  |         uint32_t sampler_state_count; | 
					
						
							|  |  |  |         uint32_t rtv_count; | 
					
						
							|  |  |  |         uint32_t dsv_count; | 
					
						
							|  |  |  |         uint32_t shader_count; | 
					
						
							|  |  |  |         uint32_t inline_shader_count; | 
					
						
							|  |  |  |     } header; | 
					
						
							| 
									
										
										
										
											2024-10-19 18:12:52 +02:00
										 |  |  |     uint32_t i; | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     fx_parser_read_u32s(parser, &header, sizeof(header)); | 
					
						
							|  |  |  |     parser->buffer_count = header.buffer_count; | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  |     parser->object_count = header.object_count; | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (parser->end - parser->ptr < header.unstructured_size) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-03-05 16:40:17 +01:00
										 |  |  |         fx_parser_error(parser, VKD3D_SHADER_ERROR_FX_INVALID_SIZE, | 
					
						
							|  |  |  |                 "Invalid unstructured data size %u.", header.unstructured_size); | 
					
						
							| 
									
										
										
										
											2025-03-05 16:34:33 +01:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parser->unstructured.ptr = parser->ptr; | 
					
						
							|  |  |  |     parser->unstructured.end = parser->ptr + header.unstructured_size; | 
					
						
							|  |  |  |     parser->unstructured.size = header.unstructured_size; | 
					
						
							|  |  |  |     fx_parser_skip(parser, header.unstructured_size); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  |     fx_parse_buffers(parser); | 
					
						
							|  |  |  |     fx_4_parse_objects(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-19 18:12:52 +02:00
										 |  |  |     for (i = 0; i < header.technique_count; ++i) | 
					
						
							|  |  |  |         fx_parse_fx_4_technique(parser); | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-05 16:34:33 +01:00
										 |  |  | static void fx_5_parse(struct fx_parser *parser) | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     struct fx_5_header | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint32_t version; | 
					
						
							|  |  |  |         uint32_t buffer_count; | 
					
						
							|  |  |  |         uint32_t numeric_variable_count; | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  |         uint32_t object_count; | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |         uint32_t shared_buffer_count; | 
					
						
							|  |  |  |         uint32_t shared_numeric_variable_count; | 
					
						
							|  |  |  |         uint32_t shared_object_count; | 
					
						
							|  |  |  |         uint32_t technique_count; | 
					
						
							|  |  |  |         uint32_t unstructured_size; | 
					
						
							|  |  |  |         uint32_t string_count; | 
					
						
							|  |  |  |         uint32_t texture_count; | 
					
						
							|  |  |  |         uint32_t depth_stencil_state_count; | 
					
						
							|  |  |  |         uint32_t blend_state_count; | 
					
						
							|  |  |  |         uint32_t rasterizer_state_count; | 
					
						
							|  |  |  |         uint32_t sampler_state_count; | 
					
						
							|  |  |  |         uint32_t rtv_count; | 
					
						
							|  |  |  |         uint32_t dsv_count; | 
					
						
							|  |  |  |         uint32_t shader_count; | 
					
						
							|  |  |  |         uint32_t inline_shader_count; | 
					
						
							|  |  |  |         uint32_t group_count; | 
					
						
							|  |  |  |         uint32_t uav_count; | 
					
						
							|  |  |  |         uint32_t interface_variable_count; | 
					
						
							|  |  |  |         uint32_t interface_variable_element_count; | 
					
						
							|  |  |  |         uint32_t class_instance_element_count; | 
					
						
							|  |  |  |     } header; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fx_parser_read_u32s(parser, &header, sizeof(header)); | 
					
						
							|  |  |  |     parser->buffer_count = header.buffer_count; | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  |     parser->object_count = header.object_count; | 
					
						
							| 
									
										
										
										
											2024-10-19 18:12:52 +02:00
										 |  |  |     parser->group_count = header.group_count; | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (parser->end - parser->ptr < header.unstructured_size) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-03-05 16:40:17 +01:00
										 |  |  |         fx_parser_error(parser, VKD3D_SHADER_ERROR_FX_INVALID_SIZE, | 
					
						
							|  |  |  |                 "Invalid unstructured data size %u.", header.unstructured_size); | 
					
						
							| 
									
										
										
										
											2025-03-05 16:34:33 +01:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parser->unstructured.ptr = parser->ptr; | 
					
						
							|  |  |  |     parser->unstructured.end = parser->ptr + header.unstructured_size; | 
					
						
							|  |  |  |     parser->unstructured.size = header.unstructured_size; | 
					
						
							|  |  |  |     fx_parser_skip(parser, header.unstructured_size); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-10 22:54:50 +02:00
										 |  |  |     fx_parse_buffers(parser); | 
					
						
							|  |  |  |     fx_4_parse_objects(parser); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-19 18:12:52 +02:00
										 |  |  |     fx_parse_groups(parser); | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  | static void fx_parser_init(struct fx_parser *parser, const struct vkd3d_shader_compile_info *compile_info, | 
					
						
							|  |  |  |         struct vkd3d_shader_message_context *message_context) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     memset(parser, 0, sizeof(*parser)); | 
					
						
							| 
									
										
										
										
											2025-06-12 00:38:10 +02:00
										 |  |  |     parser->source_type = compile_info->source_type; | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  |     parser->start = compile_info->source.code; | 
					
						
							|  |  |  |     parser->ptr = compile_info->source.code; | 
					
						
							|  |  |  |     parser->end = (uint8_t *)compile_info->source.code + compile_info->source.size; | 
					
						
							|  |  |  |     parser->message_context = message_context; | 
					
						
							|  |  |  |     vkd3d_string_buffer_init(&parser->buffer); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void fx_parser_cleanup(struct fx_parser *parser) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     free(parser->objects.types); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  | int fx_parse(const struct vkd3d_shader_compile_info *compile_info, | 
					
						
							|  |  |  |         struct vkd3d_shader_code *out, struct vkd3d_shader_message_context *message_context) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  |     struct fx_parser parser; | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |     uint32_t version; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  |     fx_parser_init(&parser, compile_info, message_context); | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (parser.end - parser.start < sizeof(version)) | 
					
						
							| 
									
										
										
										
											2025-03-05 16:40:17 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         fx_parser_error(&parser, VKD3D_SHADER_ERROR_FX_INVALID_SIZE, | 
					
						
							|  |  |  |                 "Source size %zu is smaller than the FX header size.", compile_info->source.size); | 
					
						
							| 
									
										
										
										
											2025-03-05 16:34:33 +01:00
										 |  |  |         return VKD3D_ERROR_INVALID_SHADER; | 
					
						
							| 
									
										
										
										
											2025-03-05 16:40:17 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |     version = *(uint32_t *)parser.ptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (version) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         case 0xfeff0901: | 
					
						
							| 
									
										
										
										
											2025-05-06 01:07:03 +02:00
										 |  |  |             parser.version.major = 3; | 
					
						
							| 
									
										
										
										
											2025-03-05 16:34:33 +01:00
										 |  |  |             fx_2_parse(&parser); | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |         case 0xfeff1001: | 
					
						
							| 
									
										
										
										
											2025-05-06 01:07:03 +02:00
										 |  |  |             parser.version.major = 4; | 
					
						
							|  |  |  |             fx_4_parse(&parser); | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |         case 0xfeff1011: | 
					
						
							| 
									
										
										
										
											2025-05-06 01:07:03 +02:00
										 |  |  |             parser.version.major = 4; | 
					
						
							|  |  |  |             parser.version.minor = 1; | 
					
						
							| 
									
										
										
										
											2025-03-05 16:34:33 +01:00
										 |  |  |             fx_4_parse(&parser); | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |         case 0xfeff2001: | 
					
						
							| 
									
										
										
										
											2025-05-06 01:07:03 +02:00
										 |  |  |             parser.version.major = 5; | 
					
						
							| 
									
										
										
										
											2025-03-05 16:34:33 +01:00
										 |  |  |             fx_5_parse(&parser); | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |             break; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             fx_parser_error(&parser, VKD3D_SHADER_ERROR_FX_INVALID_VERSION, | 
					
						
							|  |  |  |                     "Invalid effect binary version value 0x%08x.", version); | 
					
						
							| 
									
										
										
										
											2025-03-05 16:34:33 +01:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_shader_code_from_string_buffer(out, &parser.buffer); | 
					
						
							| 
									
										
										
										
											2025-03-07 13:50:03 +01:00
										 |  |  |     fx_parser_cleanup(&parser); | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-05 16:34:33 +01:00
										 |  |  |     if (parser.failed) | 
					
						
							|  |  |  |         return VKD3D_ERROR_INVALID_SHADER; | 
					
						
							|  |  |  |     return VKD3D_OK; | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2025-06-12 00:38:10 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | int tx_parse(const struct vkd3d_shader_compile_info *compile_info, | 
					
						
							|  |  |  |         struct vkd3d_shader_code *out, struct vkd3d_shader_message_context *message_context) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct fx_parser parser; | 
					
						
							|  |  |  |     uint32_t version; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fx_parser_init(&parser, compile_info, message_context); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (parser.end - parser.start < sizeof(version)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         fx_parser_error(&parser, VKD3D_SHADER_ERROR_FX_INVALID_SIZE, | 
					
						
							|  |  |  |                 "Source size %zu is smaller than the TX header size.", compile_info->source.size); | 
					
						
							|  |  |  |         return VKD3D_ERROR_INVALID_SHADER; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     version = *(uint32_t *)parser.ptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (version) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         case 0x54580100: | 
					
						
							|  |  |  |             fx_2_parse_fxlvm_expression(&parser, (const uint32_t *)parser.ptr, parser.end - parser.ptr); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             fx_parser_error(&parser, VKD3D_SHADER_ERROR_FX_INVALID_VERSION, | 
					
						
							|  |  |  |                     "Invalid texture shader binary version value 0x%08x.", version); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vkd3d_shader_code_from_string_buffer(out, &parser.buffer); | 
					
						
							|  |  |  |     fx_parser_cleanup(&parser); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (parser.failed) | 
					
						
							|  |  |  |         return VKD3D_ERROR_INVALID_SHADER; | 
					
						
							|  |  |  |     return VKD3D_OK; | 
					
						
							|  |  |  | } |