| 
									
										
										
										
											2017-06-16 22:38:21 +02:00
										 |  |  |  | /*
 | 
					
						
							|  |  |  |  |  * Copyright 2017 Józef Kucia 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 "vkd3d_shader_private.h"
 | 
					
						
							| 
									
										
										
										
											2020-08-06 13:23:45 +04:30
										 |  |  |  | #include "vkd3d_version.h"
 | 
					
						
							| 
									
										
										
										
											2023-10-21 21:35:08 +02:00
										 |  |  |  | #include "hlsl.h"
 | 
					
						
							| 
									
										
										
										
											2017-06-16 22:38:21 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-31 11:29:27 +01:00
										 |  |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											2022-03-01 13:21:33 +01:00
										 |  |  |  | #include <math.h>
 | 
					
						
							| 
									
										
										
										
											2019-01-31 11:29:27 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-31 11:29:29 +01:00
										 |  |  |  | VKD3D_DEBUG_ENV_NAME("VKD3D_SHADER_DEBUG"); | 
					
						
							| 
									
										
										
										
											2019-01-31 11:29:28 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-11 16:39:08 +02:00
										 |  |  |  | static inline int char_to_int(char c) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if ('0' <= c && c <= '9') | 
					
						
							|  |  |  |  |         return c - '0'; | 
					
						
							|  |  |  |  |     if ('A' <= c && c <= 'F') | 
					
						
							|  |  |  |  |         return c - 'A' + 10; | 
					
						
							|  |  |  |  |     if ('a' <= c && c <= 'f') | 
					
						
							|  |  |  |  |         return c - 'a' + 10; | 
					
						
							|  |  |  |  |     return -1; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | uint32_t vkd3d_parse_integer(const char *s) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     uint32_t base = 10, ret = 0; | 
					
						
							|  |  |  |  |     int digit; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (*s == '0') | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         base = 8; | 
					
						
							|  |  |  |  |         ++s; | 
					
						
							|  |  |  |  |         if (*s == 'x' || *s == 'X') | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             base = 16; | 
					
						
							|  |  |  |  |             ++s; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     while ((digit = char_to_int(*s++)) >= 0) | 
					
						
							|  |  |  |  |         ret = ret * base + (uint32_t)digit; | 
					
						
							|  |  |  |  |     return ret; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:28 +02:00
										 |  |  |  | bool vkd3d_shader_source_list_append(struct vkd3d_shader_source_list *l, const char *source) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     char *s; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!(s = vkd3d_strdup(source))) | 
					
						
							|  |  |  |  |         return false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!vkd3d_array_reserve((void **)&l->sources, &l->capacity, l->count + 1, sizeof(*l->sources))) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         vkd3d_free(s); | 
					
						
							|  |  |  |  |         return false; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     l->sources[l->count++] = s; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void vkd3d_shader_source_list_cleanup(struct vkd3d_shader_source_list *l) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     size_t i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = 0; i < l->count; ++i) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         vkd3d_free((void *)l->sources[i]); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     vkd3d_free(l->sources); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void vkd3d_shader_source_list_init(struct vkd3d_shader_source_list *l) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     memset(l, 0, sizeof(*l)); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-09 01:14:03 -05:00
										 |  |  |  | void vkd3d_string_buffer_init(struct vkd3d_string_buffer *buffer) | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-09-30 21:36:17 +02:00
										 |  |  |  |     buffer->buffer_size = 16; | 
					
						
							|  |  |  |  |     buffer->content_size = 0; | 
					
						
							|  |  |  |  |     buffer->buffer = vkd3d_malloc(buffer->buffer_size); | 
					
						
							| 
									
										
										
										
											2024-08-05 16:50:14 +02:00
										 |  |  |  |     VKD3D_ASSERT(buffer->buffer); | 
					
						
							| 
									
										
										
										
											2021-09-30 21:36:17 +02:00
										 |  |  |  |     memset(buffer->buffer, 0, buffer->buffer_size); | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void vkd3d_string_buffer_cleanup(struct vkd3d_string_buffer *buffer) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     vkd3d_free(buffer->buffer); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-30 15:07:50 +01:00
										 |  |  |  | void vkd3d_string_buffer_clear(struct vkd3d_string_buffer *buffer) | 
					
						
							| 
									
										
										
										
											2021-02-27 18:03:09 -06:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-02-06 15:08:49 +01:00
										 |  |  |  |     vkd3d_string_buffer_truncate(buffer, 0); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void vkd3d_string_buffer_truncate(struct vkd3d_string_buffer *buffer, size_t size) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (size < buffer->content_size) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         buffer->buffer[size] = '\0'; | 
					
						
							|  |  |  |  |         buffer->content_size = size; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-02-27 18:03:09 -06:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  | static bool vkd3d_string_buffer_resize(struct vkd3d_string_buffer *buffer, int rc) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-07-22 10:11:21 +02:00
										 |  |  |  |     size_t new_buffer_size = rc >= 0 ? buffer->content_size + rc + 1 : buffer->buffer_size * 2; | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-30 21:36:16 +02:00
										 |  |  |  |     if (!vkd3d_array_reserve((void **)&buffer->buffer, &buffer->buffer_size, new_buffer_size, 1)) | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  |     { | 
					
						
							|  |  |  |  |         ERR("Failed to grow buffer.\n"); | 
					
						
							|  |  |  |  |         buffer->buffer[buffer->content_size] = '\0'; | 
					
						
							|  |  |  |  |         return false; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | int vkd3d_string_buffer_vprintf(struct vkd3d_string_buffer *buffer, const char *format, va_list args) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-07-22 10:11:21 +02:00
										 |  |  |  |     size_t rem; | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  |     va_list a; | 
					
						
							|  |  |  |  |     int rc; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (;;) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         rem = buffer->buffer_size - buffer->content_size; | 
					
						
							|  |  |  |  |         va_copy(a, args); | 
					
						
							|  |  |  |  |         rc = vsnprintf(&buffer->buffer[buffer->content_size], rem, format, a); | 
					
						
							|  |  |  |  |         va_end(a); | 
					
						
							|  |  |  |  |         if (rc >= 0 && (unsigned int)rc < rem) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             buffer->content_size += rc; | 
					
						
							|  |  |  |  |             return 0; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (!vkd3d_string_buffer_resize(buffer, rc)) | 
					
						
							|  |  |  |  |             return -1; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  |  | int vkd3d_string_buffer_printf(struct vkd3d_string_buffer *buffer, const char *format, ...) | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  | { | 
					
						
							|  |  |  |  |     va_list args; | 
					
						
							|  |  |  |  |     int ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     va_start(args, format); | 
					
						
							|  |  |  |  |     ret = vkd3d_string_buffer_vprintf(buffer, format, args); | 
					
						
							|  |  |  |  |     va_end(args); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return ret; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-01 13:21:33 +01:00
										 |  |  |  | int vkd3d_string_buffer_print_f32(struct vkd3d_string_buffer *buffer, float f) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-07-22 10:11:21 +02:00
										 |  |  |  |     size_t idx = buffer->content_size + 1; | 
					
						
							| 
									
										
										
										
											2022-03-01 13:21:33 +01:00
										 |  |  |  |     int ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!(ret = vkd3d_string_buffer_printf(buffer, "%.8e", f)) && isfinite(f)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         if (signbit(f)) | 
					
						
							|  |  |  |  |             ++idx; | 
					
						
							|  |  |  |  |         buffer->buffer[idx] = '.'; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return ret; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-01 13:21:34 +01:00
										 |  |  |  | int vkd3d_string_buffer_print_f64(struct vkd3d_string_buffer *buffer, double d) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-07-22 10:11:21 +02:00
										 |  |  |  |     size_t idx = buffer->content_size + 1; | 
					
						
							| 
									
										
										
										
											2022-03-01 13:21:34 +01:00
										 |  |  |  |     int ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!(ret = vkd3d_string_buffer_printf(buffer, "%.16e", d)) && isfinite(d)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         if (signbit(d)) | 
					
						
							|  |  |  |  |             ++idx; | 
					
						
							|  |  |  |  |         buffer->buffer[idx] = '.'; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return ret; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-14 00:11:55 +01:00
										 |  |  |  | static char get_escape_char(char c) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     switch (c) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         case '"': | 
					
						
							|  |  |  |  |         case '\\': | 
					
						
							|  |  |  |  |             return c; | 
					
						
							|  |  |  |  |         case '\t': | 
					
						
							|  |  |  |  |             return 't'; | 
					
						
							|  |  |  |  |         case '\n': | 
					
						
							|  |  |  |  |             return 'n'; | 
					
						
							|  |  |  |  |         case '\v': | 
					
						
							|  |  |  |  |             return 'v'; | 
					
						
							|  |  |  |  |         case '\f': | 
					
						
							|  |  |  |  |             return 'f'; | 
					
						
							|  |  |  |  |         case '\r': | 
					
						
							|  |  |  |  |             return 'r'; | 
					
						
							|  |  |  |  |         default: | 
					
						
							|  |  |  |  |             return 0; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | int vkd3d_string_buffer_print_string_escaped(struct vkd3d_string_buffer *buffer, const char *s, size_t len) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     size_t content_size, start, i; | 
					
						
							|  |  |  |  |     int ret; | 
					
						
							|  |  |  |  |     char c; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     content_size = buffer->content_size; | 
					
						
							|  |  |  |  |     for (i = 0, start = 0; i < len; ++i) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         if ((c = get_escape_char(s[i]))) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             if ((ret = vkd3d_string_buffer_printf(buffer, "%.*s\\%c", (int)(i - start), &s[start], c)) < 0) | 
					
						
							|  |  |  |  |                 goto fail; | 
					
						
							|  |  |  |  |             start = i + 1; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         else if (!isprint(s[i])) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             if ((ret = vkd3d_string_buffer_printf(buffer, "%.*s\\%03o", | 
					
						
							|  |  |  |  |                     (int)(i - start), &s[start], (uint8_t)s[i])) < 0) | 
					
						
							|  |  |  |  |                 goto fail; | 
					
						
							|  |  |  |  |             start = i + 1; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     if ((ret = vkd3d_string_buffer_printf(buffer, "%.*s", (int)(len - start), &s[start])) < 0) | 
					
						
							|  |  |  |  |         goto fail; | 
					
						
							|  |  |  |  |     return ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | fail: | 
					
						
							|  |  |  |  |     buffer->content_size = content_size; | 
					
						
							|  |  |  |  |     return ret; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  |  | void vkd3d_string_buffer_trace_(const struct vkd3d_string_buffer *buffer, const char *function) | 
					
						
							| 
									
										
										
										
											2020-07-28 14:22:34 +04:30
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-09-16 22:36:35 -05:00
										 |  |  |  |     vkd3d_shader_trace_text_(buffer->buffer, buffer->content_size, function); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void vkd3d_shader_trace_text_(const char *text, size_t size, const char *function) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     const char *p, *q, *end = text + size; | 
					
						
							| 
									
										
										
										
											2020-07-28 14:22:34 +04:30
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!TRACE_ON()) | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-16 22:36:35 -05:00
										 |  |  |  |     for (p = text; p < end; p = q) | 
					
						
							| 
									
										
										
										
											2020-07-28 14:22:34 +04:30
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-10-15 10:44:45 -05:00
										 |  |  |  |         if (!(q = memchr(p, '\n', end - p))) | 
					
						
							|  |  |  |  |             q = end; | 
					
						
							| 
									
										
										
										
											2020-07-28 14:22:34 +04:30
										 |  |  |  |         else | 
					
						
							|  |  |  |  |             ++q; | 
					
						
							|  |  |  |  |         vkd3d_dbg_printf(VKD3D_DBG_LEVEL_TRACE, function, "%.*s", (int)(q - p), p); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-27 18:03:09 -06:00
										 |  |  |  | void vkd3d_string_buffer_cache_init(struct vkd3d_string_buffer_cache *cache) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     memset(cache, 0, sizeof(*cache)); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void vkd3d_string_buffer_cache_cleanup(struct vkd3d_string_buffer_cache *cache) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     unsigned int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = 0; i < cache->count; ++i) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         vkd3d_string_buffer_cleanup(cache->buffers[i]); | 
					
						
							|  |  |  |  |         vkd3d_free(cache->buffers[i]); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     vkd3d_free(cache->buffers); | 
					
						
							|  |  |  |  |     vkd3d_string_buffer_cache_init(cache); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | struct vkd3d_string_buffer *vkd3d_string_buffer_get(struct vkd3d_string_buffer_cache *cache) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct vkd3d_string_buffer *buffer; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!cache->count) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         if (!vkd3d_array_reserve((void **)&cache->buffers, &cache->capacity, | 
					
						
							|  |  |  |  |                 cache->max_count + 1, sizeof(*cache->buffers))) | 
					
						
							|  |  |  |  |             return NULL; | 
					
						
							|  |  |  |  |         ++cache->max_count; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (!(buffer = vkd3d_malloc(sizeof(*buffer)))) | 
					
						
							|  |  |  |  |             return NULL; | 
					
						
							|  |  |  |  |         vkd3d_string_buffer_init(buffer); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     else | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         buffer = cache->buffers[--cache->count]; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     vkd3d_string_buffer_clear(buffer); | 
					
						
							|  |  |  |  |     return buffer; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void vkd3d_string_buffer_release(struct vkd3d_string_buffer_cache *cache, struct vkd3d_string_buffer *buffer) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (!buffer) | 
					
						
							|  |  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2024-08-05 16:50:14 +02:00
										 |  |  |  |     VKD3D_ASSERT(cache->count + 1 <= cache->max_count); | 
					
						
							| 
									
										
										
										
											2021-02-27 18:03:09 -06:00
										 |  |  |  |     cache->buffers[cache->count++] = buffer; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-10 16:16:23 +01:00
										 |  |  |  | void vkd3d_shader_code_from_string_buffer(struct vkd3d_shader_code *code, struct vkd3d_string_buffer *buffer) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     code->code = buffer->buffer; | 
					
						
							|  |  |  |  |     code->size = buffer->content_size; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     buffer->buffer = NULL; | 
					
						
							|  |  |  |  |     buffer->buffer_size = 0; | 
					
						
							|  |  |  |  |     buffer->content_size = 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-09 01:14:03 -05:00
										 |  |  |  | void vkd3d_shader_message_context_init(struct vkd3d_shader_message_context *context, | 
					
						
							| 
									
										
										
										
											2020-12-15 11:01:32 -06:00
										 |  |  |  |         enum vkd3d_shader_log_level log_level) | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  | { | 
					
						
							|  |  |  |  |     context->log_level = log_level; | 
					
						
							| 
									
										
										
										
											2020-10-09 01:14:03 -05:00
										 |  |  |  |     vkd3d_string_buffer_init(&context->messages); | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void vkd3d_shader_message_context_cleanup(struct vkd3d_shader_message_context *context) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     vkd3d_string_buffer_cleanup(&context->messages); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:57 +04:30
										 |  |  |  | void vkd3d_shader_message_context_trace_messages_(const struct vkd3d_shader_message_context *context, | 
					
						
							| 
									
										
										
										
											2020-07-28 14:22:34 +04:30
										 |  |  |  |         const char *function) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     vkd3d_string_buffer_trace_(&context->messages, function); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-09 01:14:01 -05:00
										 |  |  |  | bool vkd3d_shader_message_context_copy_messages(struct vkd3d_shader_message_context *context, char **out) | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  | { | 
					
						
							|  |  |  |  |     char *messages; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-09 01:14:01 -05:00
										 |  |  |  |     if (!out) | 
					
						
							|  |  |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-09 01:14:01 -05:00
										 |  |  |  |     *out = NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-09 01:14:02 -05:00
										 |  |  |  |     if (!context->messages.content_size) | 
					
						
							|  |  |  |  |         return true; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-09 01:14:01 -05:00
										 |  |  |  |     if (!(messages = vkd3d_malloc(context->messages.content_size + 1))) | 
					
						
							|  |  |  |  |         return false; | 
					
						
							|  |  |  |  |     memcpy(messages, context->messages.buffer, context->messages.content_size + 1); | 
					
						
							|  |  |  |  |     *out = messages; | 
					
						
							|  |  |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-21 22:04:38 -06:00
										 |  |  |  | void vkd3d_shader_vnote(struct vkd3d_shader_message_context *context, const struct vkd3d_shader_location *location, | 
					
						
							|  |  |  |  |         enum vkd3d_shader_log_level level, const char *format, va_list args) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-05-12 21:35:20 +02:00
										 |  |  |  |     struct vkd3d_string_buffer *messages = &context->messages; | 
					
						
							|  |  |  |  |     size_t pos = messages->content_size; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!WARN_ON() && context->log_level < level) | 
					
						
							| 
									
										
										
										
											2021-02-21 22:04:38 -06:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (location) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         const char *source_name = location->source_name ? location->source_name : "<anonymous>"; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (location->line) | 
					
						
							| 
									
										
										
										
											2025-05-12 21:35:20 +02:00
										 |  |  |  |             vkd3d_string_buffer_printf(messages, "%s:%u:%u: ", | 
					
						
							| 
									
										
										
										
											2021-02-21 22:04:38 -06:00
										 |  |  |  |                     source_name, location->line, location->column); | 
					
						
							|  |  |  |  |         else | 
					
						
							| 
									
										
										
										
											2025-05-12 21:35:20 +02:00
										 |  |  |  |             vkd3d_string_buffer_printf(messages, "%s: ", source_name); | 
					
						
							| 
									
										
										
										
											2021-02-21 22:04:38 -06:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-05-12 21:35:20 +02:00
										 |  |  |  |     vkd3d_string_buffer_vprintf(messages, format, args); | 
					
						
							|  |  |  |  |     vkd3d_string_buffer_printf(messages, "\n"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     WARN("%.*s", (int)(messages->content_size - pos), &messages->buffer[pos]); | 
					
						
							|  |  |  |  |     if (context->log_level < level) | 
					
						
							|  |  |  |  |         messages->content_size = pos; | 
					
						
							| 
									
										
										
										
											2021-02-21 22:04:38 -06:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  |  | void vkd3d_shader_vwarning(struct vkd3d_shader_message_context *context, const struct vkd3d_shader_location *location, | 
					
						
							|  |  |  |  |         enum vkd3d_shader_error error, const char *format, va_list args) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-05-12 21:33:09 +02:00
										 |  |  |  |     struct vkd3d_string_buffer *messages = &context->messages; | 
					
						
							|  |  |  |  |     size_t pos = messages->content_size; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!WARN_ON() && context->log_level < VKD3D_SHADER_LOG_WARNING) | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (location) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         const char *source_name = location->source_name ? location->source_name : "<anonymous>"; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (location->line) | 
					
						
							| 
									
										
										
										
											2025-05-12 21:33:09 +02:00
										 |  |  |  |             vkd3d_string_buffer_printf(messages, "%s:%u:%u: W%04u: ", | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  |  |                     source_name, location->line, location->column, error); | 
					
						
							|  |  |  |  |         else | 
					
						
							| 
									
										
										
										
											2025-05-12 21:33:09 +02:00
										 |  |  |  |             vkd3d_string_buffer_printf(messages, "%s: W%04u: ", source_name, error); | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     else | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-05-12 21:33:09 +02:00
										 |  |  |  |         vkd3d_string_buffer_printf(messages, "W%04u: ", error); | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-05-12 21:33:09 +02:00
										 |  |  |  |     vkd3d_string_buffer_vprintf(messages, format, args); | 
					
						
							|  |  |  |  |     vkd3d_string_buffer_printf(messages, "\n"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     WARN("%.*s", (int)(messages->content_size - pos), &messages->buffer[pos]); | 
					
						
							|  |  |  |  |     if (context->log_level < VKD3D_SHADER_LOG_WARNING) | 
					
						
							|  |  |  |  |         messages->content_size = pos; | 
					
						
							| 
									
										
										
										
											2020-12-21 14:37:06 -06:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-12-15 14:07:21 +01:00
										 |  |  |  | void vkd3d_shader_warning(struct vkd3d_shader_message_context *context, const struct vkd3d_shader_location *location, | 
					
						
							|  |  |  |  |         enum vkd3d_shader_error error, const char *format, ...) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     va_list args; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     va_start(args, format); | 
					
						
							|  |  |  |  |     vkd3d_shader_vwarning(context, location, error, format, args); | 
					
						
							|  |  |  |  |     va_end(args); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-15 11:01:32 -06:00
										 |  |  |  | void vkd3d_shader_verror(struct vkd3d_shader_message_context *context, const struct vkd3d_shader_location *location, | 
					
						
							| 
									
										
										
										
											2020-08-04 14:43:29 +04:30
										 |  |  |  |         enum vkd3d_shader_error error, const char *format, va_list args) | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-05-12 21:30:30 +02:00
										 |  |  |  |     struct vkd3d_string_buffer *messages = &context->messages; | 
					
						
							|  |  |  |  |     size_t pos = messages->content_size; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!WARN_ON() && context->log_level < VKD3D_SHADER_LOG_ERROR) | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  |         return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-15 11:01:32 -06:00
										 |  |  |  |     if (location) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         const char *source_name = location->source_name ? location->source_name : "<anonymous>"; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (location->line) | 
					
						
							| 
									
										
										
										
											2025-05-12 21:30:30 +02:00
										 |  |  |  |             vkd3d_string_buffer_printf(messages, "%s:%u:%u: E%04u: ", | 
					
						
							| 
									
										
										
										
											2020-12-15 11:01:32 -06:00
										 |  |  |  |                     source_name, location->line, location->column, error); | 
					
						
							|  |  |  |  |         else | 
					
						
							| 
									
										
										
										
											2025-05-12 21:30:30 +02:00
										 |  |  |  |             vkd3d_string_buffer_printf(messages, "%s: E%04u: ", source_name, error); | 
					
						
							| 
									
										
										
										
											2020-12-15 11:01:32 -06:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  |     else | 
					
						
							| 
									
										
										
										
											2020-12-15 11:01:32 -06:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-05-12 21:30:30 +02:00
										 |  |  |  |         vkd3d_string_buffer_printf(messages, "E%04u: ", error); | 
					
						
							| 
									
										
										
										
											2020-12-15 11:01:32 -06:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-05-12 21:30:30 +02:00
										 |  |  |  |     vkd3d_string_buffer_vprintf(messages, format, args); | 
					
						
							|  |  |  |  |     vkd3d_string_buffer_printf(messages, "\n"); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     WARN("%.*s", (int)(messages->content_size - pos), &messages->buffer[pos]); | 
					
						
							|  |  |  |  |     if (context->log_level < VKD3D_SHADER_LOG_ERROR) | 
					
						
							|  |  |  |  |         messages->content_size = pos; | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-15 11:01:32 -06:00
										 |  |  |  | void vkd3d_shader_error(struct vkd3d_shader_message_context *context, const struct vkd3d_shader_location *location, | 
					
						
							| 
									
										
										
										
											2020-08-04 14:43:29 +04:30
										 |  |  |  |         enum vkd3d_shader_error error, const char *format, ...) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     va_list args; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     va_start(args, format); | 
					
						
							| 
									
										
										
										
											2020-12-15 11:01:32 -06:00
										 |  |  |  |     vkd3d_shader_verror(context, location, error, format, args); | 
					
						
							| 
									
										
										
										
											2020-08-04 14:43:29 +04:30
										 |  |  |  |     va_end(args); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-05 16:55:50 -05:00
										 |  |  |  | size_t bytecode_align(struct vkd3d_bytecode_buffer *buffer) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     size_t aligned_size = align(buffer->size, 4); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!vkd3d_array_reserve((void **)&buffer->data, &buffer->capacity, aligned_size, 1)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         buffer->status = VKD3D_ERROR_OUT_OF_MEMORY; | 
					
						
							|  |  |  |  |         return aligned_size; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-18 17:02:44 -03:00
										 |  |  |  |     if (aligned_size > buffer->size) | 
					
						
							|  |  |  |  |         memset(&buffer->data[buffer->size], 0xab, aligned_size - buffer->size); | 
					
						
							| 
									
										
										
										
											2023-04-05 16:55:50 -05:00
										 |  |  |  |     buffer->size = aligned_size; | 
					
						
							|  |  |  |  |     return aligned_size; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 14:21:46 +01:00
										 |  |  |  | size_t bytecode_put_bytes_unaligned(struct vkd3d_bytecode_buffer *buffer, const void *bytes, size_t size) | 
					
						
							| 
									
										
										
										
											2021-07-26 15:51:29 -05:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-02-06 14:21:46 +01:00
										 |  |  |  |     size_t offset = buffer->size; | 
					
						
							| 
									
										
										
										
											2021-07-26 15:51:29 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (buffer->status) | 
					
						
							|  |  |  |  |         return offset; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-04 18:27:59 +02:00
										 |  |  |  |     if (!vkd3d_array_reserve((void **)&buffer->data, &buffer->capacity, offset + size, 1)) | 
					
						
							| 
									
										
										
										
											2021-07-26 15:51:29 -05:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |         buffer->status = VKD3D_ERROR_OUT_OF_MEMORY; | 
					
						
							|  |  |  |  |         return offset; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-12-18 16:04:15 -03:00
										 |  |  |  |     if (size) | 
					
						
							|  |  |  |  |         memcpy(&buffer->data[offset], bytes, size); | 
					
						
							| 
									
										
										
										
											2023-04-04 18:27:59 +02:00
										 |  |  |  |     buffer->size = offset + size; | 
					
						
							| 
									
										
										
										
											2021-07-26 15:51:29 -05:00
										 |  |  |  |     return offset; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-06 14:21:46 +01:00
										 |  |  |  | size_t bytecode_put_bytes(struct vkd3d_bytecode_buffer *buffer, const void *bytes, size_t size) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     bytecode_align(buffer); | 
					
						
							|  |  |  |  |     return bytecode_put_bytes_unaligned(buffer, bytes, size); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  |  | size_t bytecode_reserve_bytes(struct vkd3d_bytecode_buffer *buffer, size_t size) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     size_t offset = bytecode_align(buffer); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (buffer->status) | 
					
						
							|  |  |  |  |         return offset; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!vkd3d_array_reserve((void **)&buffer->data, &buffer->capacity, offset + size, 1)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         buffer->status = VKD3D_ERROR_OUT_OF_MEMORY; | 
					
						
							|  |  |  |  |         return offset; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     memset(buffer->data + offset, 0, size); | 
					
						
							|  |  |  |  |     buffer->size = offset + size; | 
					
						
							|  |  |  |  |     return offset; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void bytecode_set_bytes(struct vkd3d_bytecode_buffer *buffer, size_t offset, | 
					
						
							|  |  |  |  |         const void *value, size_t size) | 
					
						
							| 
									
										
										
										
											2021-07-26 15:51:29 -05:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     if (buffer->status) | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-05 16:50:14 +02:00
										 |  |  |  |     VKD3D_ASSERT(vkd3d_bound_range(offset, size, buffer->size)); | 
					
						
							| 
									
										
										
										
											2023-11-05 22:07:01 +01:00
										 |  |  |  |     memcpy(buffer->data + offset, value, size); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void set_u32(struct vkd3d_bytecode_buffer *buffer, size_t offset, uint32_t value) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     bytecode_set_bytes(buffer, offset, &value, sizeof(value)); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void set_string(struct vkd3d_bytecode_buffer *buffer, size_t offset, const char *string, size_t length) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     bytecode_set_bytes(buffer, offset, string, length); | 
					
						
							| 
									
										
										
										
											2021-07-26 15:51:29 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  | struct shader_dump_data | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     uint8_t checksum[16]; | 
					
						
							|  |  |  |  |     const char *path; | 
					
						
							|  |  |  |  |     const char *profile; | 
					
						
							|  |  |  |  |     const char *source_suffix; | 
					
						
							|  |  |  |  |     const char *target_suffix; | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-27 13:27:24 -03:00
										 |  |  |  | enum shader_dump_type | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     SHADER_DUMP_TYPE_SOURCE, | 
					
						
							|  |  |  |  |     SHADER_DUMP_TYPE_PREPROC, | 
					
						
							|  |  |  |  |     SHADER_DUMP_TYPE_TARGET, | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  | static void vkd3d_shader_dump_shader(const struct shader_dump_data *dump_data, | 
					
						
							| 
									
										
										
										
											2025-01-27 13:27:24 -03:00
										 |  |  |  |         const void *data, size_t size, enum shader_dump_type type) | 
					
						
							| 
									
										
										
										
											2019-01-31 11:29:27 +01:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-08-12 15:17:30 +02:00
										 |  |  |  |     static const char hexadecimal_digits[] = "0123456789abcdef"; | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  |     const uint8_t *checksum = dump_data->checksum; | 
					
						
							| 
									
										
										
										
											2024-08-12 15:17:30 +02:00
										 |  |  |  |     char str_checksum[33]; | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  |     unsigned int pos = 0; | 
					
						
							| 
									
										
										
										
											2019-01-31 11:29:27 +01:00
										 |  |  |  |     char filename[1024]; | 
					
						
							| 
									
										
										
										
											2024-08-12 15:17:30 +02:00
										 |  |  |  |     unsigned int i; | 
					
						
							| 
									
										
										
										
											2019-01-31 11:29:27 +01:00
										 |  |  |  |     FILE *f; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  |     if (!dump_data->path) | 
					
						
							|  |  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2024-08-12 15:17:30 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  |     for (i = 0; i < ARRAY_SIZE(dump_data->checksum); ++i) | 
					
						
							| 
									
										
										
										
											2024-08-12 15:17:30 +02:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |         str_checksum[2 * i] = hexadecimal_digits[checksum[i] >> 4]; | 
					
						
							|  |  |  |  |         str_checksum[2 * i + 1] = hexadecimal_digits[checksum[i] & 0xf]; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     str_checksum[32] = '\0'; | 
					
						
							| 
									
										
										
										
											2019-01-31 11:29:27 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  |     pos = snprintf(filename, ARRAY_SIZE(filename), "%s/vkd3d-shader-%s", dump_data->path, str_checksum); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (dump_data->profile) | 
					
						
							|  |  |  |  |         pos += snprintf(filename + pos, ARRAY_SIZE(filename) - pos, "-%s", dump_data->profile); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-27 13:27:24 -03:00
										 |  |  |  |     if (type == SHADER_DUMP_TYPE_SOURCE) | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  |         pos += snprintf(filename + pos, ARRAY_SIZE(filename) - pos, "-source.%s", dump_data->source_suffix); | 
					
						
							| 
									
										
										
										
											2025-01-27 13:27:24 -03:00
										 |  |  |  |     else if (type == SHADER_DUMP_TYPE_PREPROC) | 
					
						
							|  |  |  |  |         pos += snprintf(filename + pos, ARRAY_SIZE(filename) - pos, "-preproc.%s", dump_data->source_suffix); | 
					
						
							| 
									
										
										
										
											2023-10-21 20:44:07 +02:00
										 |  |  |  |     else | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  |         pos += snprintf(filename + pos, ARRAY_SIZE(filename) - pos, "-target.%s", dump_data->target_suffix); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-12 15:17:30 +02:00
										 |  |  |  |     TRACE("Dumping shader to \"%s\".\n", filename); | 
					
						
							| 
									
										
										
										
											2019-01-31 11:29:27 +01:00
										 |  |  |  |     if ((f = fopen(filename, "wb"))) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         if (fwrite(data, 1, size, f) != size) | 
					
						
							| 
									
										
										
										
											2024-10-19 21:00:47 +02:00
										 |  |  |  |             WARN("Failed to write shader to %s.\n", filename); | 
					
						
							| 
									
										
										
										
											2019-01-31 11:29:27 +01:00
										 |  |  |  |         if (fclose(f)) | 
					
						
							| 
									
										
										
										
											2024-10-19 21:00:47 +02:00
										 |  |  |  |             WARN("Failed to close stream %s.\n", filename); | 
					
						
							| 
									
										
										
										
											2019-01-31 11:29:27 +01:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     else | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-10-19 21:00:47 +02:00
										 |  |  |  |         WARN("Failed to open %s for dumping shader.\n", filename); | 
					
						
							| 
									
										
										
										
											2019-01-31 11:29:27 +01:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-01 15:45:05 +02:00
										 |  |  |  | static const char *shader_get_source_type_suffix(enum vkd3d_shader_source_type type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     switch (type) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         case VKD3D_SHADER_SOURCE_DXBC_TPF: | 
					
						
							|  |  |  |  |             return "dxbc"; | 
					
						
							|  |  |  |  |         case VKD3D_SHADER_SOURCE_HLSL: | 
					
						
							|  |  |  |  |             return "hlsl"; | 
					
						
							| 
									
										
										
										
											2021-10-06 17:11:48 +02:00
										 |  |  |  |         case VKD3D_SHADER_SOURCE_D3D_BYTECODE: | 
					
						
							|  |  |  |  |             return "d3dbc"; | 
					
						
							| 
									
										
										
										
											2022-11-15 15:16:41 +10:00
										 |  |  |  |         case VKD3D_SHADER_SOURCE_DXBC_DXIL: | 
					
						
							|  |  |  |  |             return "dxil"; | 
					
						
							| 
									
										
										
										
											2021-09-01 15:45:05 +02:00
										 |  |  |  |         default: | 
					
						
							|  |  |  |  |             FIXME("Unhandled source type %#x.\n", type); | 
					
						
							|  |  |  |  |             return "bin"; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  | static const char *shader_get_target_type_suffix(enum vkd3d_shader_target_type type) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     switch (type) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         case VKD3D_SHADER_TARGET_SPIRV_BINARY: | 
					
						
							|  |  |  |  |             return "spv"; | 
					
						
							|  |  |  |  |         case VKD3D_SHADER_TARGET_SPIRV_TEXT: | 
					
						
							|  |  |  |  |             return "spv.s"; | 
					
						
							|  |  |  |  |         case VKD3D_SHADER_TARGET_D3D_ASM: | 
					
						
							|  |  |  |  |             return "d3d.s"; | 
					
						
							|  |  |  |  |         case VKD3D_SHADER_TARGET_D3D_BYTECODE: | 
					
						
							|  |  |  |  |             return "d3dbc"; | 
					
						
							|  |  |  |  |         case VKD3D_SHADER_TARGET_DXBC_TPF: | 
					
						
							|  |  |  |  |             return "dxbc"; | 
					
						
							|  |  |  |  |         case VKD3D_SHADER_TARGET_GLSL: | 
					
						
							|  |  |  |  |             return "glsl"; | 
					
						
							|  |  |  |  |         case VKD3D_SHADER_TARGET_FX: | 
					
						
							|  |  |  |  |             return "fx"; | 
					
						
							| 
									
										
										
										
											2024-08-27 10:34:57 +08:00
										 |  |  |  |         case VKD3D_SHADER_TARGET_MSL: | 
					
						
							|  |  |  |  |             return "msl"; | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  |         default: | 
					
						
							|  |  |  |  |             FIXME("Unhandled target type %#x.\n", type); | 
					
						
							|  |  |  |  |             return "bin"; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void fill_shader_dump_data(const struct vkd3d_shader_compile_info *compile_info, | 
					
						
							|  |  |  |  |         struct shader_dump_data *data) | 
					
						
							| 
									
										
										
										
											2019-01-31 11:29:27 +01:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     static bool enabled = true; | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     data->path = NULL; | 
					
						
							| 
									
										
										
										
											2019-01-31 11:29:27 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!enabled) | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  |     if (!(data->path = getenv("VKD3D_SHADER_DUMP_PATH"))) | 
					
						
							| 
									
										
										
										
											2019-01-31 11:29:27 +01:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |         enabled = false; | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  |     data->profile = NULL; | 
					
						
							| 
									
										
										
										
											2023-10-21 21:35:08 +02:00
										 |  |  |  |     if (compile_info->source_type == VKD3D_SHADER_SOURCE_HLSL) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  |         const struct vkd3d_shader_hlsl_source_info *hlsl_source_info; | 
					
						
							| 
									
										
										
										
											2023-10-21 21:35:08 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  |         if ((hlsl_source_info = vkd3d_find_struct(compile_info->next, HLSL_SOURCE_INFO))) | 
					
						
							|  |  |  |  |             data->profile = hlsl_source_info->profile; | 
					
						
							| 
									
										
										
										
											2023-10-21 21:35:08 +02:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  |     vkd3d_compute_md5(compile_info->source.code, compile_info->source.size, | 
					
						
							|  |  |  |  |             (uint32_t *)data->checksum, VKD3D_MD5_STANDARD); | 
					
						
							|  |  |  |  |     data->source_suffix = shader_get_source_type_suffix(compile_info->source_type); | 
					
						
							|  |  |  |  |     data->target_suffix = shader_get_target_type_suffix(compile_info->target_type); | 
					
						
							| 
									
										
										
										
											2019-01-31 11:29:27 +01:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 16:21:18 -06:00
										 |  |  |  | static void init_scan_signature_info(const struct vkd3d_shader_compile_info *info) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct vkd3d_shader_scan_signature_info *signature_info; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if ((signature_info = vkd3d_find_struct(info->next, SCAN_SIGNATURE_INFO))) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         memset(&signature_info->input, 0, sizeof(signature_info->input)); | 
					
						
							|  |  |  |  |         memset(&signature_info->output, 0, sizeof(signature_info->output)); | 
					
						
							|  |  |  |  |         memset(&signature_info->patch_constant, 0, sizeof(signature_info->patch_constant)); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-29 21:20:35 +02:00
										 |  |  |  | static const struct vkd3d_debug_option vkd3d_shader_config_options[] = | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     {"force_validation", VKD3D_SHADER_CONFIG_FLAG_FORCE_VALIDATION}, /* force validation of internal shader representations */ | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-16 11:42:16 +02:00
										 |  |  |  | uint64_t vkd3d_shader_init_config_flags(void) | 
					
						
							| 
									
										
										
										
											2023-08-29 21:20:35 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     uint64_t config_flags; | 
					
						
							|  |  |  |  |     const char *config; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     config = getenv("VKD3D_SHADER_CONFIG"); | 
					
						
							|  |  |  |  |     config_flags = vkd3d_parse_debug_options(config, vkd3d_shader_config_options, ARRAY_SIZE(vkd3d_shader_config_options)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (config_flags) | 
					
						
							|  |  |  |  |         TRACE("VKD3D_SHADER_CONFIG='%s'.\n", config); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return config_flags; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-20 15:24:37 +02:00
										 |  |  |  | void vkd3d_shader_parser_init(struct vkd3d_shader_parser *parser, | 
					
						
							| 
									
										
										
										
											2024-05-16 11:42:29 +02:00
										 |  |  |  |         struct vkd3d_shader_message_context *message_context, const char *source_name) | 
					
						
							| 
									
										
										
										
											2021-10-01 17:51:14 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     parser->message_context = message_context; | 
					
						
							| 
									
										
										
										
											2021-10-01 17:51:15 +02:00
										 |  |  |  |     parser->location.source_name = source_name; | 
					
						
							|  |  |  |  |     parser->location.line = 1; | 
					
						
							|  |  |  |  |     parser->location.column = 0; | 
					
						
							| 
									
										
										
										
											2017-08-21 12:41:07 +02:00
										 |  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-06-19 18:05:53 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-06 17:11:46 +02:00
										 |  |  |  | void VKD3D_PRINTF_FUNC(3, 4) vkd3d_shader_parser_error(struct vkd3d_shader_parser *parser, | 
					
						
							|  |  |  |  |         enum vkd3d_shader_error error, const char *format, ...) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     va_list args; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     va_start(args, format); | 
					
						
							|  |  |  |  |     vkd3d_shader_verror(parser->message_context, &parser->location, error, format, args); | 
					
						
							|  |  |  |  |     va_end(args); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     parser->failed = true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-06 17:11:48 +02:00
										 |  |  |  | void VKD3D_PRINTF_FUNC(3, 4) vkd3d_shader_parser_warning(struct vkd3d_shader_parser *parser, | 
					
						
							|  |  |  |  |         enum vkd3d_shader_error error, const char *format, ...) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     va_list args; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     va_start(args, format); | 
					
						
							|  |  |  |  |     vkd3d_shader_vwarning(parser->message_context, &parser->location, error, format, args); | 
					
						
							|  |  |  |  |     va_end(args); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 09:29:11 -06:00
										 |  |  |  | static int vkd3d_shader_validate_compile_info(const struct vkd3d_shader_compile_info *compile_info, | 
					
						
							|  |  |  |  |         bool validate_target_type) | 
					
						
							| 
									
										
										
										
											2020-06-19 16:13:36 +04:30
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-08-06 13:23:48 +04:30
										 |  |  |  |     const enum vkd3d_shader_source_type *source_types; | 
					
						
							| 
									
										
										
										
											2020-08-06 13:23:49 +04:30
										 |  |  |  |     const enum vkd3d_shader_target_type *target_types; | 
					
						
							| 
									
										
										
										
											2020-08-06 13:23:48 +04:30
										 |  |  |  |     unsigned int count, i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-19 16:13:36 +04:30
										 |  |  |  |     if (compile_info->type != VKD3D_SHADER_STRUCTURE_TYPE_COMPILE_INFO) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         WARN("Invalid structure type %#x.\n", compile_info->type); | 
					
						
							|  |  |  |  |         return VKD3D_ERROR_INVALID_ARGUMENT; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-06 13:23:48 +04:30
										 |  |  |  |     source_types = vkd3d_shader_get_supported_source_types(&count); | 
					
						
							|  |  |  |  |     for (i = 0; i < count; ++i) | 
					
						
							| 
									
										
										
										
											2020-06-19 16:13:36 +04:30
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-08-06 13:23:48 +04:30
										 |  |  |  |         if (source_types[i] == compile_info->source_type) | 
					
						
							| 
									
										
										
										
											2020-06-19 16:13:36 +04:30
										 |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2020-08-06 13:23:48 +04:30
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     if (i == count) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         WARN("Invalid shader source type %#x.\n", compile_info->source_type); | 
					
						
							|  |  |  |  |         return VKD3D_ERROR_INVALID_ARGUMENT; | 
					
						
							| 
									
										
										
										
											2020-06-19 16:13:36 +04:30
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 09:29:11 -06:00
										 |  |  |  |     if (validate_target_type) | 
					
						
							| 
									
										
										
										
											2020-06-23 13:50:12 +04:30
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-12-02 09:29:11 -06:00
										 |  |  |  |         target_types = vkd3d_shader_get_supported_target_types(compile_info->source_type, &count); | 
					
						
							|  |  |  |  |         for (i = 0; i < count; ++i) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             if (target_types[i] == compile_info->target_type) | 
					
						
							|  |  |  |  |                 break; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         if (i == count) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             WARN("Invalid shader target type %#x.\n", compile_info->target_type); | 
					
						
							|  |  |  |  |             return VKD3D_ERROR_INVALID_ARGUMENT; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-23 13:50:12 +04:30
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-19 16:13:36 +04:30
										 |  |  |  |     return VKD3D_OK; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-04 18:05:23 -05:00
										 |  |  |  | static enum vkd3d_result vsir_parse(const struct vkd3d_shader_compile_info *compile_info, uint64_t config_flags, | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:49 +02:00
										 |  |  |  |         const struct shader_dump_data *dump_data, struct vkd3d_shader_message_context *message_context, | 
					
						
							|  |  |  |  |         struct vsir_program *program, struct vkd3d_shader_code *reflection_data) | 
					
						
							| 
									
										
										
										
											2024-10-04 18:05:23 -05:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:49 +02:00
										 |  |  |  |     struct vkd3d_shader_compile_info preprocessed_info; | 
					
						
							|  |  |  |  |     struct vkd3d_shader_code preprocessed; | 
					
						
							| 
									
										
										
										
											2024-10-04 18:05:23 -05:00
										 |  |  |  |     enum vkd3d_result ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     switch (compile_info->source_type) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:49 +02:00
										 |  |  |  |         case VKD3D_SHADER_SOURCE_HLSL: | 
					
						
							|  |  |  |  |             if ((ret = preproc_lexer_parse(compile_info, &preprocessed, message_context)) >= 0) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 vkd3d_shader_dump_shader(dump_data, preprocessed.code, preprocessed.size, SHADER_DUMP_TYPE_PREPROC); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 preprocessed_info = *compile_info; | 
					
						
							|  |  |  |  |                 preprocessed_info.source = preprocessed; | 
					
						
							|  |  |  |  |                 ret = hlsl_parse(&preprocessed_info, message_context, program, reflection_data); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |                 vkd3d_shader_free_shader_code(&preprocessed); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-04 18:05:23 -05:00
										 |  |  |  |         case VKD3D_SHADER_SOURCE_D3D_BYTECODE: | 
					
						
							|  |  |  |  |             ret = d3dbc_parse(compile_info, config_flags, message_context, program); | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         case VKD3D_SHADER_SOURCE_DXBC_TPF: | 
					
						
							|  |  |  |  |             ret = tpf_parse(compile_info, config_flags, message_context, program); | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         case VKD3D_SHADER_SOURCE_DXBC_DXIL: | 
					
						
							|  |  |  |  |             ret = dxil_parse(compile_info, config_flags, message_context, program); | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         default: | 
					
						
							|  |  |  |  |             ERR("Unsupported source type %#x.\n", compile_info->source_type); | 
					
						
							|  |  |  |  |             ret = VKD3D_ERROR_INVALID_ARGUMENT; | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (ret < 0) | 
					
						
							| 
									
										
										
										
											2024-10-04 18:09:05 -05:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-10-04 18:05:23 -05:00
										 |  |  |  |         WARN("Failed to parse shader.\n"); | 
					
						
							| 
									
										
										
										
											2024-10-04 18:09:05 -05:00
										 |  |  |  |         return ret; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if ((ret = vsir_program_validate(program, config_flags, compile_info->source_name, message_context)) < 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         WARN("Failed to validate shader after parsing, ret %d.\n", ret); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (TRACE_ON()) | 
					
						
							|  |  |  |  |             vsir_program_trace(program); | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:49 +02:00
										 |  |  |  |         goto fail; | 
					
						
							| 
									
										
										
										
											2024-10-04 18:09:05 -05:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-10-04 18:05:23 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:18 +02:00
										 |  |  |  |     if (compile_info->target_type != VKD3D_SHADER_TARGET_NONE | 
					
						
							|  |  |  |  |             && (ret = vsir_program_transform_early(program, config_flags, compile_info, message_context)) < 0) | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:49 +02:00
										 |  |  |  |         goto fail; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | fail: | 
					
						
							|  |  |  |  |     vkd3d_shader_free_shader_code(reflection_data); | 
					
						
							|  |  |  |  |     vsir_program_cleanup(program); | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:18 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-04 18:05:23 -05:00
										 |  |  |  |     return ret; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  | void vkd3d_shader_free_messages(char *messages) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-04 20:31:14 +04:30
										 |  |  |  |     TRACE("messages %p.\n", messages); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-23 13:41:25 +04:30
										 |  |  |  |     vkd3d_free(messages); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 16:21:18 -06:00
										 |  |  |  | static bool vkd3d_shader_signature_from_shader_signature(struct vkd3d_shader_signature *signature, | 
					
						
							|  |  |  |  |         const struct shader_signature *src) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-02-19 14:43:31 +01:00
										 |  |  |  |     struct vkd3d_shader_signature_element *d; | 
					
						
							|  |  |  |  |     const struct signature_element *e; | 
					
						
							|  |  |  |  |     size_t count, i, j; | 
					
						
							| 
									
										
										
										
											2023-02-23 16:21:18 -06:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-19 14:43:31 +01:00
										 |  |  |  |     for (i = 0, count = 0; i < src->element_count; ++i) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         e = &src->elements[i]; | 
					
						
							|  |  |  |  |         count += e->register_count; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     signature->element_count = count; | 
					
						
							| 
									
										
										
										
											2023-02-23 16:21:18 -06:00
										 |  |  |  |     if (!src->elements) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-02-19 14:43:31 +01:00
										 |  |  |  |         VKD3D_ASSERT(!count); | 
					
						
							| 
									
										
										
										
											2023-02-23 16:21:18 -06:00
										 |  |  |  |         signature->elements = NULL; | 
					
						
							|  |  |  |  |         return true; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!(signature->elements = vkd3d_calloc(signature->element_count, sizeof(*signature->elements)))) | 
					
						
							|  |  |  |  |         return false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-19 14:43:31 +01:00
										 |  |  |  |     for (i = 0, d = signature->elements; i < src->element_count; ++i) | 
					
						
							| 
									
										
										
										
											2023-02-23 16:21:18 -06:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-02-19 14:43:31 +01:00
										 |  |  |  |         for (j = 0, e = &src->elements[i]; j < e->register_count; ++j) | 
					
						
							| 
									
										
										
										
											2024-05-21 20:11:26 +02:00
										 |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2025-02-19 14:43:31 +01:00
										 |  |  |  |             if (!(d->semantic_name = vkd3d_strdup(e->semantic_name))) | 
					
						
							| 
									
										
										
										
											2024-05-21 20:11:26 +02:00
										 |  |  |  |             { | 
					
						
							| 
									
										
										
										
											2025-02-19 14:43:31 +01:00
										 |  |  |  |                 vkd3d_shader_free_shader_signature(signature); | 
					
						
							|  |  |  |  |                 return false; | 
					
						
							| 
									
										
										
										
											2024-05-21 20:11:26 +02:00
										 |  |  |  |             } | 
					
						
							| 
									
										
										
										
											2025-02-19 14:43:31 +01:00
										 |  |  |  |             d->semantic_index = e->semantic_index + j; | 
					
						
							|  |  |  |  |             d->stream_index = e->stream_index; | 
					
						
							|  |  |  |  |             d->sysval_semantic = e->sysval_semantic; | 
					
						
							|  |  |  |  |             d->component_type = e->component_type; | 
					
						
							|  |  |  |  |             d->register_index = e->register_index + j; | 
					
						
							|  |  |  |  |             d->mask = e->mask; | 
					
						
							|  |  |  |  |             d->used_mask = e->used_mask; | 
					
						
							|  |  |  |  |             d->min_precision = e->min_precision; | 
					
						
							|  |  |  |  |             ++d; | 
					
						
							| 
									
										
										
										
											2024-05-21 20:11:26 +02:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-02-23 16:21:18 -06:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  | struct vkd3d_shader_scan_context | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |     const struct vkd3d_shader_version *version; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-31 18:52:21 -05:00
										 |  |  |  |     struct vkd3d_shader_scan_descriptor_info1 *scan_descriptor_info; | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  |     size_t descriptors_size; | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:26 +04:30
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:58 -05:00
										 |  |  |  |     struct vkd3d_shader_message_context *message_context; | 
					
						
							| 
									
										
										
										
											2020-12-15 11:01:32 -06:00
										 |  |  |  |     struct vkd3d_shader_location location; | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     struct vkd3d_shader_cf_info | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         enum | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             VKD3D_SHADER_BLOCK_IF, | 
					
						
							|  |  |  |  |             VKD3D_SHADER_BLOCK_LOOP, | 
					
						
							|  |  |  |  |             VKD3D_SHADER_BLOCK_SWITCH, | 
					
						
							|  |  |  |  |         } type; | 
					
						
							|  |  |  |  |         bool inside_block; | 
					
						
							|  |  |  |  |         bool has_default; | 
					
						
							|  |  |  |  |     } *cf_info; | 
					
						
							|  |  |  |  |     size_t cf_info_size; | 
					
						
							|  |  |  |  |     size_t cf_info_count; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-28 12:23:46 +01:00
										 |  |  |  |     enum vkd3d_shader_api_version api_version; | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     struct vkd3d_shader_scan_combined_resource_sampler_info *combined_sampler_info; | 
					
						
							|  |  |  |  |     size_t combined_samplers_size; | 
					
						
							| 
									
										
										
										
											2024-10-22 22:30:13 -04:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     enum vkd3d_shader_tessellator_output_primitive output_primitive; | 
					
						
							|  |  |  |  |     enum vkd3d_shader_tessellator_partitioning partitioning; | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  | }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-20 14:51:05 +01:00
										 |  |  |  | static VKD3D_PRINTF_FUNC(3, 4) void vkd3d_shader_scan_error(struct vkd3d_shader_scan_context *context, | 
					
						
							|  |  |  |  |         enum vkd3d_shader_error error, const char *format, ...) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     va_list args; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     va_start(args, format); | 
					
						
							|  |  |  |  |     vkd3d_shader_verror(context->message_context, &context->location, error, format, args); | 
					
						
							|  |  |  |  |     va_end(args); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-20 14:58:40 +01:00
										 |  |  |  | static void VKD3D_PRINTF_FUNC(3, 4) vkd3d_shader_scan_warning(struct vkd3d_shader_scan_context *context, | 
					
						
							|  |  |  |  |         enum vkd3d_shader_error error, const char *format, ...) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     va_list args; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     va_start(args, format); | 
					
						
							|  |  |  |  |     vkd3d_shader_vwarning(context->message_context, &context->location, error, format, args); | 
					
						
							|  |  |  |  |     va_end(args); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:58 -05:00
										 |  |  |  | static void vkd3d_shader_scan_context_init(struct vkd3d_shader_scan_context *context, | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |         const struct vkd3d_shader_version *version, | 
					
						
							| 
									
										
										
										
											2020-12-15 11:01:32 -06:00
										 |  |  |  |         const struct vkd3d_shader_compile_info *compile_info, | 
					
						
							| 
									
										
										
										
											2023-07-31 18:52:21 -05:00
										 |  |  |  |         struct vkd3d_shader_scan_descriptor_info1 *scan_descriptor_info, | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |         struct vkd3d_shader_scan_combined_resource_sampler_info *combined_sampler_info, | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:58 -05:00
										 |  |  |  |         struct vkd3d_shader_message_context *message_context) | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2022-02-28 12:23:46 +01:00
										 |  |  |  |     unsigned int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |     memset(context, 0, sizeof(*context)); | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |     context->version = version; | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:55 +04:30
										 |  |  |  |     context->scan_descriptor_info = scan_descriptor_info; | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:58 -05:00
										 |  |  |  |     context->message_context = message_context; | 
					
						
							| 
									
										
										
										
											2020-12-15 11:01:32 -06:00
										 |  |  |  |     context->location.source_name = compile_info->source_name; | 
					
						
							|  |  |  |  |     context->location.line = 2; /* Line 1 is the version token. */ | 
					
						
							| 
									
										
										
										
											2022-02-28 12:23:46 +01:00
										 |  |  |  |     context->api_version = VKD3D_SHADER_API_VERSION_1_2; | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |     context->combined_sampler_info = combined_sampler_info; | 
					
						
							| 
									
										
										
										
											2022-02-28 12:23:46 +01:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = 0; i < compile_info->option_count; ++i) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         const struct vkd3d_shader_compile_option *option = &compile_info->options[i]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (option->name == VKD3D_SHADER_COMPILE_OPTION_API_VERSION) | 
					
						
							|  |  |  |  |             context->api_version = option->value; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void vkd3d_shader_scan_context_cleanup(struct vkd3d_shader_scan_context *context) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     vkd3d_free(context->cf_info); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct vkd3d_shader_cf_info *vkd3d_shader_scan_get_current_cf_info(struct vkd3d_shader_scan_context *context) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (!context->cf_info_count) | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  |     return &context->cf_info[context->cf_info_count - 1]; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct vkd3d_shader_cf_info *vkd3d_shader_scan_push_cf_info(struct vkd3d_shader_scan_context *context) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct vkd3d_shader_cf_info *cf_info; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!vkd3d_array_reserve((void **)&context->cf_info, &context->cf_info_size, | 
					
						
							|  |  |  |  |             context->cf_info_count + 1, sizeof(*context->cf_info))) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         ERR("Failed to allocate UAV range.\n"); | 
					
						
							|  |  |  |  |         return false; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     cf_info = &context->cf_info[context->cf_info_count++]; | 
					
						
							|  |  |  |  |     memset(cf_info, 0, sizeof(*cf_info)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return cf_info; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void vkd3d_shader_scan_pop_cf_info(struct vkd3d_shader_scan_context *context) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-08-05 16:50:14 +02:00
										 |  |  |  |     VKD3D_ASSERT(context->cf_info_count); | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     --context->cf_info_count; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct vkd3d_shader_cf_info *vkd3d_shader_scan_find_innermost_breakable_cf_info( | 
					
						
							|  |  |  |  |         struct vkd3d_shader_scan_context *context) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     size_t count = context->cf_info_count; | 
					
						
							|  |  |  |  |     struct vkd3d_shader_cf_info *cf_info; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     while (count) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         cf_info = &context->cf_info[--count]; | 
					
						
							|  |  |  |  |         if (cf_info->type == VKD3D_SHADER_BLOCK_LOOP | 
					
						
							|  |  |  |  |                 || cf_info->type == VKD3D_SHADER_BLOCK_SWITCH) | 
					
						
							|  |  |  |  |             return cf_info; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct vkd3d_shader_cf_info *vkd3d_shader_scan_find_innermost_loop_cf_info( | 
					
						
							|  |  |  |  |         struct vkd3d_shader_scan_context *context) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     size_t count = context->cf_info_count; | 
					
						
							|  |  |  |  |     struct vkd3d_shader_cf_info *cf_info; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     while (count) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         cf_info = &context->cf_info[--count]; | 
					
						
							|  |  |  |  |         if (cf_info->type == VKD3D_SHADER_BLOCK_LOOP) | 
					
						
							|  |  |  |  |             return cf_info; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-31 17:59:09 -05:00
										 |  |  |  | static void vkd3d_shader_scan_add_uav_flag(const struct vkd3d_shader_scan_context *context, | 
					
						
							|  |  |  |  |         const struct vkd3d_shader_register *reg, uint32_t flag) | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:26 +04:30
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-07-31 17:59:09 -05:00
										 |  |  |  |     unsigned int range_id = reg->idx[0].offset; | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:26 +04:30
										 |  |  |  |     unsigned int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-31 17:59:09 -05:00
										 |  |  |  |     if (!context->scan_descriptor_info) | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-31 19:10:37 -05:00
										 |  |  |  |     for (i = 0; i < context->scan_descriptor_info->descriptor_count; ++i) | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:26 +04:30
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-09-01 12:44:44 +10:00
										 |  |  |  |         if (context->scan_descriptor_info->descriptors[i].type == VKD3D_SHADER_DESCRIPTOR_TYPE_UAV | 
					
						
							|  |  |  |  |                 && context->scan_descriptor_info->descriptors[i].register_id == range_id) | 
					
						
							| 
									
										
										
										
											2023-07-31 17:59:09 -05:00
										 |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2023-07-31 19:10:37 -05:00
										 |  |  |  |             context->scan_descriptor_info->descriptors[i].flags |= flag; | 
					
						
							| 
									
										
										
										
											2023-07-31 17:59:09 -05:00
										 |  |  |  |             break; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:26 +04:30
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 12:41:07 +02:00
										 |  |  |  | static bool vkd3d_shader_instruction_is_uav_read(const struct vkd3d_shader_instruction *instruction) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-04-24 20:34:42 +02:00
										 |  |  |  |     enum vkd3d_shader_opcode opcode = instruction->opcode; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |     return (VSIR_OP_ATOMIC_AND <= opcode && opcode <= VSIR_OP_ATOMIC_XOR) | 
					
						
							|  |  |  |  |             || (VSIR_OP_IMM_ATOMIC_ALLOC <= opcode && opcode <= VSIR_OP_IMM_ATOMIC_XOR) | 
					
						
							|  |  |  |  |             || opcode == VSIR_OP_LD_UAV_TYPED | 
					
						
							|  |  |  |  |             || (opcode == VSIR_OP_LD_RAW && instruction->src[1].reg.type == VKD3DSPR_UAV) | 
					
						
							|  |  |  |  |             || (opcode == VSIR_OP_LD_STRUCTURED && instruction->src[2].reg.type == VKD3DSPR_UAV); | 
					
						
							| 
									
										
										
										
											2017-08-21 12:41:07 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:27 +04:30
										 |  |  |  | static void vkd3d_shader_scan_record_uav_read(struct vkd3d_shader_scan_context *context, | 
					
						
							| 
									
										
										
										
											2017-08-21 12:41:07 +02:00
										 |  |  |  |         const struct vkd3d_shader_register *reg) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-07-31 17:59:09 -05:00
										 |  |  |  |     vkd3d_shader_scan_add_uav_flag(context, reg, VKD3D_SHADER_DESCRIPTOR_INFO_FLAG_UAV_READ); | 
					
						
							| 
									
										
										
										
											2017-08-21 12:41:07 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-01 11:27:58 +02:00
										 |  |  |  | static bool vkd3d_shader_instruction_is_uav_counter(const struct vkd3d_shader_instruction *instruction) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-04-24 20:34:42 +02:00
										 |  |  |  |     enum vkd3d_shader_opcode opcode = instruction->opcode; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |     return opcode == VSIR_OP_IMM_ATOMIC_ALLOC || opcode == VSIR_OP_IMM_ATOMIC_CONSUME; | 
					
						
							| 
									
										
										
										
											2017-09-01 11:27:58 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:26 +04:30
										 |  |  |  | static void vkd3d_shader_scan_record_uav_counter(struct vkd3d_shader_scan_context *context, | 
					
						
							| 
									
										
										
										
											2017-09-01 11:27:58 +02:00
										 |  |  |  |         const struct vkd3d_shader_register *reg) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-07-31 17:59:09 -05:00
										 |  |  |  |     vkd3d_shader_scan_add_uav_flag(context, reg, VKD3D_SHADER_DESCRIPTOR_INFO_FLAG_UAV_COUNTER); | 
					
						
							| 
									
										
										
										
											2017-09-01 11:27:58 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-18 13:44:05 +10:00
										 |  |  |  | static bool vkd3d_shader_instruction_is_uav_atomic_op(const struct vkd3d_shader_instruction *instruction) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-04-24 20:34:42 +02:00
										 |  |  |  |     enum vkd3d_shader_opcode opcode = instruction->opcode; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |     return (VSIR_OP_ATOMIC_AND <= opcode && opcode <= VSIR_OP_ATOMIC_XOR) | 
					
						
							|  |  |  |  |             || (VSIR_OP_IMM_ATOMIC_ALLOC <= opcode && opcode <= VSIR_OP_IMM_ATOMIC_XOR); | 
					
						
							| 
									
										
										
										
											2022-11-18 13:44:05 +10:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void vkd3d_shader_scan_record_uav_atomic_op(struct vkd3d_shader_scan_context *context, | 
					
						
							|  |  |  |  |         const struct vkd3d_shader_register *reg) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-07-31 17:59:09 -05:00
										 |  |  |  |     vkd3d_shader_scan_add_uav_flag(context, reg, VKD3D_SHADER_DESCRIPTOR_INFO_FLAG_UAV_ATOMICS); | 
					
						
							| 
									
										
										
										
											2022-11-18 13:44:05 +10:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-15 17:52:24 -05:00
										 |  |  |  | static struct vkd3d_shader_descriptor_info1 *vkd3d_shader_scan_add_descriptor(struct vkd3d_shader_scan_context *context, | 
					
						
							| 
									
										
										
										
											2023-07-31 18:59:16 -05:00
										 |  |  |  |         enum vkd3d_shader_descriptor_type type, const struct vkd3d_shader_register *reg, | 
					
						
							|  |  |  |  |         const struct vkd3d_shader_register_range *range, enum vkd3d_shader_resource_type resource_type, | 
					
						
							| 
									
										
										
										
											2025-07-17 15:22:54 +02:00
										 |  |  |  |         enum vsir_data_type resource_data_type) | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-07-31 18:52:21 -05:00
										 |  |  |  |     struct vkd3d_shader_scan_descriptor_info1 *info = context->scan_descriptor_info; | 
					
						
							|  |  |  |  |     struct vkd3d_shader_descriptor_info1 *d; | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-27 11:48:24 -05:00
										 |  |  |  |     if (!info) | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:55 +04:30
										 |  |  |  |     if (!vkd3d_array_reserve((void **)&info->descriptors, &context->descriptors_size, | 
					
						
							|  |  |  |  |             info->descriptor_count + 1, sizeof(*info->descriptors))) | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  |     { | 
					
						
							|  |  |  |  |         ERR("Failed to allocate descriptor info.\n"); | 
					
						
							| 
									
										
										
										
											2023-08-15 17:52:24 -05:00
										 |  |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:55 +04:30
										 |  |  |  |     d = &info->descriptors[info->descriptor_count]; | 
					
						
							| 
									
										
										
										
											2023-08-15 17:52:24 -05:00
										 |  |  |  |     memset(d, 0, sizeof(*d)); | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  |     d->type = type; | 
					
						
							| 
									
										
										
										
											2023-07-31 18:59:16 -05:00
										 |  |  |  |     d->register_id = reg->idx[0].offset; | 
					
						
							| 
									
										
										
										
											2021-06-28 13:03:11 +10:00
										 |  |  |  |     d->register_space = range->space; | 
					
						
							|  |  |  |  |     d->register_index = range->first; | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:25 +04:30
										 |  |  |  |     d->resource_type = resource_type; | 
					
						
							|  |  |  |  |     d->resource_data_type = resource_data_type; | 
					
						
							| 
									
										
										
										
											2021-07-23 23:01:38 +10:00
										 |  |  |  |     d->count = (range->last == ~0u) ? ~0u : range->last - range->first + 1; | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:55 +04:30
										 |  |  |  |     ++info->descriptor_count; | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-15 17:52:24 -05:00
										 |  |  |  |     return d; | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void vkd3d_shader_scan_constant_buffer_declaration(struct vkd3d_shader_scan_context *context, | 
					
						
							|  |  |  |  |         const struct vkd3d_shader_instruction *instruction) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     const struct vkd3d_shader_constant_buffer *cb = &instruction->declaration.cb; | 
					
						
							| 
									
										
										
										
											2023-07-31 19:34:23 -05:00
										 |  |  |  |     struct vkd3d_shader_descriptor_info1 *d; | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-31 19:34:23 -05:00
										 |  |  |  |     if (!(d = vkd3d_shader_scan_add_descriptor(context, VKD3D_SHADER_DESCRIPTOR_TYPE_CBV, | 
					
						
							| 
									
										
										
										
											2025-07-17 16:47:43 +02:00
										 |  |  |  |             &cb->src.reg, &cb->range, VKD3D_SHADER_RESOURCE_BUFFER, VSIR_DATA_U32))) | 
					
						
							| 
									
										
										
										
											2023-07-31 19:34:23 -05:00
										 |  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2023-10-19 13:15:46 +10:00
										 |  |  |  |     d->buffer_size = cb->size; | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void vkd3d_shader_scan_sampler_declaration(struct vkd3d_shader_scan_context *context, | 
					
						
							| 
									
										
										
										
											2018-10-22 01:49:15 +02:00
										 |  |  |  |         const struct vkd3d_shader_instruction *instruction) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  |     const struct vkd3d_shader_sampler *sampler = &instruction->declaration.sampler; | 
					
						
							| 
									
										
										
										
											2023-08-15 17:52:24 -05:00
										 |  |  |  |     struct vkd3d_shader_descriptor_info1 *d; | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-15 17:52:24 -05:00
										 |  |  |  |     if (!(d = vkd3d_shader_scan_add_descriptor(context, VKD3D_SHADER_DESCRIPTOR_TYPE_SAMPLER, | 
					
						
							| 
									
										
										
										
											2025-07-21 15:31:59 +02:00
										 |  |  |  |             &sampler->src.reg, &sampler->range, VKD3D_SHADER_RESOURCE_NONE, VSIR_DATA_UNUSED))) | 
					
						
							| 
									
										
										
										
											2023-08-15 17:52:24 -05:00
										 |  |  |  |         return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-22 01:49:15 +02:00
										 |  |  |  |     if (instruction->flags & VKD3DSI_SAMPLER_COMPARISON_MODE) | 
					
						
							| 
									
										
										
										
											2023-08-15 17:52:24 -05:00
										 |  |  |  |         d->flags |= VKD3D_SHADER_DESCRIPTOR_INFO_FLAG_SAMPLER_COMPARISON_MODE; | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-11 15:26:25 -05:00
										 |  |  |  | static void vkd3d_shader_scan_combined_sampler_declaration( | 
					
						
							|  |  |  |  |         struct vkd3d_shader_scan_context *context, const struct vkd3d_shader_semantic *semantic) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     vkd3d_shader_scan_add_descriptor(context, VKD3D_SHADER_DESCRIPTOR_TYPE_SAMPLER, &semantic->resource.reg.reg, | 
					
						
							| 
									
										
										
										
											2025-07-21 15:31:59 +02:00
										 |  |  |  |             &semantic->resource.range, VKD3D_SHADER_RESOURCE_NONE, VSIR_DATA_UNUSED); | 
					
						
							| 
									
										
										
										
											2023-08-11 15:26:25 -05:00
										 |  |  |  |     vkd3d_shader_scan_add_descriptor(context, VKD3D_SHADER_DESCRIPTOR_TYPE_SRV, &semantic->resource.reg.reg, | 
					
						
							| 
									
										
										
										
											2025-07-17 15:55:17 +02:00
										 |  |  |  |             &semantic->resource.range, semantic->resource_type, VSIR_DATA_F32); | 
					
						
							| 
									
										
										
										
											2023-08-11 15:26:25 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-15 16:03:50 +01:00
										 |  |  |  | const struct vkd3d_shader_descriptor_info1 *vkd3d_shader_find_descriptor( | 
					
						
							| 
									
										
										
										
											2023-11-24 17:25:55 -06:00
										 |  |  |  |         const struct vkd3d_shader_scan_descriptor_info1 *info, | 
					
						
							|  |  |  |  |         enum vkd3d_shader_descriptor_type type, unsigned int register_id) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     for (unsigned int i = 0; i < info->descriptor_count; ++i) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         const struct vkd3d_shader_descriptor_info1 *d = &info->descriptors[i]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (d->type == type && d->register_id == register_id) | 
					
						
							|  |  |  |  |             return d; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  | static void vkd3d_shader_scan_combined_sampler_usage(struct vkd3d_shader_scan_context *context, | 
					
						
							|  |  |  |  |         const struct vkd3d_shader_register *resource, const struct vkd3d_shader_register *sampler) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct vkd3d_shader_scan_combined_resource_sampler_info *info; | 
					
						
							|  |  |  |  |     struct vkd3d_shader_combined_resource_sampler_info *s; | 
					
						
							|  |  |  |  |     unsigned resource_space = 0, sampler_space = 0; | 
					
						
							|  |  |  |  |     unsigned int resource_idx, sampler_idx, i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!(info = context->combined_sampler_info)) | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (resource->type == VKD3DSPR_RESOURCE) | 
					
						
							|  |  |  |  |         resource_idx = resource->idx[1].offset; | 
					
						
							|  |  |  |  |     else | 
					
						
							|  |  |  |  |         resource_idx = resource->idx[0].offset; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!sampler) | 
					
						
							|  |  |  |  |         sampler_idx = VKD3D_SHADER_DUMMY_SAMPLER_INDEX; | 
					
						
							|  |  |  |  |     else if (sampler->type == VKD3DSPR_SAMPLER) | 
					
						
							|  |  |  |  |         sampler_idx = sampler->idx[1].offset; | 
					
						
							|  |  |  |  |     else | 
					
						
							|  |  |  |  |         sampler_idx = sampler->idx[0].offset; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (vkd3d_shader_ver_ge(context->version, 5, 1)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         const struct vkd3d_shader_descriptor_info1 *d; | 
					
						
							| 
									
										
										
										
											2023-11-20 14:58:40 +01:00
										 |  |  |  |         bool dynamic_resource, dynamic_sampler; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if ((dynamic_resource = resource->idx[1].rel_addr)) | 
					
						
							|  |  |  |  |             vkd3d_shader_scan_warning(context, VKD3D_SHADER_WARNING_VSIR_DYNAMIC_DESCRIPTOR_ARRAY, | 
					
						
							|  |  |  |  |                     "Resource descriptor array %u is being dynamically indexed, " | 
					
						
							|  |  |  |  |                     "not recording a combined resource-sampler pair.", resource->idx[0].offset); | 
					
						
							|  |  |  |  |         if ((dynamic_sampler = sampler && sampler->idx[1].rel_addr)) | 
					
						
							|  |  |  |  |             vkd3d_shader_scan_warning(context, VKD3D_SHADER_WARNING_VSIR_DYNAMIC_DESCRIPTOR_ARRAY, | 
					
						
							|  |  |  |  |                     "Sampler descriptor array %u is being dynamically indexed, " | 
					
						
							|  |  |  |  |                     "not recording a combined resource-sampler pair.", sampler->idx[0].offset); | 
					
						
							|  |  |  |  |         if (dynamic_resource || dynamic_sampler) | 
					
						
							|  |  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-15 16:03:50 +01:00
										 |  |  |  |         if ((d = vkd3d_shader_find_descriptor(context->scan_descriptor_info, | 
					
						
							| 
									
										
										
										
											2023-11-24 17:25:55 -06:00
										 |  |  |  |                 VKD3D_SHADER_DESCRIPTOR_TYPE_SRV, resource->idx[0].offset))) | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |             resource_space = d->register_space; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-15 16:03:50 +01:00
										 |  |  |  |         if (sampler && (d = vkd3d_shader_find_descriptor(context->scan_descriptor_info, | 
					
						
							| 
									
										
										
										
											2023-11-24 17:25:55 -06:00
										 |  |  |  |                 VKD3D_SHADER_DESCRIPTOR_TYPE_SAMPLER, sampler->idx[0].offset))) | 
					
						
							|  |  |  |  |             sampler_space = d->register_space; | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = 0; i < info->combined_sampler_count; ++i) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         s = &info->combined_samplers[i]; | 
					
						
							|  |  |  |  |         if (s->resource_space == resource_space && s->resource_index == resource_idx | 
					
						
							|  |  |  |  |                 && s->sampler_space == sampler_space && s->sampler_index == sampler_idx) | 
					
						
							|  |  |  |  |             return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!vkd3d_array_reserve((void **)&info->combined_samplers, &context->combined_samplers_size, | 
					
						
							|  |  |  |  |             info->combined_sampler_count + 1, sizeof(*info->combined_samplers))) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         ERR("Failed to allocate combined sampler info.\n"); | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     s = &info->combined_samplers[info->combined_sampler_count++]; | 
					
						
							|  |  |  |  |     s->resource_space = resource_space; | 
					
						
							|  |  |  |  |     s->resource_index = resource_idx; | 
					
						
							|  |  |  |  |     s->sampler_space = sampler_space; | 
					
						
							|  |  |  |  |     s->sampler_index = sampler_idx; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-06 12:06:14 -05:00
										 |  |  |  | static void vkd3d_shader_scan_sample_instruction(struct vkd3d_shader_scan_context *context, | 
					
						
							|  |  |  |  |         const struct vkd3d_shader_register *resource, const struct vkd3d_shader_register *sampler) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     unsigned int resource_idx = resource->idx[0].offset; | 
					
						
							|  |  |  |  |     unsigned int sampler_idx = sampler->idx[0].offset; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     vkd3d_shader_scan_combined_sampler_usage(context, resource, sampler); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!context->scan_descriptor_info) | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Sample instructions lowered from 1.x texture instructions have no
 | 
					
						
							|  |  |  |  |      * DCL, so we need to add the resource if it didn't already exist. | 
					
						
							|  |  |  |  |      * Such descriptors have a fixed count, type, etc. */ | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!vkd3d_shader_find_descriptor(context->scan_descriptor_info, | 
					
						
							|  |  |  |  |             VKD3D_SHADER_DESCRIPTOR_TYPE_SRV, resource_idx)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         struct vkd3d_shader_register_range range = {.first = resource_idx, .last = resource_idx}; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         vkd3d_shader_scan_add_descriptor(context, VKD3D_SHADER_DESCRIPTOR_TYPE_SRV, resource, | 
					
						
							|  |  |  |  |                 &range, VKD3D_SHADER_RESOURCE_TEXTURE_2D, VSIR_DATA_F32); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!vkd3d_shader_find_descriptor(context->scan_descriptor_info, | 
					
						
							|  |  |  |  |             VKD3D_SHADER_DESCRIPTOR_TYPE_SAMPLER, sampler_idx)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         struct vkd3d_shader_register_range range = {.first = sampler_idx, .last = sampler_idx}; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         vkd3d_shader_scan_add_descriptor(context, VKD3D_SHADER_DESCRIPTOR_TYPE_SAMPLER, resource, | 
					
						
							|  |  |  |  |                 &range, VKD3D_SHADER_RESOURCE_NONE, VSIR_DATA_UNUSED); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  | static void vkd3d_shader_scan_resource_declaration(struct vkd3d_shader_scan_context *context, | 
					
						
							| 
									
										
										
										
											2020-06-30 15:02:03 +04:30
										 |  |  |  |         const struct vkd3d_shader_resource *resource, enum vkd3d_shader_resource_type resource_type, | 
					
						
							| 
									
										
										
										
											2025-07-17 15:22:54 +02:00
										 |  |  |  |         enum vsir_data_type resource_data_type, unsigned int sample_count, | 
					
						
							| 
									
										
										
										
											2025-04-03 22:46:57 +02:00
										 |  |  |  |         unsigned int structure_stride, bool raw, uint32_t flags) | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-07-31 19:22:33 -05:00
										 |  |  |  |     struct vkd3d_shader_descriptor_info1 *d; | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  |     enum vkd3d_shader_descriptor_type type; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-30 15:02:03 +04:30
										 |  |  |  |     if (resource->reg.reg.type == VKD3DSPR_UAV) | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  |         type = VKD3D_SHADER_DESCRIPTOR_TYPE_UAV; | 
					
						
							|  |  |  |  |     else | 
					
						
							|  |  |  |  |         type = VKD3D_SHADER_DESCRIPTOR_TYPE_SRV; | 
					
						
							| 
									
										
										
										
											2023-07-31 19:22:33 -05:00
										 |  |  |  |     if (!(d = vkd3d_shader_scan_add_descriptor(context, type, &resource->reg.reg, | 
					
						
							|  |  |  |  |             &resource->range, resource_type, resource_data_type))) | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  |     d->sample_count = sample_count; | 
					
						
							| 
									
										
										
										
											2023-07-31 19:56:43 -05:00
										 |  |  |  |     d->structure_stride = structure_stride; | 
					
						
							| 
									
										
										
										
											2023-08-09 18:59:51 -05:00
										 |  |  |  |     if (raw) | 
					
						
							|  |  |  |  |         d->flags |= VKD3D_SHADER_DESCRIPTOR_INFO_FLAG_RAW_BUFFER; | 
					
						
							| 
									
										
										
										
											2023-08-16 00:09:11 +10:00
										 |  |  |  |     if (type == VKD3D_SHADER_DESCRIPTOR_TYPE_UAV) | 
					
						
							|  |  |  |  |         d->uav_flags = flags; | 
					
						
							| 
									
										
										
										
											2020-06-30 15:02:03 +04:30
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void vkd3d_shader_scan_typed_resource_declaration(struct vkd3d_shader_scan_context *context, | 
					
						
							|  |  |  |  |         const struct vkd3d_shader_instruction *instruction) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     const struct vkd3d_shader_semantic *semantic = &instruction->declaration.semantic; | 
					
						
							| 
									
										
										
										
											2022-02-28 12:23:46 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-03 22:46:57 +02:00
										 |  |  |  |     if (semantic->resource_data_type[0] != semantic->resource_data_type[1] | 
					
						
							|  |  |  |  |             || semantic->resource_data_type[0] != semantic->resource_data_type[2] | 
					
						
							|  |  |  |  |             || semantic->resource_data_type[0] != semantic->resource_data_type[3]) | 
					
						
							|  |  |  |  |         FIXME("Resource data types are different (%#x, %#x, %#x, %#x).\n", | 
					
						
							|  |  |  |  |                 semantic->resource_data_type[0], semantic->resource_data_type[1], | 
					
						
							|  |  |  |  |                 semantic->resource_data_type[2], semantic->resource_data_type[3]); | 
					
						
							| 
									
										
										
										
											2022-02-28 12:23:46 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-03 22:46:57 +02:00
										 |  |  |  |     vkd3d_shader_scan_resource_declaration(context, &semantic->resource, semantic->resource_type, | 
					
						
							|  |  |  |  |             semantic->resource_data_type[0], semantic->sample_count, 0, false, instruction->flags); | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  | static int vkd3d_shader_scan_instruction(struct vkd3d_shader_scan_context *context, | 
					
						
							| 
									
										
										
										
											2017-08-21 12:41:07 +02:00
										 |  |  |  |         const struct vkd3d_shader_instruction *instruction) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |     const struct vkd3d_shader_register *sampler_reg; | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |     struct vkd3d_shader_cf_info *cf_info; | 
					
						
							| 
									
										
										
										
											2017-08-21 12:41:07 +02:00
										 |  |  |  |     unsigned int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-20 15:22:01 +01:00
										 |  |  |  |     context->location = instruction->location; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-24 20:34:42 +02:00
										 |  |  |  |     switch (instruction->opcode) | 
					
						
							| 
									
										
										
										
											2019-02-07 09:59:18 +01:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_DCL_CONSTANT_BUFFER: | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  |             vkd3d_shader_scan_constant_buffer_declaration(context, instruction); | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_DCL_SAMPLER: | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  |             vkd3d_shader_scan_sampler_declaration(context, instruction); | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_DCL: | 
					
						
							| 
									
										
										
										
											2023-10-25 13:58:13 -05:00
										 |  |  |  |             if (instruction->declaration.semantic.resource_type == VKD3D_SHADER_RESOURCE_NONE) | 
					
						
							|  |  |  |  |                 break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-11 15:26:25 -05:00
										 |  |  |  |             if (instruction->declaration.semantic.resource.reg.reg.type == VKD3DSPR_COMBINED_SAMPLER) | 
					
						
							|  |  |  |  |             { | 
					
						
							|  |  |  |  |                 vkd3d_shader_scan_combined_sampler_declaration(context, &instruction->declaration.semantic); | 
					
						
							|  |  |  |  |                 break; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             /* fall through */ | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_DCL_UAV_TYPED: | 
					
						
							| 
									
										
										
										
											2020-06-30 15:02:03 +04:30
										 |  |  |  |             vkd3d_shader_scan_typed_resource_declaration(context, instruction); | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_DCL_RESOURCE_RAW: | 
					
						
							|  |  |  |  |         case VSIR_OP_DCL_UAV_RAW: | 
					
						
							| 
									
										
										
										
											2020-06-30 15:02:03 +04:30
										 |  |  |  |             vkd3d_shader_scan_resource_declaration(context, &instruction->declaration.raw_resource.resource, | 
					
						
							| 
									
										
										
										
											2025-07-17 16:47:43 +02:00
										 |  |  |  |                     VKD3D_SHADER_RESOURCE_BUFFER, VSIR_DATA_U32, 0, 0, true, instruction->flags); | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_DCL_RESOURCE_STRUCTURED: | 
					
						
							|  |  |  |  |         case VSIR_OP_DCL_UAV_STRUCTURED: | 
					
						
							| 
									
										
										
										
											2020-06-30 15:02:03 +04:30
										 |  |  |  |             vkd3d_shader_scan_resource_declaration(context, &instruction->declaration.structured_resource.resource, | 
					
						
							| 
									
										
										
										
											2025-07-17 16:47:43 +02:00
										 |  |  |  |                     VKD3D_SHADER_RESOURCE_BUFFER, VSIR_DATA_U32, 0, | 
					
						
							| 
									
										
										
										
											2023-08-16 00:09:11 +10:00
										 |  |  |  |                     instruction->declaration.structured_resource.byte_stride, false, instruction->flags); | 
					
						
							| 
									
										
										
										
											2019-02-07 09:59:18 +01:00
										 |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_DCL_TESSELLATOR_OUTPUT_PRIMITIVE: | 
					
						
							| 
									
										
										
										
											2024-10-22 22:30:13 -04:00
										 |  |  |  |             context->output_primitive = instruction->declaration.tessellator_output_primitive; | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_DCL_TESSELLATOR_PARTITIONING: | 
					
						
							| 
									
										
										
										
											2024-10-22 22:30:13 -04:00
										 |  |  |  |             context->partitioning = instruction->declaration.tessellator_partitioning; | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_IF: | 
					
						
							|  |  |  |  |         case VSIR_OP_IFC: | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |             cf_info = vkd3d_shader_scan_push_cf_info(context); | 
					
						
							|  |  |  |  |             cf_info->type = VKD3D_SHADER_BLOCK_IF; | 
					
						
							|  |  |  |  |             cf_info->inside_block = true; | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_ELSE: | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |             if (!(cf_info = vkd3d_shader_scan_get_current_cf_info(context)) || cf_info->type != VKD3D_SHADER_BLOCK_IF) | 
					
						
							|  |  |  |  |             { | 
					
						
							| 
									
										
										
										
											2020-12-14 22:49:15 -06:00
										 |  |  |  |                 vkd3d_shader_scan_error(context, VKD3D_SHADER_ERROR_TPF_MISMATCHED_CF, | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |                         "Encountered ‘else’ instruction without corresponding ‘if’ block."); | 
					
						
							|  |  |  |  |                 return VKD3D_ERROR_INVALID_SHADER; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             cf_info->inside_block = true; | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_ENDIF: | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |             if (!(cf_info = vkd3d_shader_scan_get_current_cf_info(context)) || cf_info->type != VKD3D_SHADER_BLOCK_IF) | 
					
						
							|  |  |  |  |             { | 
					
						
							| 
									
										
										
										
											2020-12-14 22:49:15 -06:00
										 |  |  |  |                 vkd3d_shader_scan_error(context, VKD3D_SHADER_ERROR_TPF_MISMATCHED_CF, | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |                         "Encountered ‘endif’ instruction without corresponding ‘if’ block."); | 
					
						
							|  |  |  |  |                 return VKD3D_ERROR_INVALID_SHADER; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             vkd3d_shader_scan_pop_cf_info(context); | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_LOOP: | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |             cf_info = vkd3d_shader_scan_push_cf_info(context); | 
					
						
							|  |  |  |  |             cf_info->type = VKD3D_SHADER_BLOCK_LOOP; | 
					
						
							| 
									
										
										
										
											2021-11-08 14:50:17 +01:00
										 |  |  |  |             cf_info->inside_block = true; | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_ENDLOOP: | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |             if (!(cf_info = vkd3d_shader_scan_get_current_cf_info(context)) || cf_info->type != VKD3D_SHADER_BLOCK_LOOP) | 
					
						
							|  |  |  |  |             { | 
					
						
							| 
									
										
										
										
											2020-12-14 22:49:15 -06:00
										 |  |  |  |                 vkd3d_shader_scan_error(context, VKD3D_SHADER_ERROR_TPF_MISMATCHED_CF, | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |                         "Encountered ‘endloop’ instruction without corresponding ‘loop’ block."); | 
					
						
							|  |  |  |  |                 return VKD3D_ERROR_INVALID_SHADER; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             vkd3d_shader_scan_pop_cf_info(context); | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_SWITCH: | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |             cf_info = vkd3d_shader_scan_push_cf_info(context); | 
					
						
							|  |  |  |  |             cf_info->type = VKD3D_SHADER_BLOCK_SWITCH; | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_ENDSWITCH: | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |             if (!(cf_info = vkd3d_shader_scan_get_current_cf_info(context)) | 
					
						
							|  |  |  |  |                     || cf_info->type != VKD3D_SHADER_BLOCK_SWITCH || cf_info->inside_block) | 
					
						
							|  |  |  |  |             { | 
					
						
							| 
									
										
										
										
											2020-12-14 22:49:15 -06:00
										 |  |  |  |                 vkd3d_shader_scan_error(context, VKD3D_SHADER_ERROR_TPF_MISMATCHED_CF, | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |                         "Encountered ‘endswitch’ instruction without corresponding ‘switch’ block."); | 
					
						
							|  |  |  |  |                 return VKD3D_ERROR_INVALID_SHADER; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             vkd3d_shader_scan_pop_cf_info(context); | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_CASE: | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |             if (!(cf_info = vkd3d_shader_scan_get_current_cf_info(context)) | 
					
						
							|  |  |  |  |                     || cf_info->type != VKD3D_SHADER_BLOCK_SWITCH) | 
					
						
							|  |  |  |  |             { | 
					
						
							| 
									
										
										
										
											2020-12-14 22:49:15 -06:00
										 |  |  |  |                 vkd3d_shader_scan_error(context, VKD3D_SHADER_ERROR_TPF_MISMATCHED_CF, | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |                         "Encountered ‘case’ instruction outside switch block."); | 
					
						
							|  |  |  |  |                 return VKD3D_ERROR_INVALID_SHADER; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             cf_info->inside_block = true; | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_DEFAULT: | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |             if (!(cf_info = vkd3d_shader_scan_get_current_cf_info(context)) | 
					
						
							|  |  |  |  |                     || cf_info->type != VKD3D_SHADER_BLOCK_SWITCH) | 
					
						
							|  |  |  |  |             { | 
					
						
							| 
									
										
										
										
											2020-12-14 22:49:15 -06:00
										 |  |  |  |                 vkd3d_shader_scan_error(context, VKD3D_SHADER_ERROR_TPF_MISMATCHED_CF, | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |                         "Encountered ‘default’ instruction outside switch block."); | 
					
						
							|  |  |  |  |                 return VKD3D_ERROR_INVALID_SHADER; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             if (cf_info->has_default) | 
					
						
							|  |  |  |  |             { | 
					
						
							| 
									
										
										
										
											2020-12-14 22:49:15 -06:00
										 |  |  |  |                 vkd3d_shader_scan_error(context, VKD3D_SHADER_ERROR_TPF_MISMATCHED_CF, | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |                         "Encountered duplicate ‘default’ instruction inside the current switch block."); | 
					
						
							|  |  |  |  |                 return VKD3D_ERROR_INVALID_SHADER; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             cf_info->inside_block = true; | 
					
						
							|  |  |  |  |             cf_info->has_default = true; | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_BREAK: | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |             if (!(cf_info = vkd3d_shader_scan_find_innermost_breakable_cf_info(context))) | 
					
						
							|  |  |  |  |             { | 
					
						
							| 
									
										
										
										
											2020-12-14 22:49:15 -06:00
										 |  |  |  |                 vkd3d_shader_scan_error(context, VKD3D_SHADER_ERROR_TPF_MISMATCHED_CF, | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |                         "Encountered ‘break’ instruction outside breakable block."); | 
					
						
							|  |  |  |  |                 return VKD3D_ERROR_INVALID_SHADER; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             cf_info->inside_block = false; | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_BREAKP: | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |             if (!(cf_info = vkd3d_shader_scan_find_innermost_loop_cf_info(context))) | 
					
						
							|  |  |  |  |             { | 
					
						
							| 
									
										
										
										
											2020-12-14 22:49:15 -06:00
										 |  |  |  |                 vkd3d_shader_scan_error(context, VKD3D_SHADER_ERROR_TPF_MISMATCHED_CF, | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |                         "Encountered ‘breakp’ instruction outside loop."); | 
					
						
							|  |  |  |  |                 return VKD3D_ERROR_INVALID_SHADER; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_CONTINUE: | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |             if (!(cf_info = vkd3d_shader_scan_find_innermost_loop_cf_info(context))) | 
					
						
							|  |  |  |  |             { | 
					
						
							| 
									
										
										
										
											2020-12-14 22:49:15 -06:00
										 |  |  |  |                 vkd3d_shader_scan_error(context, VKD3D_SHADER_ERROR_TPF_MISMATCHED_CF, | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |                         "Encountered ‘continue’ instruction outside loop."); | 
					
						
							|  |  |  |  |                 return VKD3D_ERROR_INVALID_SHADER; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             cf_info->inside_block = false; | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_CONTINUEP: | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |             if (!(cf_info = vkd3d_shader_scan_find_innermost_loop_cf_info(context))) | 
					
						
							|  |  |  |  |             { | 
					
						
							| 
									
										
										
										
											2020-12-14 22:49:15 -06:00
										 |  |  |  |                 vkd3d_shader_scan_error(context, VKD3D_SHADER_ERROR_TPF_MISMATCHED_CF, | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |                         "Encountered ‘continue’ instruction outside loop."); | 
					
						
							|  |  |  |  |                 return VKD3D_ERROR_INVALID_SHADER; | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_RET: | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |             if (context->cf_info_count) | 
					
						
							|  |  |  |  |                 context->cf_info[context->cf_info_count - 1].inside_block = false; | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_TEX: | 
					
						
							|  |  |  |  |         case VSIR_OP_TEXBEM: | 
					
						
							|  |  |  |  |         case VSIR_OP_TEXBEML: | 
					
						
							|  |  |  |  |         case VSIR_OP_TEXDP3TEX: | 
					
						
							|  |  |  |  |         case VSIR_OP_TEXM3x2TEX: | 
					
						
							|  |  |  |  |         case VSIR_OP_TEXM3x3SPEC: | 
					
						
							|  |  |  |  |         case VSIR_OP_TEXM3x3TEX: | 
					
						
							|  |  |  |  |         case VSIR_OP_TEXM3x3VSPEC: | 
					
						
							|  |  |  |  |         case VSIR_OP_TEXREG2AR: | 
					
						
							|  |  |  |  |         case VSIR_OP_TEXREG2GB: | 
					
						
							|  |  |  |  |         case VSIR_OP_TEXREG2RGB: | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |             sampler_reg = &instruction->dst[0].reg; | 
					
						
							|  |  |  |  |             vkd3d_shader_scan_combined_sampler_usage(context, sampler_reg, sampler_reg); | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_GATHER4: | 
					
						
							|  |  |  |  |         case VSIR_OP_GATHER4_C: | 
					
						
							|  |  |  |  |         case VSIR_OP_SAMPLE_B: | 
					
						
							|  |  |  |  |         case VSIR_OP_SAMPLE_C: | 
					
						
							|  |  |  |  |         case VSIR_OP_SAMPLE_C_LZ: | 
					
						
							|  |  |  |  |         case VSIR_OP_SAMPLE_GRAD: | 
					
						
							|  |  |  |  |         case VSIR_OP_SAMPLE_LOD: | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |             vkd3d_shader_scan_combined_sampler_usage(context, &instruction->src[1].reg, &instruction->src[2].reg); | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-08-06 12:06:14 -05:00
										 |  |  |  |         case VSIR_OP_SAMPLE: | 
					
						
							|  |  |  |  |             vkd3d_shader_scan_sample_instruction(context, &instruction->src[1].reg, &instruction->src[2].reg); | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_GATHER4_PO: | 
					
						
							|  |  |  |  |         case VSIR_OP_GATHER4_PO_C: | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |             vkd3d_shader_scan_combined_sampler_usage(context, &instruction->src[2].reg, &instruction->src[3].reg); | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_LD: | 
					
						
							|  |  |  |  |         case VSIR_OP_LD2DMS: | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |             vkd3d_shader_scan_combined_sampler_usage(context, &instruction->src[1].reg, NULL); | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_BUFINFO: | 
					
						
							|  |  |  |  |         case VSIR_OP_SAMPLE_INFO: | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |             if (instruction->src[0].reg.type == VKD3DSPR_RESOURCE) | 
					
						
							|  |  |  |  |                 vkd3d_shader_scan_combined_sampler_usage(context, &instruction->src[0].reg, NULL); | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_LD_RAW: | 
					
						
							|  |  |  |  |         case VSIR_OP_RESINFO: | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |             if (instruction->src[1].reg.type == VKD3DSPR_RESOURCE) | 
					
						
							|  |  |  |  |                 vkd3d_shader_scan_combined_sampler_usage(context, &instruction->src[1].reg, NULL); | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2025-06-24 13:11:57 +02:00
										 |  |  |  |         case VSIR_OP_LD_STRUCTURED: | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |             if (instruction->src[2].reg.type == VKD3DSPR_RESOURCE) | 
					
						
							|  |  |  |  |                 vkd3d_shader_scan_combined_sampler_usage(context, &instruction->src[2].reg, NULL); | 
					
						
							|  |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2019-02-07 09:59:18 +01:00
										 |  |  |  |         default: | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-10-22 01:49:15 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 12:41:07 +02:00
										 |  |  |  |     if (vkd3d_shader_instruction_is_uav_read(instruction)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         for (i = 0; i < instruction->dst_count; ++i) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             if (instruction->dst[i].reg.type == VKD3DSPR_UAV) | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:27 +04:30
										 |  |  |  |                 vkd3d_shader_scan_record_uav_read(context, &instruction->dst[i].reg); | 
					
						
							| 
									
										
										
										
											2017-08-21 12:41:07 +02:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |         for (i = 0; i < instruction->src_count; ++i) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             if (instruction->src[i].reg.type == VKD3DSPR_UAV) | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:27 +04:30
										 |  |  |  |                 vkd3d_shader_scan_record_uav_read(context, &instruction->src[i].reg); | 
					
						
							| 
									
										
										
										
											2017-08-21 12:41:07 +02:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-09-01 11:27:58 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (vkd3d_shader_instruction_is_uav_counter(instruction)) | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:26 +04:30
										 |  |  |  |         vkd3d_shader_scan_record_uav_counter(context, &instruction->src[0].reg); | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-18 13:44:05 +10:00
										 |  |  |  |     if (vkd3d_shader_instruction_is_uav_atomic_op(instruction)) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         for (i = 0; i < instruction->dst_count; ++i) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             if (instruction->dst[i].reg.type == VKD3DSPR_UAV) | 
					
						
							|  |  |  |  |                 vkd3d_shader_scan_record_uav_atomic_op(context, &instruction->dst[i].reg); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |     return VKD3D_OK; | 
					
						
							| 
									
										
										
										
											2017-08-21 12:41:07 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-17 15:22:54 +02:00
										 |  |  |  | static enum vkd3d_shader_resource_data_type vkd3d_resource_data_type_from_data_type(enum vsir_data_type data_type) | 
					
						
							| 
									
										
										
										
											2025-04-03 22:46:57 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     switch (data_type) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-07-17 16:03:55 +02:00
										 |  |  |  |         case VSIR_DATA_F32: | 
					
						
							|  |  |  |  |             return VKD3D_SHADER_RESOURCE_DATA_FLOAT; | 
					
						
							|  |  |  |  |         case VSIR_DATA_F64: | 
					
						
							|  |  |  |  |             return VKD3D_SHADER_RESOURCE_DATA_DOUBLE; | 
					
						
							| 
									
										
										
										
											2025-07-17 16:29:38 +02:00
										 |  |  |  |         case VSIR_DATA_I32: | 
					
						
							|  |  |  |  |             return VKD3D_SHADER_RESOURCE_DATA_INT; | 
					
						
							| 
									
										
										
										
											2025-07-17 16:47:43 +02:00
										 |  |  |  |         case VSIR_DATA_U32: | 
					
						
							|  |  |  |  |             return VKD3D_SHADER_RESOURCE_DATA_UINT; | 
					
						
							| 
									
										
										
										
											2025-07-21 15:09:30 +02:00
										 |  |  |  |         case VSIR_DATA_SNORM: | 
					
						
							|  |  |  |  |             return VKD3D_SHADER_RESOURCE_DATA_SNORM; | 
					
						
							| 
									
										
										
										
											2025-07-21 15:05:46 +02:00
										 |  |  |  |         case VSIR_DATA_UNORM: | 
					
						
							|  |  |  |  |             return VKD3D_SHADER_RESOURCE_DATA_UNORM; | 
					
						
							| 
									
										
										
										
											2025-07-21 15:15:35 +02:00
										 |  |  |  |         case VSIR_DATA_MIXED: | 
					
						
							|  |  |  |  |             return VKD3D_SHADER_RESOURCE_DATA_MIXED; | 
					
						
							| 
									
										
										
										
											2025-07-21 15:17:51 +02:00
										 |  |  |  |         case VSIR_DATA_CONTINUED: | 
					
						
							|  |  |  |  |             return VKD3D_SHADER_RESOURCE_DATA_CONTINUED; | 
					
						
							| 
									
										
										
										
											2025-07-21 15:31:59 +02:00
										 |  |  |  |         case VSIR_DATA_UNUSED: | 
					
						
							|  |  |  |  |             return VKD3D_SHADER_RESOURCE_DATA_NONE; | 
					
						
							| 
									
										
										
										
											2025-04-03 22:46:57 +02:00
										 |  |  |  |         default: | 
					
						
							|  |  |  |  |             ERR("Invalid resource data type %#x.\n", data_type); | 
					
						
							|  |  |  |  |             return VKD3D_SHADER_RESOURCE_DATA_FLOAT; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static enum vkd3d_result convert_descriptor_info(struct vkd3d_shader_scan_context *context, | 
					
						
							|  |  |  |  |         struct vkd3d_shader_scan_descriptor_info *info, const struct vkd3d_shader_scan_descriptor_info1 *info1) | 
					
						
							| 
									
										
										
										
											2023-07-31 18:52:21 -05:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     unsigned int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!(info->descriptors = vkd3d_calloc(info1->descriptor_count, sizeof(*info->descriptors)))) | 
					
						
							|  |  |  |  |         return VKD3D_ERROR_OUT_OF_MEMORY; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = 0; i < info1->descriptor_count; ++i) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         const struct vkd3d_shader_descriptor_info1 *src = &info1->descriptors[i]; | 
					
						
							|  |  |  |  |         struct vkd3d_shader_descriptor_info *dst = &info->descriptors[i]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         dst->type = src->type; | 
					
						
							|  |  |  |  |         dst->register_space = src->register_space; | 
					
						
							|  |  |  |  |         dst->register_index = src->register_index; | 
					
						
							|  |  |  |  |         dst->resource_type = src->resource_type; | 
					
						
							| 
									
										
										
										
											2025-04-03 22:46:57 +02:00
										 |  |  |  |         dst->resource_data_type = vkd3d_resource_data_type_from_data_type(src->resource_data_type); | 
					
						
							| 
									
										
										
										
											2023-07-31 18:52:21 -05:00
										 |  |  |  |         dst->flags = src->flags; | 
					
						
							|  |  |  |  |         dst->count = src->count; | 
					
						
							| 
									
										
										
										
											2025-04-03 22:46:57 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-28 11:36:33 +01:00
										 |  |  |  |         if (context->api_version <= VKD3D_SHADER_API_VERSION_1_15 | 
					
						
							|  |  |  |  |                 && dst->type == VKD3D_SHADER_DESCRIPTOR_TYPE_SAMPLER) | 
					
						
							|  |  |  |  |             dst->resource_data_type = VKD3D_SHADER_RESOURCE_DATA_UINT; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-03 22:46:57 +02:00
										 |  |  |  |         if (context->api_version < VKD3D_SHADER_API_VERSION_1_3 | 
					
						
							|  |  |  |  |                 && dst->resource_data_type >= VKD3D_SHADER_RESOURCE_DATA_MIXED) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             ERR("Invalid resource data type %#x for API version %#x.\n", | 
					
						
							|  |  |  |  |                     src->resource_data_type, context->api_version); | 
					
						
							|  |  |  |  |             dst->resource_data_type = VKD3D_SHADER_RESOURCE_DATA_FLOAT; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-07-31 18:52:21 -05:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |     info->descriptor_count = info1->descriptor_count; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return VKD3D_OK; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-14 23:14:06 +01:00
										 |  |  |  | void vkd3d_shader_free_scan_descriptor_info1(struct vkd3d_shader_scan_descriptor_info1 *scan_descriptor_info) | 
					
						
							| 
									
										
										
										
											2023-07-31 18:52:21 -05:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     TRACE("scan_descriptor_info %p.\n", scan_descriptor_info); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     vkd3d_free(scan_descriptor_info->descriptors); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-18 23:41:40 +02:00
										 |  |  |  | static int vsir_program_scan(struct vsir_program *program, const struct vkd3d_shader_compile_info *compile_info, | 
					
						
							| 
									
										
										
										
											2025-03-14 23:14:06 +01:00
										 |  |  |  |         struct vkd3d_shader_message_context *message_context, bool add_descriptor_info) | 
					
						
							| 
									
										
										
										
											2017-08-21 12:41:07 +02:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-07-22 18:51:36 +02:00
										 |  |  |  |     struct vsir_program_iterator it = vsir_program_iterator(&program->instructions); | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |     struct vkd3d_shader_scan_combined_resource_sampler_info *combined_sampler_info; | 
					
						
							| 
									
										
										
										
											2024-10-22 22:30:13 -04:00
										 |  |  |  |     struct vkd3d_shader_scan_hull_shader_tessellation_info *tessellation_info; | 
					
						
							| 
									
										
										
										
											2025-05-31 19:53:41 +02:00
										 |  |  |  |     struct vkd3d_shader_scan_thread_group_size_info *thread_group_size_info; | 
					
						
							| 
									
										
										
										
											2023-07-31 18:52:21 -05:00
										 |  |  |  |     struct vkd3d_shader_scan_descriptor_info *descriptor_info; | 
					
						
							| 
									
										
										
										
											2023-02-23 16:21:18 -06:00
										 |  |  |  |     struct vkd3d_shader_scan_signature_info *signature_info; | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  |     struct vkd3d_shader_scan_context context; | 
					
						
							| 
									
										
										
										
											2025-07-22 18:51:36 +02:00
										 |  |  |  |     struct vkd3d_shader_instruction *ins; | 
					
						
							| 
									
										
										
										
											2023-01-20 12:23:27 +10:00
										 |  |  |  |     int ret = VKD3D_OK; | 
					
						
							|  |  |  |  |     unsigned int i; | 
					
						
							| 
									
										
										
										
											2017-08-21 12:41:07 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-31 18:52:21 -05:00
										 |  |  |  |     descriptor_info = vkd3d_find_struct(compile_info->next, SCAN_DESCRIPTOR_INFO); | 
					
						
							| 
									
										
										
										
											2025-03-14 23:14:06 +01:00
										 |  |  |  |     if (descriptor_info) | 
					
						
							|  |  |  |  |         add_descriptor_info = true; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 16:21:18 -06:00
										 |  |  |  |     signature_info = vkd3d_find_struct(compile_info->next, SCAN_SIGNATURE_INFO); | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:55 +04:30
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |     if ((combined_sampler_info = vkd3d_find_struct(compile_info->next, SCAN_COMBINED_RESOURCE_SAMPLER_INFO))) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         combined_sampler_info->combined_samplers = NULL; | 
					
						
							|  |  |  |  |         combined_sampler_info->combined_sampler_count = 0; | 
					
						
							| 
									
										
										
										
											2025-03-14 23:14:06 +01:00
										 |  |  |  |         add_descriptor_info = true; | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-22 22:30:13 -04:00
										 |  |  |  |     tessellation_info = vkd3d_find_struct(compile_info->next, SCAN_HULL_SHADER_TESSELLATION_INFO); | 
					
						
							| 
									
										
										
										
											2025-05-31 19:53:41 +02:00
										 |  |  |  |     thread_group_size_info = vkd3d_find_struct(compile_info->next, SCAN_THREAD_GROUP_SIZE_INFO); | 
					
						
							| 
									
										
										
										
											2024-10-22 22:30:13 -04:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-18 23:29:18 +02:00
										 |  |  |  |     vkd3d_shader_scan_context_init(&context, &program->shader_version, compile_info, | 
					
						
							| 
									
										
										
										
											2025-03-14 23:14:06 +01:00
										 |  |  |  |             add_descriptor_info ? &program->descriptors : NULL, combined_sampler_info, message_context); | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-15 16:03:50 +01:00
										 |  |  |  |     if (add_descriptor_info) | 
					
						
							|  |  |  |  |         program->has_descriptor_info = true; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-28 14:22:33 +04:30
										 |  |  |  |     if (TRACE_ON()) | 
					
						
							| 
									
										
										
										
											2024-10-08 15:29:52 +02:00
										 |  |  |  |         vsir_program_trace(program); | 
					
						
							| 
									
										
										
										
											2020-07-28 14:22:33 +04:30
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-22 18:51:36 +02:00
										 |  |  |  |     for (ins = vsir_program_iterator_head(&it); ins; ins = vsir_program_iterator_next(&it)) | 
					
						
							| 
									
										
										
										
											2017-08-21 12:41:07 +02:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-07-22 18:51:36 +02:00
										 |  |  |  |         if ((ret = vkd3d_shader_scan_instruction(&context, ins)) < 0) | 
					
						
							| 
									
										
										
										
											2023-01-20 12:23:27 +10:00
										 |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2017-08-21 12:41:07 +02:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-18 23:29:18 +02:00
										 |  |  |  |     for (i = 0; i < ARRAY_SIZE(program->flat_constant_count); ++i) | 
					
						
							| 
									
										
										
										
											2023-02-28 00:07:14 -06:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |         struct vkd3d_shader_register_range range = {.space = 0, .first = i, .last = i}; | 
					
						
							| 
									
										
										
										
											2023-07-31 18:59:16 -05:00
										 |  |  |  |         struct vkd3d_shader_register reg = {.idx[0].offset = i, .idx_count = 1}; | 
					
						
							| 
									
										
										
										
											2024-04-18 23:29:18 +02:00
										 |  |  |  |         unsigned int size = program->flat_constant_count[i]; | 
					
						
							| 
									
										
										
										
											2023-07-31 19:34:23 -05:00
										 |  |  |  |         struct vkd3d_shader_descriptor_info1 *d; | 
					
						
							| 
									
										
										
										
											2023-02-28 00:07:14 -06:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-18 23:29:18 +02:00
										 |  |  |  |         if (size) | 
					
						
							| 
									
										
										
										
											2023-07-31 19:34:23 -05:00
										 |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2025-07-17 16:47:43 +02:00
										 |  |  |  |             if ((d = vkd3d_shader_scan_add_descriptor(&context, VKD3D_SHADER_DESCRIPTOR_TYPE_CBV, | 
					
						
							|  |  |  |  |                     ®, &range, VKD3D_SHADER_RESOURCE_BUFFER, VSIR_DATA_U32))) | 
					
						
							| 
									
										
										
										
											2023-07-31 19:34:23 -05:00
										 |  |  |  |                 d->buffer_size = size * 16; | 
					
						
							|  |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-02-28 00:07:14 -06:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 16:21:18 -06:00
										 |  |  |  |     if (!ret && signature_info) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-04-18 23:29:18 +02:00
										 |  |  |  |         if (!vkd3d_shader_signature_from_shader_signature(&signature_info->input, &program->input_signature) | 
					
						
							| 
									
										
										
										
											2023-02-23 16:21:18 -06:00
										 |  |  |  |                 || !vkd3d_shader_signature_from_shader_signature(&signature_info->output, | 
					
						
							| 
									
										
										
										
											2024-04-18 23:29:18 +02:00
										 |  |  |  |                         &program->output_signature) | 
					
						
							| 
									
										
										
										
											2023-02-23 16:21:18 -06:00
										 |  |  |  |                 || !vkd3d_shader_signature_from_shader_signature(&signature_info->patch_constant, | 
					
						
							| 
									
										
										
										
											2024-04-18 23:29:18 +02:00
										 |  |  |  |                         &program->patch_constant_signature)) | 
					
						
							| 
									
										
										
										
											2023-02-23 16:21:18 -06:00
										 |  |  |  |         { | 
					
						
							|  |  |  |  |             ret = VKD3D_ERROR_OUT_OF_MEMORY; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-31 18:52:21 -05:00
										 |  |  |  |     if (!ret && descriptor_info) | 
					
						
							| 
									
										
										
										
											2025-04-03 22:46:57 +02:00
										 |  |  |  |         ret = convert_descriptor_info(&context, descriptor_info, &program->descriptors); | 
					
						
							| 
									
										
										
										
											2023-07-31 18:52:21 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-22 22:30:13 -04:00
										 |  |  |  |     if (!ret && tessellation_info) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         tessellation_info->output_primitive = context.output_primitive; | 
					
						
							|  |  |  |  |         tessellation_info->partitioning = context.partitioning; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-31 19:53:41 +02:00
										 |  |  |  |     if (!ret && thread_group_size_info) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         thread_group_size_info->x = program->thread_group_size.x; | 
					
						
							|  |  |  |  |         thread_group_size_info->y = program->thread_group_size.y; | 
					
						
							|  |  |  |  |         thread_group_size_info->z = program->thread_group_size.z; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-31 18:36:30 -05:00
										 |  |  |  |     if (ret < 0) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  |         if (combined_sampler_info) | 
					
						
							|  |  |  |  |             vkd3d_shader_free_scan_combined_resource_sampler_info(combined_sampler_info); | 
					
						
							| 
									
										
										
										
											2023-07-31 18:52:21 -05:00
										 |  |  |  |         if (descriptor_info) | 
					
						
							|  |  |  |  |             vkd3d_shader_free_scan_descriptor_info(descriptor_info); | 
					
						
							| 
									
										
										
										
											2023-07-31 18:36:30 -05:00
										 |  |  |  |         if (signature_info) | 
					
						
							|  |  |  |  |             vkd3d_shader_free_scan_signature_info(signature_info); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-03-14 23:14:06 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:54 +04:30
										 |  |  |  |     vkd3d_shader_scan_context_cleanup(&context); | 
					
						
							| 
									
										
										
										
											2021-10-06 17:11:49 +02:00
										 |  |  |  |     return ret; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:58 -05:00
										 |  |  |  | int vkd3d_shader_scan(const struct vkd3d_shader_compile_info *compile_info, char **messages) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct vkd3d_shader_message_context message_context; | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:55 +02:00
										 |  |  |  |     struct vkd3d_shader_code reflection_data = {0}; | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  |     struct shader_dump_data dump_data; | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:55 +02:00
										 |  |  |  |     struct vsir_program program; | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:58 -05:00
										 |  |  |  |     int ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     TRACE("compile_info %p, messages %p.\n", compile_info, messages); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (messages) | 
					
						
							|  |  |  |  |         *messages = NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-12-28 16:20:17 -06:00
										 |  |  |  |     if ((ret = vkd3d_shader_validate_compile_info(compile_info, false)) < 0) | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:58 -05:00
										 |  |  |  |         return ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 16:21:18 -06:00
										 |  |  |  |     init_scan_signature_info(compile_info); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-15 11:01:32 -06:00
										 |  |  |  |     vkd3d_shader_message_context_init(&message_context, compile_info->log_level); | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:58 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  |     fill_shader_dump_data(compile_info, &dump_data); | 
					
						
							| 
									
										
										
										
											2025-01-27 13:27:24 -03:00
										 |  |  |  |     vkd3d_shader_dump_shader(&dump_data, compile_info->source.code, compile_info->source.size, SHADER_DUMP_TYPE_SOURCE); | 
					
						
							| 
									
										
										
										
											2023-10-21 21:35:08 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:55 +02:00
										 |  |  |  |     if (!(ret = vsir_parse(compile_info, vkd3d_shader_init_config_flags(), | 
					
						
							|  |  |  |  |             &dump_data, &message_context, &program, &reflection_data))) | 
					
						
							| 
									
										
										
										
											2024-03-01 11:46:51 +01:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:55 +02:00
										 |  |  |  |         ret = vsir_program_scan(&program, compile_info, &message_context, false); | 
					
						
							|  |  |  |  |         vkd3d_shader_free_shader_code(&reflection_data); | 
					
						
							|  |  |  |  |         vsir_program_cleanup(&program); | 
					
						
							| 
									
										
										
										
											2021-09-01 15:45:07 +02:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:58 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     vkd3d_shader_message_context_trace_messages(&message_context); | 
					
						
							| 
									
										
										
										
											2020-10-09 01:14:01 -05:00
										 |  |  |  |     if (!vkd3d_shader_message_context_copy_messages(&message_context, messages)) | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:58 -05:00
										 |  |  |  |         ret = VKD3D_ERROR_OUT_OF_MEMORY; | 
					
						
							|  |  |  |  |     vkd3d_shader_message_context_cleanup(&message_context); | 
					
						
							|  |  |  |  |     return ret; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:49 +02:00
										 |  |  |  | static int vsir_program_compile(struct vsir_program *program, const struct vkd3d_shader_code *reflection_data, | 
					
						
							| 
									
										
										
										
											2025-07-22 22:04:01 +02:00
										 |  |  |  |         uint64_t config_flags, const struct vkd3d_shader_compile_info *compile_info, | 
					
						
							|  |  |  |  |         struct vkd3d_shader_code *out, struct vkd3d_shader_message_context *message_context) | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:59 -05:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-05-06 14:00:58 +02:00
										 |  |  |  |     struct vkd3d_shader_scan_combined_resource_sampler_info combined_sampler_info; | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:59 -05:00
										 |  |  |  |     struct vkd3d_shader_compile_info scan_info; | 
					
						
							| 
									
										
										
										
											2025-08-12 15:49:05 +02:00
										 |  |  |  |     enum vsir_asm_flags asm_flags; | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:59 -05:00
										 |  |  |  |     int ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     scan_info = *compile_info; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-12 16:53:39 -06:00
										 |  |  |  |     switch (compile_info->target_type) | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:59 -05:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-03-12 16:53:39 -06:00
										 |  |  |  |         case VKD3D_SHADER_TARGET_D3D_ASM: | 
					
						
							| 
									
										
										
										
											2025-07-22 22:14:22 +02:00
										 |  |  |  |             if ((ret = vsir_program_scan(program, &scan_info, message_context, true)) < 0) | 
					
						
							|  |  |  |  |                 return ret; | 
					
						
							| 
									
										
										
										
											2025-08-12 15:49:05 +02:00
										 |  |  |  |             asm_flags = VSIR_ASM_FLAG_NONE; | 
					
						
							|  |  |  |  |             if (program->shader_version.major < 6 && compile_info->source_type != VKD3D_SHADER_SOURCE_DXBC_TPF | 
					
						
							|  |  |  |  |                     && compile_info->source_type != VKD3D_SHADER_SOURCE_D3D_BYTECODE) | 
					
						
							|  |  |  |  |                 asm_flags |= VSIR_ASM_FLAG_ALLOCATE_TEMPS; | 
					
						
							|  |  |  |  |             ret = d3d_asm_compile(program, compile_info, out, asm_flags, message_context); | 
					
						
							| 
									
										
										
										
											2022-03-12 16:53:39 -06:00
										 |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:59 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-22 22:04:01 +02:00
										 |  |  |  |         case VKD3D_SHADER_TARGET_D3D_BYTECODE: | 
					
						
							|  |  |  |  |             if ((ret = vsir_program_scan(program, &scan_info, message_context, true)) < 0) | 
					
						
							|  |  |  |  |                 return ret; | 
					
						
							|  |  |  |  |             ret = d3dbc_compile(program, config_flags, compile_info, reflection_data, out, message_context); | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         case VKD3D_SHADER_TARGET_DXBC_TPF: | 
					
						
							|  |  |  |  |             if ((ret = vsir_program_scan(program, &scan_info, message_context, true)) < 0) | 
					
						
							|  |  |  |  |                 return ret; | 
					
						
							|  |  |  |  |             ret = tpf_compile(program, config_flags, reflection_data, out, message_context); | 
					
						
							|  |  |  |  |             break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-12 16:53:39 -06:00
										 |  |  |  |         case VKD3D_SHADER_TARGET_GLSL: | 
					
						
							| 
									
										
										
										
											2024-05-06 14:00:58 +02:00
										 |  |  |  |             combined_sampler_info.type = VKD3D_SHADER_STRUCTURE_TYPE_SCAN_COMBINED_RESOURCE_SAMPLER_INFO; | 
					
						
							|  |  |  |  |             combined_sampler_info.next = scan_info.next; | 
					
						
							|  |  |  |  |             scan_info.next = &combined_sampler_info; | 
					
						
							| 
									
										
										
										
											2025-03-14 23:14:06 +01:00
										 |  |  |  |             if ((ret = vsir_program_scan(program, &scan_info, message_context, true)) < 0) | 
					
						
							| 
									
										
										
										
											2023-12-13 15:50:01 -06:00
										 |  |  |  |                 return ret; | 
					
						
							| 
									
										
										
										
											2025-03-14 23:14:06 +01:00
										 |  |  |  |             ret = glsl_compile(program, config_flags, | 
					
						
							| 
									
										
										
										
											2024-05-06 14:00:58 +02:00
										 |  |  |  |                     &combined_sampler_info, compile_info, out, message_context); | 
					
						
							|  |  |  |  |             vkd3d_shader_free_scan_combined_resource_sampler_info(&combined_sampler_info); | 
					
						
							| 
									
										
										
										
											2022-03-12 16:53:39 -06:00
										 |  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2021-08-19 20:18:13 +05:30
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-12 16:53:39 -06:00
										 |  |  |  |         case VKD3D_SHADER_TARGET_SPIRV_BINARY: | 
					
						
							|  |  |  |  |         case VKD3D_SHADER_TARGET_SPIRV_TEXT: | 
					
						
							| 
									
										
										
										
											2025-03-14 23:14:06 +01:00
										 |  |  |  |             if ((ret = vsir_program_scan(program, &scan_info, message_context, true)) < 0) | 
					
						
							| 
									
										
										
										
											2023-12-13 15:50:01 -06:00
										 |  |  |  |                 return ret; | 
					
						
							| 
									
										
										
										
											2025-03-14 23:14:06 +01:00
										 |  |  |  |             ret = spirv_compile(program, config_flags, compile_info, out, message_context); | 
					
						
							| 
									
										
										
										
											2022-03-12 16:53:39 -06:00
										 |  |  |  |             break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-27 10:34:57 +08:00
										 |  |  |  |         case VKD3D_SHADER_TARGET_MSL: | 
					
						
							| 
									
										
										
										
											2025-03-14 23:14:06 +01:00
										 |  |  |  |             if ((ret = vsir_program_scan(program, &scan_info, message_context, true)) < 0) | 
					
						
							| 
									
										
										
										
											2024-10-09 16:36:06 +08:00
										 |  |  |  |                 return ret; | 
					
						
							| 
									
										
										
										
											2025-03-14 23:14:06 +01:00
										 |  |  |  |             ret = msl_compile(program, config_flags, compile_info, out, message_context); | 
					
						
							| 
									
										
										
										
											2024-08-27 10:34:57 +08:00
										 |  |  |  |             break; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-12 16:53:39 -06:00
										 |  |  |  |         default: | 
					
						
							|  |  |  |  |             /* Validation should prevent us from reaching this. */ | 
					
						
							| 
									
										
										
										
											2023-12-13 15:50:01 -06:00
										 |  |  |  |             vkd3d_unreachable(); | 
					
						
							| 
									
										
										
										
											2022-03-12 16:53:39 -06:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return ret; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-08-19 20:18:13 +05:30
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:49 +02:00
										 |  |  |  | static int fx_compile(const struct vkd3d_shader_compile_info *compile_info, | 
					
						
							| 
									
										
										
										
											2025-01-27 13:27:24 -03:00
										 |  |  |  |         const struct shader_dump_data *dump_data, struct vkd3d_shader_code *out, | 
					
						
							|  |  |  |  |         struct vkd3d_shader_message_context *message_context) | 
					
						
							| 
									
										
										
										
											2020-09-25 16:53:00 -05:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2025-07-22 22:56:23 +02:00
										 |  |  |  |     struct vkd3d_shader_compile_info preprocessed_info; | 
					
						
							| 
									
										
										
										
											2021-01-27 10:29:44 -06:00
										 |  |  |  |     struct vkd3d_shader_code preprocessed; | 
					
						
							|  |  |  |  |     int ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if ((ret = preproc_lexer_parse(compile_info, &preprocessed, message_context))) | 
					
						
							|  |  |  |  |         return ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-27 13:27:24 -03:00
										 |  |  |  |     vkd3d_shader_dump_shader(dump_data, preprocessed.code, preprocessed.size, SHADER_DUMP_TYPE_PREPROC); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-22 22:56:23 +02:00
										 |  |  |  |     preprocessed_info = *compile_info; | 
					
						
							|  |  |  |  |     preprocessed_info.source = preprocessed; | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:49 +02:00
										 |  |  |  |     ret = hlsl_compile_effect(&preprocessed_info, message_context, out); | 
					
						
							| 
									
										
										
										
											2021-01-27 10:29:44 -06:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     vkd3d_shader_free_shader_code(&preprocessed); | 
					
						
							|  |  |  |  |     return ret; | 
					
						
							| 
									
										
										
										
											2020-09-25 16:53:00 -05:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | int vkd3d_shader_compile(const struct vkd3d_shader_compile_info *compile_info, | 
					
						
							|  |  |  |  |         struct vkd3d_shader_code *out, char **messages) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct vkd3d_shader_message_context message_context; | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  |     struct shader_dump_data dump_data; | 
					
						
							| 
									
										
										
										
											2020-09-25 16:53:00 -05:00
										 |  |  |  |     int ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     TRACE("compile_info %p, out %p, messages %p.\n", compile_info, out, messages); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (messages) | 
					
						
							|  |  |  |  |         *messages = NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 09:29:11 -06:00
										 |  |  |  |     if ((ret = vkd3d_shader_validate_compile_info(compile_info, true)) < 0) | 
					
						
							| 
									
										
										
										
											2020-09-25 16:53:00 -05:00
										 |  |  |  |         return ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 16:21:18 -06:00
										 |  |  |  |     init_scan_signature_info(compile_info); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-15 11:01:32 -06:00
										 |  |  |  |     vkd3d_shader_message_context_init(&message_context, compile_info->log_level); | 
					
						
							| 
									
										
										
										
											2020-09-25 16:53:00 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  |     fill_shader_dump_data(compile_info, &dump_data); | 
					
						
							| 
									
										
										
										
											2025-01-27 13:27:24 -03:00
										 |  |  |  |     vkd3d_shader_dump_shader(&dump_data, compile_info->source.code, compile_info->source.size, SHADER_DUMP_TYPE_SOURCE); | 
					
						
							| 
									
										
										
										
											2023-10-21 21:35:08 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:49 +02:00
										 |  |  |  |     if (compile_info->source_type == VKD3D_SHADER_SOURCE_HLSL | 
					
						
							|  |  |  |  |             && compile_info->target_type == VKD3D_SHADER_TARGET_FX) | 
					
						
							| 
									
										
										
										
											2020-09-25 16:53:00 -05:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:49 +02:00
										 |  |  |  |         ret = fx_compile(compile_info, &dump_data, out, &message_context); | 
					
						
							| 
									
										
										
										
											2024-03-01 11:47:57 +01:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |  |     else if (compile_info->source_type == VKD3D_SHADER_SOURCE_FX) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         ret = fx_parse(compile_info, out, &message_context); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-06-12 00:38:10 +02:00
										 |  |  |  |     else if (compile_info->source_type == VKD3D_SHADER_SOURCE_TX) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         ret = tx_parse(compile_info, out, &message_context); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-03-01 11:47:57 +01:00
										 |  |  |  |     else | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-05-16 11:42:16 +02:00
										 |  |  |  |         uint64_t config_flags = vkd3d_shader_init_config_flags(); | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:49 +02:00
										 |  |  |  |         struct vkd3d_shader_code reflection_data = {0}; | 
					
						
							| 
									
										
										
										
											2024-05-16 11:42:29 +02:00
										 |  |  |  |         struct vsir_program program; | 
					
						
							| 
									
										
										
										
											2024-03-01 12:00:49 +01:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:49 +02:00
										 |  |  |  |         if (!(ret = vsir_parse(compile_info, config_flags, &dump_data, | 
					
						
							|  |  |  |  |                 &message_context, &program, &reflection_data))) | 
					
						
							| 
									
										
										
										
											2024-03-01 12:00:49 +01:00
										 |  |  |  |         { | 
					
						
							| 
									
										
										
										
											2025-08-05 21:17:49 +02:00
										 |  |  |  |             ret = vsir_program_compile(&program, &reflection_data, config_flags, compile_info, out, &message_context); | 
					
						
							|  |  |  |  |             vkd3d_shader_free_shader_code(&reflection_data); | 
					
						
							| 
									
										
										
										
											2024-05-16 11:42:29 +02:00
										 |  |  |  |             vsir_program_cleanup(&program); | 
					
						
							| 
									
										
										
										
											2024-03-01 12:00:49 +01:00
										 |  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-09-25 16:53:00 -05:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-19 20:59:56 +02:00
										 |  |  |  |     if (ret >= 0) | 
					
						
							| 
									
										
										
										
											2025-01-27 13:27:24 -03:00
										 |  |  |  |         vkd3d_shader_dump_shader(&dump_data, out->code, out->size, SHADER_DUMP_TYPE_TARGET); | 
					
						
							| 
									
										
										
										
											2024-08-12 16:27:00 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:59 -05:00
										 |  |  |  |     vkd3d_shader_message_context_trace_messages(&message_context); | 
					
						
							| 
									
										
										
										
											2020-10-09 01:14:01 -05:00
										 |  |  |  |     if (!vkd3d_shader_message_context_copy_messages(&message_context, messages)) | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:59 -05:00
										 |  |  |  |         ret = VKD3D_ERROR_OUT_OF_MEMORY; | 
					
						
							|  |  |  |  |     vkd3d_shader_message_context_cleanup(&message_context); | 
					
						
							|  |  |  |  |     return ret; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2023-10-15 01:26:32 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | void vkd3d_shader_free_scan_combined_resource_sampler_info( | 
					
						
							|  |  |  |  |         struct vkd3d_shader_scan_combined_resource_sampler_info *info) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     TRACE("info %p.\n", info); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     vkd3d_free(info->combined_samplers); | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-09-25 16:52:59 -05:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:55 +04:30
										 |  |  |  | void vkd3d_shader_free_scan_descriptor_info(struct vkd3d_shader_scan_descriptor_info *scan_descriptor_info) | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-04 20:31:14 +04:30
										 |  |  |  |     TRACE("scan_descriptor_info %p.\n", scan_descriptor_info); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:55 +04:30
										 |  |  |  |     vkd3d_free(scan_descriptor_info->descriptors); | 
					
						
							| 
									
										
										
										
											2020-06-25 10:48:24 +04:30
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-23 16:21:18 -06:00
										 |  |  |  | void vkd3d_shader_free_scan_signature_info(struct vkd3d_shader_scan_signature_info *info) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     TRACE("info %p.\n", info); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     vkd3d_shader_free_shader_signature(&info->input); | 
					
						
							|  |  |  |  |     vkd3d_shader_free_shader_signature(&info->output); | 
					
						
							|  |  |  |  |     vkd3d_shader_free_shader_signature(&info->patch_constant); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-16 22:38:21 +02:00
										 |  |  |  | void vkd3d_shader_free_shader_code(struct vkd3d_shader_code *shader_code) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-04 20:31:14 +04:30
										 |  |  |  |     TRACE("shader_code %p.\n", shader_code); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-16 22:38:21 +02:00
										 |  |  |  |     vkd3d_free((void *)shader_code->code); | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-06-27 13:16:47 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-21 15:20:03 +04:30
										 |  |  |  | static void vkd3d_shader_free_root_signature_v_1_0(struct vkd3d_shader_root_signature_desc *root_signature) | 
					
						
							| 
									
										
										
										
											2017-06-27 13:16:47 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     unsigned int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-15 15:43:51 +01:00
										 |  |  |  |     for (i = 0; i < root_signature->parameter_count; ++i) | 
					
						
							| 
									
										
										
										
											2017-06-27 13:16:47 +02:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-07-17 14:14:25 +04:30
										 |  |  |  |         const struct vkd3d_shader_root_parameter *parameter = &root_signature->parameters[i]; | 
					
						
							| 
									
										
										
										
											2017-06-27 13:16:47 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-14 15:52:55 +04:30
										 |  |  |  |         if (parameter->parameter_type == VKD3D_SHADER_ROOT_PARAMETER_TYPE_DESCRIPTOR_TABLE) | 
					
						
							| 
									
										
										
										
											2018-02-15 15:43:51 +01:00
										 |  |  |  |             vkd3d_free((void *)parameter->u.descriptor_table.descriptor_ranges); | 
					
						
							| 
									
										
										
										
											2017-06-27 13:16:47 +02:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-02-15 15:43:51 +01:00
										 |  |  |  |     vkd3d_free((void *)root_signature->parameters); | 
					
						
							|  |  |  |  |     vkd3d_free((void *)root_signature->static_samplers); | 
					
						
							| 
									
										
										
										
											2017-06-27 13:16:47 +02:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     memset(root_signature, 0, sizeof(*root_signature)); | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-09-11 22:35:16 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-22 15:54:11 +04:30
										 |  |  |  | static void vkd3d_shader_free_root_signature_v_1_1(struct vkd3d_shader_root_signature_desc1 *root_signature) | 
					
						
							| 
									
										
										
										
											2019-04-19 11:07:31 +02:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     unsigned int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = 0; i < root_signature->parameter_count; ++i) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-07-21 15:20:01 +04:30
										 |  |  |  |         const struct vkd3d_shader_root_parameter1 *parameter = &root_signature->parameters[i]; | 
					
						
							| 
									
										
										
										
											2019-04-19 11:07:31 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-14 15:52:55 +04:30
										 |  |  |  |         if (parameter->parameter_type == VKD3D_SHADER_ROOT_PARAMETER_TYPE_DESCRIPTOR_TABLE) | 
					
						
							| 
									
										
										
										
											2019-04-19 11:07:31 +02:00
										 |  |  |  |             vkd3d_free((void *)parameter->u.descriptor_table.descriptor_ranges); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     vkd3d_free((void *)root_signature->parameters); | 
					
						
							|  |  |  |  |     vkd3d_free((void *)root_signature->static_samplers); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     memset(root_signature, 0, sizeof(*root_signature)); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-22 15:54:12 +04:30
										 |  |  |  | void vkd3d_shader_free_root_signature(struct vkd3d_shader_versioned_root_signature_desc *desc) | 
					
						
							| 
									
										
										
										
											2019-04-24 16:05:48 +02:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-04 20:31:14 +04:30
										 |  |  |  |     TRACE("desc %p.\n", desc); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-21 15:20:02 +04:30
										 |  |  |  |     if (desc->version == VKD3D_SHADER_ROOT_SIGNATURE_VERSION_1_0) | 
					
						
							| 
									
										
										
										
											2019-04-24 16:05:48 +02:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |         vkd3d_shader_free_root_signature_v_1_0(&desc->u.v_1_0); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-07-21 15:20:02 +04:30
										 |  |  |  |     else if (desc->version == VKD3D_SHADER_ROOT_SIGNATURE_VERSION_1_1) | 
					
						
							| 
									
										
										
										
											2019-04-24 16:05:48 +02:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |         vkd3d_shader_free_root_signature_v_1_1(&desc->u.v_1_1); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     else if (desc->version) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         FIXME("Unknown version %#x.\n", desc->version); | 
					
						
							|  |  |  |  |         return; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     desc->version = 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-04 16:22:57 +10:00
										 |  |  |  | void shader_signature_cleanup(struct shader_signature *signature) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-05-21 20:11:26 +02:00
										 |  |  |  |     for (unsigned int i = 0; i < signature->element_count; ++i) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         vkd3d_free((void *)signature->elements[i].semantic_name); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-04-04 16:22:57 +10:00
										 |  |  |  |     vkd3d_free(signature->elements); | 
					
						
							|  |  |  |  |     signature->elements = NULL; | 
					
						
							| 
									
										
										
										
											2024-10-09 11:36:06 +02:00
										 |  |  |  |     signature->elements_capacity = 0; | 
					
						
							|  |  |  |  |     signature->element_count = 0; | 
					
						
							| 
									
										
										
										
											2023-04-04 16:22:57 +10:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-15 15:43:52 +01:00
										 |  |  |  | int vkd3d_shader_parse_input_signature(const struct vkd3d_shader_code *dxbc, | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:56 +04:30
										 |  |  |  |         struct vkd3d_shader_signature *signature, char **messages) | 
					
						
							| 
									
										
										
										
											2017-09-11 22:35:16 +02:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:56 +04:30
										 |  |  |  |     struct vkd3d_shader_message_context message_context; | 
					
						
							| 
									
										
										
										
											2023-04-04 16:22:57 +10:00
										 |  |  |  |     struct shader_signature shader_signature; | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:56 +04:30
										 |  |  |  |     int ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     TRACE("dxbc {%p, %zu}, signature %p, messages %p.\n", dxbc->code, dxbc->size, signature, messages); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (messages) | 
					
						
							|  |  |  |  |         *messages = NULL; | 
					
						
							| 
									
										
										
										
											2020-12-15 11:01:32 -06:00
										 |  |  |  |     vkd3d_shader_message_context_init(&message_context, VKD3D_SHADER_LOG_INFO); | 
					
						
							| 
									
										
										
										
											2017-09-11 22:35:16 +02:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-04 16:22:57 +10:00
										 |  |  |  |     ret = shader_parse_input_signature(dxbc, &message_context, &shader_signature); | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:56 +04:30
										 |  |  |  |     vkd3d_shader_message_context_trace_messages(&message_context); | 
					
						
							| 
									
										
										
										
											2020-10-09 01:14:01 -05:00
										 |  |  |  |     if (!vkd3d_shader_message_context_copy_messages(&message_context, messages)) | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:56 +04:30
										 |  |  |  |         ret = VKD3D_ERROR_OUT_OF_MEMORY; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     vkd3d_shader_message_context_cleanup(&message_context); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-04 16:22:57 +10:00
										 |  |  |  |     if (!vkd3d_shader_signature_from_shader_signature(signature, &shader_signature)) | 
					
						
							|  |  |  |  |         ret = VKD3D_ERROR_OUT_OF_MEMORY; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     shader_signature_cleanup(&shader_signature); | 
					
						
							| 
									
										
										
										
											2020-07-30 14:59:56 +04:30
										 |  |  |  |     return ret; | 
					
						
							| 
									
										
										
										
											2017-09-11 22:35:16 +02:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | struct vkd3d_shader_signature_element *vkd3d_shader_find_signature_element( | 
					
						
							|  |  |  |  |         const struct vkd3d_shader_signature *signature, const char *semantic_name, | 
					
						
							|  |  |  |  |         unsigned int semantic_index, unsigned int stream_index) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-04-13 12:37:38 +02:00
										 |  |  |  |     struct vkd3d_shader_signature_element *e; | 
					
						
							| 
									
										
										
										
											2017-09-11 22:35:16 +02:00
										 |  |  |  |     unsigned int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     TRACE("signature %p, semantic_name %s, semantic_index %u, stream_index %u.\n", | 
					
						
							|  |  |  |  |             signature, debugstr_a(semantic_name), semantic_index, stream_index); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     e = signature->elements; | 
					
						
							|  |  |  |  |     for (i = 0; i < signature->element_count; ++i) | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-04-30 14:33:48 +02:00
										 |  |  |  |         if (!ascii_strcasecmp(e[i].semantic_name, semantic_name) | 
					
						
							| 
									
										
										
										
											2017-09-11 22:35:16 +02:00
										 |  |  |  |                 && e[i].semantic_index == semantic_index | 
					
						
							|  |  |  |  |                 && e[i].stream_index == stream_index) | 
					
						
							|  |  |  |  |             return &e[i]; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void vkd3d_shader_free_shader_signature(struct vkd3d_shader_signature *signature) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     TRACE("signature %p.\n", signature); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-21 20:11:26 +02:00
										 |  |  |  |     for (unsigned int i = 0; i < signature->element_count; ++i) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         vkd3d_free((void *)signature->elements[i].semantic_name); | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-09-11 22:35:16 +02:00
										 |  |  |  |     vkd3d_free(signature->elements); | 
					
						
							|  |  |  |  |     signature->elements = NULL; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-08-06 13:23:45 +04:30
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | const char *vkd3d_shader_get_version(unsigned int *major, unsigned int *minor) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     int x, y; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     TRACE("major %p, minor %p.\n", major, minor); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (major || minor) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         vkd3d_parse_version(PACKAGE_VERSION, &x, &y); | 
					
						
							|  |  |  |  |         if (major) | 
					
						
							|  |  |  |  |             *major = x; | 
					
						
							|  |  |  |  |         if (minor) | 
					
						
							|  |  |  |  |             *minor = y; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return "vkd3d-shader " PACKAGE_VERSION VKD3D_VCS_ID; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-08-06 13:23:48 +04:30
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | const enum vkd3d_shader_source_type *vkd3d_shader_get_supported_source_types(unsigned int *count) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     static const enum vkd3d_shader_source_type types[] = | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         VKD3D_SHADER_SOURCE_DXBC_TPF, | 
					
						
							| 
									
										
										
										
											2020-09-25 16:53:00 -05:00
										 |  |  |  |         VKD3D_SHADER_SOURCE_HLSL, | 
					
						
							| 
									
										
										
										
											2021-10-06 17:11:48 +02:00
										 |  |  |  |         VKD3D_SHADER_SOURCE_D3D_BYTECODE, | 
					
						
							| 
									
										
										
										
											2022-11-15 15:16:41 +10:00
										 |  |  |  |         VKD3D_SHADER_SOURCE_DXBC_DXIL, | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |  |         VKD3D_SHADER_SOURCE_FX, | 
					
						
							| 
									
										
										
										
											2025-06-12 00:38:10 +02:00
										 |  |  |  |         VKD3D_SHADER_SOURCE_TX, | 
					
						
							| 
									
										
										
										
											2020-08-06 13:23:48 +04:30
										 |  |  |  |     }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     TRACE("count %p.\n", count); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     *count = ARRAY_SIZE(types); | 
					
						
							|  |  |  |  |     return types; | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-08-06 13:23:49 +04:30
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | const enum vkd3d_shader_target_type *vkd3d_shader_get_supported_target_types( | 
					
						
							|  |  |  |  |         enum vkd3d_shader_source_type source_type, unsigned int *count) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     static const enum vkd3d_shader_target_type dxbc_tpf_types[] = | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         VKD3D_SHADER_TARGET_SPIRV_BINARY, | 
					
						
							| 
									
										
										
										
											2024-10-31 13:40:09 -05:00
										 |  |  |  | #if defined(HAVE_SPIRV_TOOLS) || defined(VKD3D_SHADER_UNSUPPORTED_SPIRV_PARSER)
 | 
					
						
							| 
									
										
										
										
											2020-09-07 12:35:41 +01:00
										 |  |  |  |         VKD3D_SHADER_TARGET_SPIRV_TEXT, | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-09-25 02:26:38 +01:00
										 |  |  |  |         VKD3D_SHADER_TARGET_D3D_ASM, | 
					
						
							| 
									
										
										
										
											2024-03-07 20:55:39 +01:00
										 |  |  |  | #ifdef VKD3D_SHADER_UNSUPPORTED_GLSL
 | 
					
						
							| 
									
										
										
										
											2021-08-18 09:49:14 +05:30
										 |  |  |  |         VKD3D_SHADER_TARGET_GLSL, | 
					
						
							| 
									
										
										
										
											2024-08-27 10:34:57 +08:00
										 |  |  |  | #endif
 | 
					
						
							|  |  |  |  | #ifdef VKD3D_SHADER_UNSUPPORTED_MSL
 | 
					
						
							|  |  |  |  |         VKD3D_SHADER_TARGET_MSL, | 
					
						
							| 
									
										
										
										
											2022-03-02 19:06:23 +01:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-08-06 13:23:49 +04:30
										 |  |  |  |     }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-25 16:53:00 -05:00
										 |  |  |  |     static const enum vkd3d_shader_target_type hlsl_types[] = | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-04-18 16:36:21 +02:00
										 |  |  |  |         VKD3D_SHADER_TARGET_SPIRV_BINARY, | 
					
						
							| 
									
										
										
										
											2024-10-31 13:40:09 -05:00
										 |  |  |  | #if defined(HAVE_SPIRV_TOOLS) || defined(VKD3D_SHADER_UNSUPPORTED_SPIRV_PARSER)
 | 
					
						
							| 
									
										
										
										
											2024-04-18 16:36:21 +02:00
										 |  |  |  |         VKD3D_SHADER_TARGET_SPIRV_TEXT, | 
					
						
							| 
									
										
										
										
											2025-03-25 12:46:42 -05:00
										 |  |  |  | #endif
 | 
					
						
							|  |  |  |  | #ifdef VKD3D_SHADER_UNSUPPORTED_GLSL
 | 
					
						
							|  |  |  |  |         VKD3D_SHADER_TARGET_GLSL, | 
					
						
							| 
									
										
										
										
											2024-04-18 16:36:21 +02:00
										 |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2024-04-17 20:12:59 +02:00
										 |  |  |  |         VKD3D_SHADER_TARGET_D3D_ASM, | 
					
						
							| 
									
										
										
										
											2022-02-28 12:23:43 +01:00
										 |  |  |  |         VKD3D_SHADER_TARGET_D3D_BYTECODE, | 
					
						
							| 
									
										
										
										
											2020-09-25 16:53:00 -05:00
										 |  |  |  |         VKD3D_SHADER_TARGET_DXBC_TPF, | 
					
						
							| 
									
										
										
										
											2023-11-07 17:53:39 +01:00
										 |  |  |  |         VKD3D_SHADER_TARGET_FX, | 
					
						
							| 
									
										
										
										
											2025-08-12 23:01:18 +02:00
										 |  |  |  | #ifdef VKD3D_SHADER_UNSUPPORTED_MSL
 | 
					
						
							|  |  |  |  |         VKD3D_SHADER_TARGET_MSL, | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-09-25 16:53:00 -05:00
										 |  |  |  |     }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-06 17:11:48 +02:00
										 |  |  |  |     static const enum vkd3d_shader_target_type d3dbc_types[] = | 
					
						
							|  |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-03-12 16:57:27 -06:00
										 |  |  |  |         VKD3D_SHADER_TARGET_SPIRV_BINARY, | 
					
						
							| 
									
										
										
										
											2024-10-31 13:40:09 -05:00
										 |  |  |  | #if defined(HAVE_SPIRV_TOOLS) || defined(VKD3D_SHADER_UNSUPPORTED_SPIRV_PARSER)
 | 
					
						
							| 
									
										
										
										
											2022-03-12 16:57:27 -06:00
										 |  |  |  |         VKD3D_SHADER_TARGET_SPIRV_TEXT, | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2021-10-06 17:11:48 +02:00
										 |  |  |  |         VKD3D_SHADER_TARGET_D3D_ASM, | 
					
						
							|  |  |  |  |     }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-07 20:55:39 +01:00
										 |  |  |  |     static const enum vkd3d_shader_target_type dxbc_dxil_types[] = | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         VKD3D_SHADER_TARGET_SPIRV_BINARY, | 
					
						
							| 
									
										
										
										
											2024-10-31 13:40:09 -05:00
										 |  |  |  | #if defined(HAVE_SPIRV_TOOLS) || defined(VKD3D_SHADER_UNSUPPORTED_SPIRV_PARSER)
 | 
					
						
							| 
									
										
										
										
											2024-03-07 20:55:39 +01:00
										 |  |  |  |         VKD3D_SHADER_TARGET_SPIRV_TEXT, | 
					
						
							|  |  |  |  | # endif
 | 
					
						
							|  |  |  |  |         VKD3D_SHADER_TARGET_D3D_ASM, | 
					
						
							| 
									
										
										
										
											2025-07-27 13:49:11 +02:00
										 |  |  |  | #ifdef VKD3D_SHADER_UNSUPPORTED_MSL
 | 
					
						
							|  |  |  |  |         VKD3D_SHADER_TARGET_MSL, | 
					
						
							|  |  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2024-03-07 20:55:39 +01:00
										 |  |  |  |     }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |  |     static const enum vkd3d_shader_target_type fx_types[] = | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         VKD3D_SHADER_TARGET_D3D_ASM, | 
					
						
							|  |  |  |  |     }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-12 00:38:10 +02:00
										 |  |  |  |     static const enum vkd3d_shader_target_type tx_types[] = | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         VKD3D_SHADER_TARGET_D3D_ASM, | 
					
						
							|  |  |  |  |     }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-06 13:23:49 +04:30
										 |  |  |  |     TRACE("source_type %#x, count %p.\n", source_type, count); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     switch (source_type) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         case VKD3D_SHADER_SOURCE_DXBC_TPF: | 
					
						
							|  |  |  |  |             *count = ARRAY_SIZE(dxbc_tpf_types); | 
					
						
							|  |  |  |  |             return dxbc_tpf_types; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-25 16:53:00 -05:00
										 |  |  |  |         case VKD3D_SHADER_SOURCE_HLSL: | 
					
						
							|  |  |  |  |             *count = ARRAY_SIZE(hlsl_types); | 
					
						
							|  |  |  |  |             return hlsl_types; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-06 17:11:48 +02:00
										 |  |  |  |         case VKD3D_SHADER_SOURCE_D3D_BYTECODE: | 
					
						
							|  |  |  |  |             *count = ARRAY_SIZE(d3dbc_types); | 
					
						
							|  |  |  |  |             return d3dbc_types; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-07 20:55:39 +01:00
										 |  |  |  |         case VKD3D_SHADER_SOURCE_DXBC_DXIL: | 
					
						
							|  |  |  |  |             *count = ARRAY_SIZE(dxbc_dxil_types); | 
					
						
							|  |  |  |  |             return dxbc_dxil_types; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-04 23:36:28 +02:00
										 |  |  |  |         case VKD3D_SHADER_SOURCE_FX: | 
					
						
							|  |  |  |  |             *count = ARRAY_SIZE(fx_types); | 
					
						
							|  |  |  |  |             return fx_types; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-12 00:38:10 +02:00
										 |  |  |  |         case VKD3D_SHADER_SOURCE_TX: | 
					
						
							|  |  |  |  |             *count = ARRAY_SIZE(tx_types); | 
					
						
							|  |  |  |  |             return tx_types; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-06 13:23:49 +04:30
										 |  |  |  |         default: | 
					
						
							|  |  |  |  |             *count = 0; | 
					
						
							|  |  |  |  |             return NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-09-25 16:53:01 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | int vkd3d_shader_preprocess(const struct vkd3d_shader_compile_info *compile_info, | 
					
						
							|  |  |  |  |         struct vkd3d_shader_code *out, char **messages) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  |  |     struct vkd3d_shader_message_context message_context; | 
					
						
							| 
									
										
										
										
											2025-08-15 13:11:58 -05:00
										 |  |  |  |     struct shader_dump_data dump_data; | 
					
						
							| 
									
										
										
										
											2020-12-02 09:29:11 -06:00
										 |  |  |  |     int ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-25 16:53:01 -05:00
										 |  |  |  |     TRACE("compile_info %p, out %p, messages %p.\n", compile_info, out, messages); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-21 17:23:44 -06:00
										 |  |  |  |     if (messages) | 
					
						
							|  |  |  |  |         *messages = NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-02 09:29:11 -06:00
										 |  |  |  |     if ((ret = vkd3d_shader_validate_compile_info(compile_info, false)) < 0) | 
					
						
							|  |  |  |  |         return ret; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-15 11:01:32 -06:00
										 |  |  |  |     vkd3d_shader_message_context_init(&message_context, compile_info->log_level); | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-15 13:11:58 -05:00
										 |  |  |  |     fill_shader_dump_data(compile_info, &dump_data); | 
					
						
							|  |  |  |  |     vkd3d_shader_dump_shader(&dump_data, compile_info->source.code, compile_info->source.size, SHADER_DUMP_TYPE_SOURCE); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if ((ret = preproc_lexer_parse(compile_info, out, &message_context)) >= 0) | 
					
						
							|  |  |  |  |         vkd3d_shader_dump_shader(&dump_data, out->code, out->size, SHADER_DUMP_TYPE_PREPROC); | 
					
						
							| 
									
										
										
										
											2020-12-07 12:56:30 -06:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     vkd3d_shader_message_context_trace_messages(&message_context); | 
					
						
							|  |  |  |  |     if (!vkd3d_shader_message_context_copy_messages(&message_context, messages)) | 
					
						
							|  |  |  |  |         ret = VKD3D_ERROR_OUT_OF_MEMORY; | 
					
						
							|  |  |  |  |     vkd3d_shader_message_context_cleanup(&message_context); | 
					
						
							|  |  |  |  |     return ret; | 
					
						
							| 
									
										
										
										
											2020-09-25 16:53:01 -05:00
										 |  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-06-01 19:01:00 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | void vkd3d_shader_set_log_callback(PFN_vkd3d_log callback) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     vkd3d_dbg_set_log_callback(callback); | 
					
						
							|  |  |  |  | } | 
					
						
							| 
									
										
										
										
											2023-01-20 11:13:01 +10:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct vkd3d_shader_param_node *shader_param_allocator_node_create( | 
					
						
							|  |  |  |  |         struct vkd3d_shader_param_allocator *allocator) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct vkd3d_shader_param_node *node; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!(node = vkd3d_malloc(offsetof(struct vkd3d_shader_param_node, param[allocator->count * allocator->stride])))) | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  |     node->next = NULL; | 
					
						
							|  |  |  |  |     return node; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void shader_param_allocator_init(struct vkd3d_shader_param_allocator *allocator, | 
					
						
							| 
									
										
										
										
											2025-07-22 10:01:45 +02:00
										 |  |  |  |         size_t count, size_t stride) | 
					
						
							| 
									
										
										
										
											2023-01-20 11:13:01 +10:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-08-17 23:38:42 +10:00
										 |  |  |  |     allocator->count = max(count, MAX_REG_OUTPUT); | 
					
						
							| 
									
										
										
										
											2023-01-20 11:13:01 +10:00
										 |  |  |  |     allocator->stride = stride; | 
					
						
							|  |  |  |  |     allocator->head = NULL; | 
					
						
							|  |  |  |  |     allocator->current = NULL; | 
					
						
							|  |  |  |  |     allocator->index = allocator->count; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void shader_param_allocator_destroy(struct vkd3d_shader_param_allocator *allocator) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     struct vkd3d_shader_param_node *current = allocator->head; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     while (current) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         struct vkd3d_shader_param_node *next = current->next; | 
					
						
							|  |  |  |  |         vkd3d_free(current); | 
					
						
							|  |  |  |  |         current = next; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-22 10:01:45 +02:00
										 |  |  |  | void *shader_param_allocator_get(struct vkd3d_shader_param_allocator *allocator, size_t count) | 
					
						
							| 
									
										
										
										
											2023-01-20 11:13:01 +10:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     void *params; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-06 16:51:36 -04:00
										 |  |  |  |     if (!allocator->current || count > allocator->count - allocator->index) | 
					
						
							| 
									
										
										
										
											2023-01-20 11:13:01 +10:00
										 |  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-02-09 12:02:13 +10:00
										 |  |  |  |         struct vkd3d_shader_param_node *next; | 
					
						
							| 
									
										
										
										
											2023-01-20 11:13:01 +10:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-09 12:02:13 +10:00
										 |  |  |  |         /* Monolithic switch has no definite parameter count limit. */ | 
					
						
							|  |  |  |  |         allocator->count = max(allocator->count, count); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (!(next = shader_param_allocator_node_create(allocator))) | 
					
						
							| 
									
										
										
										
											2023-01-20 11:13:01 +10:00
										 |  |  |  |             return NULL; | 
					
						
							|  |  |  |  |         if (allocator->current) | 
					
						
							|  |  |  |  |             allocator->current->next = next; | 
					
						
							| 
									
										
										
										
											2023-02-03 19:35:16 -06:00
										 |  |  |  |         else | 
					
						
							|  |  |  |  |             allocator->head = next; | 
					
						
							| 
									
										
										
										
											2023-01-20 11:13:01 +10:00
										 |  |  |  |         allocator->current = next; | 
					
						
							|  |  |  |  |         allocator->index = 0; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     params = &allocator->current->param[allocator->index * allocator->stride]; | 
					
						
							|  |  |  |  |     allocator->index += count; | 
					
						
							|  |  |  |  |     return params; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-22 10:08:44 +02:00
										 |  |  |  | bool shader_instruction_array_init(struct vkd3d_shader_instruction_array *instructions, size_t reserve) | 
					
						
							| 
									
										
										
										
											2023-01-20 11:13:01 +10:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     memset(instructions, 0, sizeof(*instructions)); | 
					
						
							|  |  |  |  |     /* Size the parameter initial allocations so they are large enough for most shaders. The
 | 
					
						
							|  |  |  |  |      * code path for chained allocations will be tested if a few shaders need to use it. */ | 
					
						
							|  |  |  |  |     shader_param_allocator_init(&instructions->dst_params, reserve - reserve / 8u, | 
					
						
							| 
									
										
										
										
											2025-07-28 22:27:41 +02:00
										 |  |  |  |             sizeof(struct vkd3d_shader_dst_param)); | 
					
						
							|  |  |  |  |     shader_param_allocator_init(&instructions->src_params, reserve * 2u, sizeof(struct vkd3d_shader_src_param)); | 
					
						
							| 
									
										
										
										
											2023-01-20 11:13:01 +10:00
										 |  |  |  |     return shader_instruction_array_reserve(instructions, reserve); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-22 10:08:44 +02:00
										 |  |  |  | bool shader_instruction_array_reserve(struct vkd3d_shader_instruction_array *instructions, size_t reserve) | 
					
						
							| 
									
										
										
										
											2023-01-20 11:13:01 +10:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     if (!vkd3d_array_reserve((void **)&instructions->elements, &instructions->capacity, reserve, | 
					
						
							|  |  |  |  |             sizeof(*instructions->elements))) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         ERR("Failed to allocate instructions.\n"); | 
					
						
							|  |  |  |  |         return false; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |     return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-10 17:23:04 -03:00
										 |  |  |  | bool shader_instruction_array_insert_at(struct vkd3d_shader_instruction_array *instructions, | 
					
						
							| 
									
										
										
										
											2025-07-22 10:08:44 +02:00
										 |  |  |  |         size_t idx, size_t count) | 
					
						
							| 
									
										
										
										
											2024-01-10 17:23:04 -03:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2024-08-05 16:50:14 +02:00
										 |  |  |  |     VKD3D_ASSERT(idx <= instructions->count); | 
					
						
							| 
									
										
										
										
											2024-01-10 17:23:04 -03:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!shader_instruction_array_reserve(instructions, instructions->count + count)) | 
					
						
							|  |  |  |  |         return false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     memmove(&instructions->elements[idx + count], &instructions->elements[idx], | 
					
						
							|  |  |  |  |             (instructions->count - idx) * sizeof(*instructions->elements)); | 
					
						
							|  |  |  |  |     memset(&instructions->elements[idx], 0, count * sizeof(*instructions->elements)); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     instructions->count += count; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-15 11:54:03 +10:00
										 |  |  |  | bool shader_instruction_array_add_icb(struct vkd3d_shader_instruction_array *instructions, | 
					
						
							| 
									
										
										
										
											2023-01-20 11:35:38 +10:00
										 |  |  |  |         struct vkd3d_shader_immediate_constant_buffer *icb) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (!vkd3d_array_reserve((void **)&instructions->icbs, &instructions->icb_capacity, instructions->icb_count + 1, | 
					
						
							|  |  |  |  |             sizeof(*instructions->icbs))) | 
					
						
							| 
									
										
										
										
											2023-11-15 11:54:03 +10:00
										 |  |  |  |         return false; | 
					
						
							|  |  |  |  |     instructions->icbs[instructions->icb_count++] = icb; | 
					
						
							|  |  |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2023-01-20 11:35:38 +10:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-01 11:59:28 +10:00
										 |  |  |  | static struct vkd3d_shader_src_param *shader_instruction_array_clone_src_params( | 
					
						
							|  |  |  |  |         struct vkd3d_shader_instruction_array *instructions, const struct vkd3d_shader_src_param *params, | 
					
						
							| 
									
										
										
										
											2025-07-22 10:01:45 +02:00
										 |  |  |  |         size_t count); | 
					
						
							| 
									
										
										
										
											2023-03-01 11:59:28 +10:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | static bool shader_register_clone_relative_addresses(struct vkd3d_shader_register *reg, | 
					
						
							|  |  |  |  |         struct vkd3d_shader_instruction_array *instructions) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     unsigned int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-26 17:06:16 +10:00
										 |  |  |  |     for (i = 0; i < reg->idx_count; ++i) | 
					
						
							| 
									
										
										
										
											2023-03-01 11:59:28 +10:00
										 |  |  |  |     { | 
					
						
							|  |  |  |  |         if (!reg->idx[i].rel_addr) | 
					
						
							|  |  |  |  |             continue; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (!(reg->idx[i].rel_addr = shader_instruction_array_clone_src_params(instructions, reg->idx[i].rel_addr, 1))) | 
					
						
							|  |  |  |  |             return false; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return true; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct vkd3d_shader_dst_param *shader_instruction_array_clone_dst_params( | 
					
						
							|  |  |  |  |         struct vkd3d_shader_instruction_array *instructions, const struct vkd3d_shader_dst_param *params, | 
					
						
							| 
									
										
										
										
											2025-07-22 10:01:45 +02:00
										 |  |  |  |         size_t count) | 
					
						
							| 
									
										
										
										
											2023-03-01 11:59:28 +10:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct vkd3d_shader_dst_param *dst_params; | 
					
						
							| 
									
										
										
										
											2025-07-22 10:01:45 +02:00
										 |  |  |  |     size_t i; | 
					
						
							| 
									
										
										
										
											2023-03-01 11:59:28 +10:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!(dst_params = shader_dst_param_allocator_get(&instructions->dst_params, count))) | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     memcpy(dst_params, params, count * sizeof(*params)); | 
					
						
							|  |  |  |  |     for (i = 0; i < count; ++i) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         if (!shader_register_clone_relative_addresses(&dst_params[i].reg, instructions)) | 
					
						
							|  |  |  |  |             return NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return dst_params; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static struct vkd3d_shader_src_param *shader_instruction_array_clone_src_params( | 
					
						
							|  |  |  |  |         struct vkd3d_shader_instruction_array *instructions, const struct vkd3d_shader_src_param *params, | 
					
						
							| 
									
										
										
										
											2025-07-22 10:01:45 +02:00
										 |  |  |  |         size_t count) | 
					
						
							| 
									
										
										
										
											2023-03-01 11:59:28 +10:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct vkd3d_shader_src_param *src_params; | 
					
						
							| 
									
										
										
										
											2025-07-22 10:01:45 +02:00
										 |  |  |  |     size_t i; | 
					
						
							| 
									
										
										
										
											2023-03-01 11:59:28 +10:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (!(src_params = shader_src_param_allocator_get(&instructions->src_params, count))) | 
					
						
							|  |  |  |  |         return NULL; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     memcpy(src_params, params, count * sizeof(*params)); | 
					
						
							|  |  |  |  |     for (i = 0; i < count; ++i) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         if (!shader_register_clone_relative_addresses(&src_params[i].reg, instructions)) | 
					
						
							|  |  |  |  |             return NULL; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return src_params; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* NOTE: Immediate constant buffers are not cloned, so the source must not be destroyed while the
 | 
					
						
							|  |  |  |  |  * destination is in use. This seems like a reasonable requirement given how this is currently used. */ | 
					
						
							|  |  |  |  | bool shader_instruction_array_clone_instruction(struct vkd3d_shader_instruction_array *instructions, | 
					
						
							| 
									
										
										
										
											2025-07-22 10:08:44 +02:00
										 |  |  |  |         size_t dst, size_t src) | 
					
						
							| 
									
										
										
										
											2023-03-01 11:59:28 +10:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     struct vkd3d_shader_instruction *ins = &instructions->elements[dst]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     *ins = instructions->elements[src]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     if (ins->dst_count && ins->dst && !(ins->dst = shader_instruction_array_clone_dst_params(instructions, | 
					
						
							|  |  |  |  |             ins->dst, ins->dst_count))) | 
					
						
							|  |  |  |  |         return false; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return !ins->src_count || !!(ins->src = shader_instruction_array_clone_src_params(instructions, | 
					
						
							|  |  |  |  |             ins->src, ins->src_count)); | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-20 11:13:01 +10:00
										 |  |  |  | void shader_instruction_array_destroy(struct vkd3d_shader_instruction_array *instructions) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-01-20 11:35:38 +10:00
										 |  |  |  |     unsigned int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-20 11:13:01 +10:00
										 |  |  |  |     vkd3d_free(instructions->elements); | 
					
						
							|  |  |  |  |     shader_param_allocator_destroy(&instructions->dst_params); | 
					
						
							|  |  |  |  |     shader_param_allocator_destroy(&instructions->src_params); | 
					
						
							| 
									
										
										
										
											2023-01-20 11:35:38 +10:00
										 |  |  |  |     for (i = 0; i < instructions->icb_count; ++i) | 
					
						
							|  |  |  |  |         vkd3d_free(instructions->icbs[i]); | 
					
						
							|  |  |  |  |     vkd3d_free(instructions->icbs); | 
					
						
							| 
									
										
										
										
											2023-01-20 11:13:01 +10:00
										 |  |  |  | } | 
					
						
							| 
									
										
										
										
											2023-07-31 00:45:54 -05:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | void vkd3d_shader_build_varying_map(const struct vkd3d_shader_signature *output_signature, | 
					
						
							|  |  |  |  |         const struct vkd3d_shader_signature *input_signature, | 
					
						
							|  |  |  |  |         unsigned int *ret_count, struct vkd3d_shader_varying_map *varyings) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     unsigned int count = 0; | 
					
						
							|  |  |  |  |     unsigned int i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     TRACE("output_signature %p, input_signature %p, ret_count %p, varyings %p.\n", | 
					
						
							|  |  |  |  |             output_signature, input_signature, ret_count, varyings); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     for (i = 0; i < input_signature->element_count; ++i) | 
					
						
							|  |  |  |  |     { | 
					
						
							|  |  |  |  |         const struct vkd3d_shader_signature_element *input_element, *output_element; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         input_element = &input_signature->elements[i]; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (input_element->sysval_semantic != VKD3D_SHADER_SV_NONE) | 
					
						
							|  |  |  |  |             continue; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         varyings[count].input_register_index = input_element->register_index; | 
					
						
							|  |  |  |  |         varyings[count].input_mask = input_element->mask; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         if ((output_element = vkd3d_shader_find_signature_element(output_signature, | 
					
						
							|  |  |  |  |                 input_element->semantic_name, input_element->semantic_index, 0))) | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             varyings[count].output_signature_index = output_element - output_signature->elements; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         else | 
					
						
							|  |  |  |  |         { | 
					
						
							|  |  |  |  |             varyings[count].output_signature_index = output_signature->element_count; | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         ++count; | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     *ret_count = count; | 
					
						
							|  |  |  |  | } |