| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2021 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <array>
 | 
					
						
							| 
									
										
										
										
											2022-03-22 21:11:24 +00:00
										 |  |  | #include <bit>
 | 
					
						
							| 
									
										
										
										
											2021-04-05 22:25:22 -04:00
										 |  |  | #include <climits>
 | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-25 22:26:23 -03:00
										 |  |  | #include <boost/container/static_vector.hpp>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  | #include <fmt/format.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  | #include "common/div_ceil.h"
 | 
					
						
							| 
									
										
										
										
											2021-08-01 18:57:45 -03:00
										 |  |  | #include "shader_recompiler/backend/spirv/emit_spirv.h"
 | 
					
						
							| 
									
										
										
										
											2021-12-05 16:42:03 -05:00
										 |  |  | #include "shader_recompiler/backend/spirv/spirv_emit_context.h"
 | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Shader::Backend::SPIRV { | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  | namespace { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  | enum class Operation { | 
					
						
							| 
									
										
										
										
											2021-04-11 02:07:02 -04:00
										 |  |  |     Increment, | 
					
						
							|  |  |  |     Decrement, | 
					
						
							|  |  |  |     FPAdd, | 
					
						
							|  |  |  |     FPMin, | 
					
						
							|  |  |  |     FPMax, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  | Id ImageType(EmitContext& ctx, const TextureDescriptor& desc) { | 
					
						
							|  |  |  |     const spv::ImageFormat format{spv::ImageFormat::Unknown}; | 
					
						
							|  |  |  |     const Id type{ctx.F32[1]}; | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  |     const bool depth{desc.is_depth}; | 
					
						
							| 
									
										
										
										
											2023-01-20 19:30:45 -05:00
										 |  |  |     const bool ms{desc.is_multisample}; | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     switch (desc.type) { | 
					
						
							|  |  |  |     case TextureType::Color1D: | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  |         return ctx.TypeImage(type, spv::Dim::Dim1D, depth, false, false, 1, format); | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     case TextureType::ColorArray1D: | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  |         return ctx.TypeImage(type, spv::Dim::Dim1D, depth, true, false, 1, format); | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     case TextureType::Color2D: | 
					
						
							| 
									
										
										
										
											2022-08-10 10:10:32 +08:00
										 |  |  |     case TextureType::Color2DRect: | 
					
						
							| 
									
										
										
										
											2023-01-20 19:30:45 -05:00
										 |  |  |         return ctx.TypeImage(type, spv::Dim::Dim2D, depth, false, ms, 1, format); | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     case TextureType::ColorArray2D: | 
					
						
							| 
									
										
										
										
											2023-01-20 19:30:45 -05:00
										 |  |  |         return ctx.TypeImage(type, spv::Dim::Dim2D, depth, true, ms, 1, format); | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     case TextureType::Color3D: | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  |         return ctx.TypeImage(type, spv::Dim::Dim3D, depth, false, false, 1, format); | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     case TextureType::ColorCube: | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  |         return ctx.TypeImage(type, spv::Dim::Cube, depth, false, false, 1, format); | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     case TextureType::ColorArrayCube: | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  |         return ctx.TypeImage(type, spv::Dim::Cube, depth, true, false, 1, format); | 
					
						
							| 
									
										
										
										
											2021-04-06 02:56:15 -03:00
										 |  |  |     case TextureType::Buffer: | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     } | 
					
						
							|  |  |  |     throw InvalidArgument("Invalid texture type {}", desc.type); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 21:36:36 -03:00
										 |  |  | spv::ImageFormat GetImageFormat(ImageFormat format) { | 
					
						
							|  |  |  |     switch (format) { | 
					
						
							|  |  |  |     case ImageFormat::Typeless: | 
					
						
							|  |  |  |         return spv::ImageFormat::Unknown; | 
					
						
							|  |  |  |     case ImageFormat::R8_UINT: | 
					
						
							|  |  |  |         return spv::ImageFormat::R8ui; | 
					
						
							|  |  |  |     case ImageFormat::R8_SINT: | 
					
						
							|  |  |  |         return spv::ImageFormat::R8i; | 
					
						
							|  |  |  |     case ImageFormat::R16_UINT: | 
					
						
							|  |  |  |         return spv::ImageFormat::R16ui; | 
					
						
							|  |  |  |     case ImageFormat::R16_SINT: | 
					
						
							|  |  |  |         return spv::ImageFormat::R16i; | 
					
						
							|  |  |  |     case ImageFormat::R32_UINT: | 
					
						
							|  |  |  |         return spv::ImageFormat::R32ui; | 
					
						
							|  |  |  |     case ImageFormat::R32G32_UINT: | 
					
						
							|  |  |  |         return spv::ImageFormat::Rg32ui; | 
					
						
							|  |  |  |     case ImageFormat::R32G32B32A32_UINT: | 
					
						
							|  |  |  |         return spv::ImageFormat::Rgba32ui; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     throw InvalidArgument("Invalid image format {}", format); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  | Id ImageType(EmitContext& ctx, const ImageDescriptor& desc) { | 
					
						
							| 
									
										
										
										
											2021-04-14 21:36:36 -03:00
										 |  |  |     const spv::ImageFormat format{GetImageFormat(desc.format)}; | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  |     const Id type{ctx.U32[1]}; | 
					
						
							|  |  |  |     switch (desc.type) { | 
					
						
							|  |  |  |     case TextureType::Color1D: | 
					
						
							|  |  |  |         return ctx.TypeImage(type, spv::Dim::Dim1D, false, false, false, 2, format); | 
					
						
							|  |  |  |     case TextureType::ColorArray1D: | 
					
						
							|  |  |  |         return ctx.TypeImage(type, spv::Dim::Dim1D, false, true, false, 2, format); | 
					
						
							|  |  |  |     case TextureType::Color2D: | 
					
						
							|  |  |  |         return ctx.TypeImage(type, spv::Dim::Dim2D, false, false, false, 2, format); | 
					
						
							|  |  |  |     case TextureType::ColorArray2D: | 
					
						
							|  |  |  |         return ctx.TypeImage(type, spv::Dim::Dim2D, false, true, false, 2, format); | 
					
						
							|  |  |  |     case TextureType::Color3D: | 
					
						
							|  |  |  |         return ctx.TypeImage(type, spv::Dim::Dim3D, false, false, false, 2, format); | 
					
						
							|  |  |  |     case TextureType::Buffer: | 
					
						
							|  |  |  |         throw NotImplementedException("Image buffer"); | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     throw InvalidArgument("Invalid texture type {}", desc.type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  | Id DefineVariable(EmitContext& ctx, Id type, std::optional<spv::BuiltIn> builtin, | 
					
						
							|  |  |  |                   spv::StorageClass storage_class) { | 
					
						
							|  |  |  |     const Id pointer_type{ctx.TypePointer(storage_class, type)}; | 
					
						
							|  |  |  |     const Id id{ctx.AddGlobalVariable(pointer_type, storage_class)}; | 
					
						
							|  |  |  |     if (builtin) { | 
					
						
							|  |  |  |         ctx.Decorate(id, spv::Decoration::BuiltIn, *builtin); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ctx.interfaces.push_back(id); | 
					
						
							|  |  |  |     return id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-12 19:41:22 -03:00
										 |  |  | u32 NumVertices(InputTopology input_topology) { | 
					
						
							|  |  |  |     switch (input_topology) { | 
					
						
							|  |  |  |     case InputTopology::Points: | 
					
						
							|  |  |  |         return 1; | 
					
						
							|  |  |  |     case InputTopology::Lines: | 
					
						
							|  |  |  |         return 2; | 
					
						
							|  |  |  |     case InputTopology::LinesAdjacency: | 
					
						
							|  |  |  |         return 4; | 
					
						
							|  |  |  |     case InputTopology::Triangles: | 
					
						
							|  |  |  |         return 3; | 
					
						
							|  |  |  |     case InputTopology::TrianglesAdjacency: | 
					
						
							|  |  |  |         return 6; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     throw InvalidArgument("Invalid input topology {}", input_topology); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  | Id DefineInput(EmitContext& ctx, Id type, bool per_invocation, | 
					
						
							|  |  |  |                std::optional<spv::BuiltIn> builtin = std::nullopt) { | 
					
						
							|  |  |  |     switch (ctx.stage) { | 
					
						
							|  |  |  |     case Stage::TessellationControl: | 
					
						
							|  |  |  |     case Stage::TessellationEval: | 
					
						
							|  |  |  |         if (per_invocation) { | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |             type = ctx.TypeArray(type, ctx.Const(32u)); | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Stage::Geometry: | 
					
						
							|  |  |  |         if (per_invocation) { | 
					
						
							| 
									
										
										
										
											2021-05-21 02:12:32 -03:00
										 |  |  |             const u32 num_vertices{NumVertices(ctx.runtime_info.input_topology)}; | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |             type = ctx.TypeArray(type, ctx.Const(num_vertices)); | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2021-04-12 19:41:22 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |     return DefineVariable(ctx, type, builtin, spv::StorageClass::Input); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  | Id DefineOutput(EmitContext& ctx, Id type, std::optional<u32> invocations, | 
					
						
							|  |  |  |                 std::optional<spv::BuiltIn> builtin = std::nullopt) { | 
					
						
							|  |  |  |     if (invocations && ctx.stage == Stage::TessellationControl) { | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |         type = ctx.TypeArray(type, ctx.Const(*invocations)); | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |     return DefineVariable(ctx, type, builtin, spv::StorageClass::Output); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-03-24 01:33:45 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  | void DefineGenericOutput(EmitContext& ctx, size_t index, std::optional<u32> invocations) { | 
					
						
							| 
									
										
										
										
											2021-04-14 01:04:59 -03:00
										 |  |  |     static constexpr std::string_view swizzle{"xyzw"}; | 
					
						
							|  |  |  |     const size_t base_attr_index{static_cast<size_t>(IR::Attribute::Generic0X) + index * 4}; | 
					
						
							|  |  |  |     u32 element{0}; | 
					
						
							|  |  |  |     while (element < 4) { | 
					
						
							|  |  |  |         const u32 remainder{4 - element}; | 
					
						
							|  |  |  |         const TransformFeedbackVarying* xfb_varying{}; | 
					
						
							| 
									
										
										
										
											2022-01-17 12:45:34 +03:00
										 |  |  |         const size_t xfb_varying_index{base_attr_index + element}; | 
					
						
							| 
									
										
										
										
											2023-05-23 14:45:54 +01:00
										 |  |  |         if (xfb_varying_index < ctx.runtime_info.xfb_count) { | 
					
						
							| 
									
										
										
										
											2022-01-17 12:45:34 +03:00
										 |  |  |             xfb_varying = &ctx.runtime_info.xfb_varyings[xfb_varying_index]; | 
					
						
							|  |  |  |             xfb_varying = xfb_varying->components > 0 ? xfb_varying : nullptr; | 
					
						
							| 
									
										
										
										
											2021-04-14 01:04:59 -03:00
										 |  |  |         } | 
					
						
							|  |  |  |         const u32 num_components{xfb_varying ? xfb_varying->components : remainder}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         const Id id{DefineOutput(ctx, ctx.F32[num_components], invocations)}; | 
					
						
							| 
									
										
										
										
											2021-04-14 01:04:59 -03:00
										 |  |  |         ctx.Decorate(id, spv::Decoration::Location, static_cast<u32>(index)); | 
					
						
							|  |  |  |         if (element > 0) { | 
					
						
							|  |  |  |             ctx.Decorate(id, spv::Decoration::Component, element); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (xfb_varying) { | 
					
						
							|  |  |  |             ctx.Decorate(id, spv::Decoration::XfbBuffer, xfb_varying->buffer); | 
					
						
							|  |  |  |             ctx.Decorate(id, spv::Decoration::XfbStride, xfb_varying->stride); | 
					
						
							|  |  |  |             ctx.Decorate(id, spv::Decoration::Offset, xfb_varying->offset); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (num_components < 4 || element > 0) { | 
					
						
							|  |  |  |             const std::string_view subswizzle{swizzle.substr(element, num_components)}; | 
					
						
							|  |  |  |             ctx.Name(id, fmt::format("out_attr{}_{}", index, subswizzle)); | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             ctx.Name(id, fmt::format("out_attr{}", index)); | 
					
						
							| 
									
										
										
										
											2021-04-14 01:04:59 -03:00
										 |  |  |         } | 
					
						
							|  |  |  |         const GenericElementInfo info{ | 
					
						
							|  |  |  |             .id = id, | 
					
						
							|  |  |  |             .first_element = element, | 
					
						
							|  |  |  |             .num_components = num_components, | 
					
						
							|  |  |  |         }; | 
					
						
							| 
									
										
										
										
											2021-05-30 20:45:05 -03:00
										 |  |  |         std::fill_n(ctx.output_generics[index].begin() + element, num_components, info); | 
					
						
							| 
									
										
										
										
											2021-04-14 01:04:59 -03:00
										 |  |  |         element += num_components; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-24 01:33:45 -03:00
										 |  |  | Id GetAttributeType(EmitContext& ctx, AttributeType type) { | 
					
						
							|  |  |  |     switch (type) { | 
					
						
							|  |  |  |     case AttributeType::Float: | 
					
						
							|  |  |  |         return ctx.F32[4]; | 
					
						
							|  |  |  |     case AttributeType::SignedInt: | 
					
						
							|  |  |  |         return ctx.TypeVector(ctx.TypeInt(32, true), 4); | 
					
						
							|  |  |  |     case AttributeType::UnsignedInt: | 
					
						
							|  |  |  |         return ctx.U32[4]; | 
					
						
							| 
									
										
										
										
											2023-02-18 18:23:36 +00:00
										 |  |  |     case AttributeType::SignedScaled: | 
					
						
							|  |  |  |         return ctx.profile.support_scaled_attributes ? ctx.F32[4] | 
					
						
							|  |  |  |                                                      : ctx.TypeVector(ctx.TypeInt(32, true), 4); | 
					
						
							|  |  |  |     case AttributeType::UnsignedScaled: | 
					
						
							|  |  |  |         return ctx.profile.support_scaled_attributes ? ctx.F32[4] : ctx.U32[4]; | 
					
						
							| 
									
										
										
										
											2021-03-27 04:59:58 -03:00
										 |  |  |     case AttributeType::Disabled: | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2021-03-24 01:33:45 -03:00
										 |  |  |     } | 
					
						
							|  |  |  |     throw InvalidArgument("Invalid attribute type {}", type); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-18 18:23:36 +00:00
										 |  |  | InputGenericInfo GetAttributeInfo(EmitContext& ctx, AttributeType type, Id id) { | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |     switch (type) { | 
					
						
							|  |  |  |     case AttributeType::Float: | 
					
						
							| 
									
										
										
										
											2023-02-18 18:23:36 +00:00
										 |  |  |         return InputGenericInfo{id, ctx.input_f32, ctx.F32[1], InputGenericLoadOp::None}; | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |     case AttributeType::UnsignedInt: | 
					
						
							| 
									
										
										
										
											2023-02-18 18:23:36 +00:00
										 |  |  |         return InputGenericInfo{id, ctx.input_u32, ctx.U32[1], InputGenericLoadOp::Bitcast}; | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |     case AttributeType::SignedInt: | 
					
						
							| 
									
										
										
										
											2023-02-18 18:23:36 +00:00
										 |  |  |         return InputGenericInfo{id, ctx.input_s32, ctx.TypeInt(32, true), | 
					
						
							|  |  |  |                                 InputGenericLoadOp::Bitcast}; | 
					
						
							|  |  |  |     case AttributeType::SignedScaled: | 
					
						
							|  |  |  |         return ctx.profile.support_scaled_attributes | 
					
						
							|  |  |  |                    ? InputGenericInfo{id, ctx.input_f32, ctx.F32[1], InputGenericLoadOp::None} | 
					
						
							|  |  |  |                    : InputGenericInfo{id, ctx.input_s32, ctx.TypeInt(32, true), | 
					
						
							|  |  |  |                                       InputGenericLoadOp::SToF}; | 
					
						
							|  |  |  |     case AttributeType::UnsignedScaled: | 
					
						
							|  |  |  |         return ctx.profile.support_scaled_attributes | 
					
						
							|  |  |  |                    ? InputGenericInfo{id, ctx.input_f32, ctx.F32[1], InputGenericLoadOp::None} | 
					
						
							|  |  |  |                    : InputGenericInfo{id, ctx.input_u32, ctx.U32[1], InputGenericLoadOp::UToF}; | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |     case AttributeType::Disabled: | 
					
						
							| 
									
										
										
										
											2023-02-18 18:23:36 +00:00
										 |  |  |         return InputGenericInfo{}; | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     throw InvalidArgument("Invalid attribute type {}", type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-23 03:52:54 -03:00
										 |  |  | std::string_view StageName(Stage stage) { | 
					
						
							|  |  |  |     switch (stage) { | 
					
						
							|  |  |  |     case Stage::VertexA: | 
					
						
							|  |  |  |         return "vs_a"; | 
					
						
							|  |  |  |     case Stage::VertexB: | 
					
						
							|  |  |  |         return "vs"; | 
					
						
							|  |  |  |     case Stage::TessellationControl: | 
					
						
							|  |  |  |         return "tcs"; | 
					
						
							|  |  |  |     case Stage::TessellationEval: | 
					
						
							|  |  |  |         return "tes"; | 
					
						
							|  |  |  |     case Stage::Geometry: | 
					
						
							|  |  |  |         return "gs"; | 
					
						
							|  |  |  |     case Stage::Fragment: | 
					
						
							|  |  |  |         return "fs"; | 
					
						
							|  |  |  |     case Stage::Compute: | 
					
						
							|  |  |  |         return "cs"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     throw InvalidArgument("Invalid stage {}", stage); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename... Args> | 
					
						
							|  |  |  | void Name(EmitContext& ctx, Id object, std::string_view format_str, Args&&... args) { | 
					
						
							| 
									
										
										
										
											2021-06-28 23:44:03 -04:00
										 |  |  |     ctx.Name(object, fmt::format(fmt::runtime(format_str), StageName(ctx.stage), | 
					
						
							|  |  |  |                                  std::forward<Args>(args)...) | 
					
						
							|  |  |  |                          .c_str()); | 
					
						
							| 
									
										
										
										
											2021-05-23 03:52:54 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  | void DefineConstBuffers(EmitContext& ctx, const Info& info, Id UniformDefinitions::*member_type, | 
					
						
							|  |  |  |                         u32 binding, Id type, char type_char, u32 element_size) { | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |     const Id array_type{ctx.TypeArray(type, ctx.Const(65536U / element_size))}; | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |     ctx.Decorate(array_type, spv::Decoration::ArrayStride, element_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const Id struct_type{ctx.TypeStruct(array_type)}; | 
					
						
							| 
									
										
										
										
											2021-05-23 03:52:54 -03:00
										 |  |  |     Name(ctx, struct_type, "{}_cbuf_block_{}{}", ctx.stage, type_char, element_size * CHAR_BIT); | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |     ctx.Decorate(struct_type, spv::Decoration::Block); | 
					
						
							|  |  |  |     ctx.MemberName(struct_type, 0, "data"); | 
					
						
							|  |  |  |     ctx.MemberDecorate(struct_type, 0, spv::Decoration::Offset, 0U); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const Id struct_pointer_type{ctx.TypePointer(spv::StorageClass::Uniform, struct_type)}; | 
					
						
							|  |  |  |     const Id uniform_type{ctx.TypePointer(spv::StorageClass::Uniform, type)}; | 
					
						
							|  |  |  |     ctx.uniform_types.*member_type = uniform_type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (const ConstantBufferDescriptor& desc : info.constant_buffer_descriptors) { | 
					
						
							|  |  |  |         const Id id{ctx.AddGlobalVariable(struct_pointer_type, spv::StorageClass::Uniform)}; | 
					
						
							|  |  |  |         ctx.Decorate(id, spv::Decoration::Binding, binding); | 
					
						
							|  |  |  |         ctx.Decorate(id, spv::Decoration::DescriptorSet, 0U); | 
					
						
							|  |  |  |         ctx.Name(id, fmt::format("c{}", desc.index)); | 
					
						
							|  |  |  |         for (size_t i = 0; i < desc.count; ++i) { | 
					
						
							|  |  |  |             ctx.cbufs[desc.index + i].*member_type = id; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (ctx.profile.supported_spirv >= 0x00010400) { | 
					
						
							|  |  |  |             ctx.interfaces.push_back(id); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         binding += desc.count; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void DefineSsbos(EmitContext& ctx, StorageTypeDefinition& type_def, | 
					
						
							|  |  |  |                  Id StorageDefinitions::*member_type, const Info& info, u32 binding, Id type, | 
					
						
							|  |  |  |                  u32 stride) { | 
					
						
							|  |  |  |     const Id array_type{ctx.TypeRuntimeArray(type)}; | 
					
						
							|  |  |  |     ctx.Decorate(array_type, spv::Decoration::ArrayStride, stride); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const Id struct_type{ctx.TypeStruct(array_type)}; | 
					
						
							|  |  |  |     ctx.Decorate(struct_type, spv::Decoration::Block); | 
					
						
							|  |  |  |     ctx.MemberDecorate(struct_type, 0, spv::Decoration::Offset, 0U); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const Id struct_pointer{ctx.TypePointer(spv::StorageClass::StorageBuffer, struct_type)}; | 
					
						
							|  |  |  |     type_def.array = struct_pointer; | 
					
						
							|  |  |  |     type_def.element = ctx.TypePointer(spv::StorageClass::StorageBuffer, type); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u32 index{}; | 
					
						
							|  |  |  |     for (const StorageBufferDescriptor& desc : info.storage_buffers_descriptors) { | 
					
						
							|  |  |  |         const Id id{ctx.AddGlobalVariable(struct_pointer, spv::StorageClass::StorageBuffer)}; | 
					
						
							|  |  |  |         ctx.Decorate(id, spv::Decoration::Binding, binding); | 
					
						
							|  |  |  |         ctx.Decorate(id, spv::Decoration::DescriptorSet, 0U); | 
					
						
							|  |  |  |         ctx.Name(id, fmt::format("ssbo{}", index)); | 
					
						
							|  |  |  |         if (ctx.profile.supported_spirv >= 0x00010400) { | 
					
						
							|  |  |  |             ctx.interfaces.push_back(id); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         for (size_t i = 0; i < desc.count; ++i) { | 
					
						
							|  |  |  |             ctx.ssbos[index + i].*member_type = id; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         index += desc.count; | 
					
						
							|  |  |  |         binding += desc.count; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id CasFunction(EmitContext& ctx, Operation operation, Id value_type) { | 
					
						
							|  |  |  |     const Id func_type{ctx.TypeFunction(value_type, value_type, value_type)}; | 
					
						
							|  |  |  |     const Id func{ctx.OpFunction(value_type, spv::FunctionControlMask::MaskNone, func_type)}; | 
					
						
							|  |  |  |     const Id op_a{ctx.OpFunctionParameter(value_type)}; | 
					
						
							|  |  |  |     const Id op_b{ctx.OpFunctionParameter(value_type)}; | 
					
						
							|  |  |  |     ctx.AddLabel(); | 
					
						
							|  |  |  |     Id result{}; | 
					
						
							|  |  |  |     switch (operation) { | 
					
						
							|  |  |  |     case Operation::Increment: { | 
					
						
							|  |  |  |         const Id pred{ctx.OpUGreaterThanEqual(ctx.U1, op_a, op_b)}; | 
					
						
							|  |  |  |         const Id incr{ctx.OpIAdd(value_type, op_a, ctx.Constant(value_type, 1))}; | 
					
						
							|  |  |  |         result = ctx.OpSelect(value_type, pred, ctx.u32_zero_value, incr); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     case Operation::Decrement: { | 
					
						
							|  |  |  |         const Id lhs{ctx.OpIEqual(ctx.U1, op_a, ctx.Constant(value_type, 0u))}; | 
					
						
							|  |  |  |         const Id rhs{ctx.OpUGreaterThan(ctx.U1, op_a, op_b)}; | 
					
						
							|  |  |  |         const Id pred{ctx.OpLogicalOr(ctx.U1, lhs, rhs)}; | 
					
						
							|  |  |  |         const Id decr{ctx.OpISub(value_type, op_a, ctx.Constant(value_type, 1))}; | 
					
						
							|  |  |  |         result = ctx.OpSelect(value_type, pred, op_b, decr); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     case Operation::FPAdd: | 
					
						
							|  |  |  |         result = ctx.OpFAdd(value_type, op_a, op_b); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Operation::FPMin: | 
					
						
							|  |  |  |         result = ctx.OpFMin(value_type, op_a, op_b); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Operation::FPMax: | 
					
						
							|  |  |  |         result = ctx.OpFMax(value_type, op_a, op_b); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ctx.OpReturnValue(result); | 
					
						
							|  |  |  |     ctx.OpFunctionEnd(); | 
					
						
							|  |  |  |     return func; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id CasLoop(EmitContext& ctx, Operation operation, Id array_pointer, Id element_pointer, | 
					
						
							|  |  |  |            Id value_type, Id memory_type, spv::Scope scope) { | 
					
						
							|  |  |  |     const bool is_shared{scope == spv::Scope::Workgroup}; | 
					
						
							|  |  |  |     const bool is_struct{!is_shared || ctx.profile.support_explicit_workgroup_layout}; | 
					
						
							|  |  |  |     const Id cas_func{CasFunction(ctx, operation, value_type)}; | 
					
						
							|  |  |  |     const Id zero{ctx.u32_zero_value}; | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |     const Id scope_id{ctx.Const(static_cast<u32>(scope))}; | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const Id loop_header{ctx.OpLabel()}; | 
					
						
							|  |  |  |     const Id continue_block{ctx.OpLabel()}; | 
					
						
							|  |  |  |     const Id merge_block{ctx.OpLabel()}; | 
					
						
							|  |  |  |     const Id func_type{is_shared | 
					
						
							|  |  |  |                            ? ctx.TypeFunction(value_type, ctx.U32[1], value_type) | 
					
						
							|  |  |  |                            : ctx.TypeFunction(value_type, ctx.U32[1], value_type, array_pointer)}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const Id func{ctx.OpFunction(value_type, spv::FunctionControlMask::MaskNone, func_type)}; | 
					
						
							|  |  |  |     const Id index{ctx.OpFunctionParameter(ctx.U32[1])}; | 
					
						
							|  |  |  |     const Id op_b{ctx.OpFunctionParameter(value_type)}; | 
					
						
							|  |  |  |     const Id base{is_shared ? ctx.shared_memory_u32 : ctx.OpFunctionParameter(array_pointer)}; | 
					
						
							|  |  |  |     ctx.AddLabel(); | 
					
						
							|  |  |  |     ctx.OpBranch(loop_header); | 
					
						
							|  |  |  |     ctx.AddLabel(loop_header); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ctx.OpLoopMerge(merge_block, continue_block, spv::LoopControlMask::MaskNone); | 
					
						
							|  |  |  |     ctx.OpBranch(continue_block); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ctx.AddLabel(continue_block); | 
					
						
							|  |  |  |     const Id word_pointer{is_struct ? ctx.OpAccessChain(element_pointer, base, zero, index) | 
					
						
							|  |  |  |                                     : ctx.OpAccessChain(element_pointer, base, index)}; | 
					
						
							|  |  |  |     if (value_type.value == ctx.F32[2].value) { | 
					
						
							|  |  |  |         const Id u32_value{ctx.OpLoad(ctx.U32[1], word_pointer)}; | 
					
						
							|  |  |  |         const Id value{ctx.OpUnpackHalf2x16(ctx.F32[2], u32_value)}; | 
					
						
							|  |  |  |         const Id new_value{ctx.OpFunctionCall(value_type, cas_func, value, op_b)}; | 
					
						
							|  |  |  |         const Id u32_new_value{ctx.OpPackHalf2x16(ctx.U32[1], new_value)}; | 
					
						
							|  |  |  |         const Id atomic_res{ctx.OpAtomicCompareExchange(ctx.U32[1], word_pointer, scope_id, zero, | 
					
						
							|  |  |  |                                                         zero, u32_new_value, u32_value)}; | 
					
						
							|  |  |  |         const Id success{ctx.OpIEqual(ctx.U1, atomic_res, u32_value)}; | 
					
						
							|  |  |  |         ctx.OpBranchConditional(success, merge_block, loop_header); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ctx.AddLabel(merge_block); | 
					
						
							|  |  |  |         ctx.OpReturnValue(ctx.OpUnpackHalf2x16(ctx.F32[2], atomic_res)); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         const Id value{ctx.OpLoad(memory_type, word_pointer)}; | 
					
						
							|  |  |  |         const bool matching_type{value_type.value == memory_type.value}; | 
					
						
							|  |  |  |         const Id bitcast_value{matching_type ? value : ctx.OpBitcast(value_type, value)}; | 
					
						
							|  |  |  |         const Id cal_res{ctx.OpFunctionCall(value_type, cas_func, bitcast_value, op_b)}; | 
					
						
							|  |  |  |         const Id new_value{matching_type ? cal_res : ctx.OpBitcast(memory_type, cal_res)}; | 
					
						
							|  |  |  |         const Id atomic_res{ctx.OpAtomicCompareExchange(ctx.U32[1], word_pointer, scope_id, zero, | 
					
						
							|  |  |  |                                                         zero, new_value, value)}; | 
					
						
							|  |  |  |         const Id success{ctx.OpIEqual(ctx.U1, atomic_res, value)}; | 
					
						
							|  |  |  |         ctx.OpBranchConditional(success, merge_block, loop_header); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ctx.AddLabel(merge_block); | 
					
						
							|  |  |  |         ctx.OpReturnValue(ctx.OpBitcast(value_type, atomic_res)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ctx.OpFunctionEnd(); | 
					
						
							|  |  |  |     return func; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | template <typename Desc> | 
					
						
							| 
									
										
										
										
											2021-05-23 03:52:54 -03:00
										 |  |  | std::string NameOf(Stage stage, const Desc& desc, std::string_view prefix) { | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |     if (desc.count > 1) { | 
					
						
							| 
									
										
										
										
											2021-05-23 03:52:54 -03:00
										 |  |  |         return fmt::format("{}_{}{}_{:02x}x{}", StageName(stage), prefix, desc.cbuf_index, | 
					
						
							|  |  |  |                            desc.cbuf_offset, desc.count); | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2021-05-23 03:52:54 -03:00
										 |  |  |         return fmt::format("{}_{}{}_{:02x}", StageName(stage), prefix, desc.cbuf_index, | 
					
						
							|  |  |  |                            desc.cbuf_offset); | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id DescType(EmitContext& ctx, Id sampled_type, Id pointer_type, u32 count) { | 
					
						
							|  |  |  |     if (count > 1) { | 
					
						
							|  |  |  |         const Id array_type{ctx.TypeArray(sampled_type, ctx.Const(count))}; | 
					
						
							|  |  |  |         return ctx.TypePointer(spv::StorageClass::UniformConstant, array_type); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         return pointer_type; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  | } // Anonymous namespace
 | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | void VectorTypes::Define(Sirit::Module& sirit_ctx, Id base_type, std::string_view name) { | 
					
						
							|  |  |  |     defs[0] = sirit_ctx.Name(base_type, name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::array<char, 6> def_name; | 
					
						
							|  |  |  |     for (int i = 1; i < 4; ++i) { | 
					
						
							|  |  |  |         const std::string_view def_name_view( | 
					
						
							|  |  |  |             def_name.data(), | 
					
						
							|  |  |  |             fmt::format_to_n(def_name.data(), def_name.size(), "{}x{}", name, i + 1).size); | 
					
						
							| 
									
										
										
										
											2021-04-05 22:25:22 -04:00
										 |  |  |         defs[static_cast<size_t>(i)] = | 
					
						
							|  |  |  |             sirit_ctx.Name(sirit_ctx.TypeVector(base_type, i + 1), def_name_view); | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-21 02:12:32 -03:00
										 |  |  | EmitContext::EmitContext(const Profile& profile_, const RuntimeInfo& runtime_info_, | 
					
						
							|  |  |  |                          IR::Program& program, Bindings& bindings) | 
					
						
							| 
									
										
										
										
											2021-08-01 18:57:45 -03:00
										 |  |  |     : Sirit::Module(profile_.supported_spirv), profile{profile_}, runtime_info{runtime_info_}, | 
					
						
							|  |  |  |       stage{program.stage}, texture_rescaling_index{bindings.texture_scaling_index}, | 
					
						
							|  |  |  |       image_rescaling_index{bindings.image_scaling_index} { | 
					
						
							| 
									
										
										
										
											2021-05-23 03:58:11 -03:00
										 |  |  |     const bool is_unified{profile.unified_descriptor_binding}; | 
					
						
							| 
									
										
										
										
											2021-05-21 02:12:32 -03:00
										 |  |  |     u32& uniform_binding{is_unified ? bindings.unified : bindings.uniform_buffer}; | 
					
						
							|  |  |  |     u32& storage_binding{is_unified ? bindings.unified : bindings.storage_buffer}; | 
					
						
							|  |  |  |     u32& texture_binding{is_unified ? bindings.unified : bindings.texture}; | 
					
						
							|  |  |  |     u32& image_binding{is_unified ? bindings.unified : bindings.image}; | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     AddCapability(spv::Capability::Shader); | 
					
						
							|  |  |  |     DefineCommonTypes(program.info); | 
					
						
							|  |  |  |     DefineCommonConstants(); | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |     DefineInterfaces(program); | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  |     DefineLocalMemory(program); | 
					
						
							|  |  |  |     DefineSharedMemory(program); | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |     DefineSharedMemoryFunctions(program); | 
					
						
							| 
									
										
										
										
											2021-05-23 03:58:11 -03:00
										 |  |  |     DefineConstantBuffers(program.info, uniform_binding); | 
					
						
							| 
									
										
										
										
											2022-03-17 13:30:21 -04:00
										 |  |  |     DefineConstantBufferIndirectFunctions(program.info); | 
					
						
							| 
									
										
										
										
											2021-05-23 03:58:11 -03:00
										 |  |  |     DefineStorageBuffers(program.info, storage_binding); | 
					
						
							|  |  |  |     DefineTextureBuffers(program.info, texture_binding); | 
					
						
							|  |  |  |     DefineImageBuffers(program.info, image_binding); | 
					
						
							| 
									
										
										
										
											2021-08-01 18:57:45 -03:00
										 |  |  |     DefineTextures(program.info, texture_binding, bindings.texture_scaling_index); | 
					
						
							|  |  |  |     DefineImages(program.info, image_binding, bindings.image_scaling_index); | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |     DefineAttributeMemAccess(program.info); | 
					
						
							| 
									
										
										
										
											2021-04-19 16:33:23 -03:00
										 |  |  |     DefineGlobalMemoryFunctions(program.info); | 
					
						
							| 
									
										
										
										
											2021-07-25 22:26:23 -03:00
										 |  |  |     DefineRescalingInput(program.info); | 
					
						
							| 
									
										
										
										
											2022-09-01 22:05:11 +08:00
										 |  |  |     DefineRenderArea(program.info); | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | EmitContext::~EmitContext() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitContext::Def(const IR::Value& value) { | 
					
						
							|  |  |  |     if (!value.IsImmediate()) { | 
					
						
							| 
									
										
										
										
											2021-04-01 01:07:51 -03:00
										 |  |  |         return value.InstRecursive()->Definition<Id>(); | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     } | 
					
						
							|  |  |  |     switch (value.Type()) { | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     case IR::Type::Void: | 
					
						
							|  |  |  |         // Void instructions are used for optional arguments (e.g. texture offsets)
 | 
					
						
							|  |  |  |         // They are not meant to be used in the SPIR-V module
 | 
					
						
							|  |  |  |         return Id{}; | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     case IR::Type::U1: | 
					
						
							|  |  |  |         return value.U1() ? true_value : false_value; | 
					
						
							|  |  |  |     case IR::Type::U32: | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |         return Const(value.U32()); | 
					
						
							| 
									
										
										
										
											2021-03-20 05:04:12 -03:00
										 |  |  |     case IR::Type::U64: | 
					
						
							|  |  |  |         return Constant(U64, value.U64()); | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     case IR::Type::F32: | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |         return Const(value.F32()); | 
					
						
							| 
									
										
										
										
											2021-03-18 02:53:57 -04:00
										 |  |  |     case IR::Type::F64: | 
					
						
							|  |  |  |         return Constant(F64[1], value.F64()); | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     default: | 
					
						
							|  |  |  |         throw NotImplementedException("Immediate type {}", value.Type()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-23 03:58:11 -03:00
										 |  |  | Id EmitContext::BitOffset8(const IR::Value& offset) { | 
					
						
							|  |  |  |     if (offset.IsImmediate()) { | 
					
						
							|  |  |  |         return Const((offset.U32() % 4) * 8); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return OpBitwiseAnd(U32[1], OpShiftLeftLogical(U32[1], Def(offset), Const(3u)), Const(24u)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitContext::BitOffset16(const IR::Value& offset) { | 
					
						
							|  |  |  |     if (offset.IsImmediate()) { | 
					
						
							|  |  |  |         return Const(((offset.U32() / 2) % 2) * 16); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return OpBitwiseAnd(U32[1], OpShiftLeftLogical(U32[1], Def(offset), Const(3u)), Const(16u)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  | void EmitContext::DefineCommonTypes(const Info& info) { | 
					
						
							|  |  |  |     void_id = TypeVoid(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     U1 = Name(TypeBool(), "u1"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     F32.Define(*this, TypeFloat(32), "f32"); | 
					
						
							|  |  |  |     U32.Define(*this, TypeInt(32, false), "u32"); | 
					
						
							| 
									
										
										
										
											2021-05-23 03:58:11 -03:00
										 |  |  |     S32.Define(*this, TypeInt(32, true), "s32"); | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  |     private_u32 = Name(TypePointer(spv::StorageClass::Private, U32[1]), "private_u32"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-19 19:28:31 -03:00
										 |  |  |     input_f32 = Name(TypePointer(spv::StorageClass::Input, F32[1]), "input_f32"); | 
					
						
							| 
									
										
										
										
											2021-03-24 01:33:45 -03:00
										 |  |  |     input_u32 = Name(TypePointer(spv::StorageClass::Input, U32[1]), "input_u32"); | 
					
						
							|  |  |  |     input_s32 = Name(TypePointer(spv::StorageClass::Input, TypeInt(32, true)), "input_s32"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-19 19:28:31 -03:00
										 |  |  |     output_f32 = Name(TypePointer(spv::StorageClass::Output, F32[1]), "output_f32"); | 
					
						
							| 
									
										
										
										
											2021-04-16 16:31:15 -03:00
										 |  |  |     output_u32 = Name(TypePointer(spv::StorageClass::Output, U32[1]), "output_u32"); | 
					
						
							| 
									
										
										
										
											2021-03-19 19:28:31 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-23 04:13:34 -03:00
										 |  |  |     if (info.uses_int8 && profile.support_int8) { | 
					
						
							| 
									
										
										
										
											2021-03-09 17:14:57 -03:00
										 |  |  |         AddCapability(spv::Capability::Int8); | 
					
						
							|  |  |  |         U8 = Name(TypeInt(8, false), "u8"); | 
					
						
							|  |  |  |         S8 = Name(TypeInt(8, true), "s8"); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-23 04:13:34 -03:00
										 |  |  |     if (info.uses_int16 && profile.support_int16) { | 
					
						
							| 
									
										
										
										
											2021-03-09 17:14:57 -03:00
										 |  |  |         AddCapability(spv::Capability::Int16); | 
					
						
							|  |  |  |         U16 = Name(TypeInt(16, false), "u16"); | 
					
						
							|  |  |  |         S16 = Name(TypeInt(16, true), "s16"); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-11-19 15:53:19 +00:00
										 |  |  |     if (info.uses_int64 && profile.support_int64) { | 
					
						
							| 
									
										
										
										
											2021-03-09 17:14:57 -03:00
										 |  |  |         AddCapability(spv::Capability::Int64); | 
					
						
							|  |  |  |         U64 = Name(TypeInt(64, false), "u64"); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     if (info.uses_fp16) { | 
					
						
							|  |  |  |         AddCapability(spv::Capability::Float16); | 
					
						
							|  |  |  |         F16.Define(*this, TypeFloat(16), "f16"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (info.uses_fp64) { | 
					
						
							|  |  |  |         AddCapability(spv::Capability::Float64); | 
					
						
							|  |  |  |         F64.Define(*this, TypeFloat(64), "f64"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitContext::DefineCommonConstants() { | 
					
						
							|  |  |  |     true_value = ConstantTrue(U1); | 
					
						
							|  |  |  |     false_value = ConstantFalse(U1); | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |     u32_zero_value = Const(0U); | 
					
						
							|  |  |  |     f32_zero_value = Const(0.0f); | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  | void EmitContext::DefineInterfaces(const IR::Program& program) { | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |     DefineInputs(program); | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |     DefineOutputs(program); | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  | void EmitContext::DefineLocalMemory(const IR::Program& program) { | 
					
						
							|  |  |  |     if (program.local_memory_size == 0) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const u32 num_elements{Common::DivCeil(program.local_memory_size, 4U)}; | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |     const Id type{TypeArray(U32[1], Const(num_elements))}; | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  |     const Id pointer{TypePointer(spv::StorageClass::Private, type)}; | 
					
						
							|  |  |  |     local_memory = AddGlobalVariable(pointer, spv::StorageClass::Private); | 
					
						
							|  |  |  |     if (profile.supported_spirv >= 0x00010400) { | 
					
						
							|  |  |  |         interfaces.push_back(local_memory); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitContext::DefineSharedMemory(const IR::Program& program) { | 
					
						
							|  |  |  |     if (program.shared_memory_size == 0) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const auto make{[&](Id element_type, u32 element_size) { | 
					
						
							|  |  |  |         const u32 num_elements{Common::DivCeil(program.shared_memory_size, element_size)}; | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |         const Id array_type{TypeArray(element_type, Const(num_elements))}; | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  |         Decorate(array_type, spv::Decoration::ArrayStride, element_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const Id struct_type{TypeStruct(array_type)}; | 
					
						
							|  |  |  |         MemberDecorate(struct_type, 0U, spv::Decoration::Offset, 0U); | 
					
						
							|  |  |  |         Decorate(struct_type, spv::Decoration::Block); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const Id pointer{TypePointer(spv::StorageClass::Workgroup, struct_type)}; | 
					
						
							|  |  |  |         const Id element_pointer{TypePointer(spv::StorageClass::Workgroup, element_type)}; | 
					
						
							|  |  |  |         const Id variable{AddGlobalVariable(pointer, spv::StorageClass::Workgroup)}; | 
					
						
							|  |  |  |         Decorate(variable, spv::Decoration::Aliased); | 
					
						
							|  |  |  |         interfaces.push_back(variable); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         return std::make_tuple(variable, element_pointer, pointer); | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  |     }}; | 
					
						
							|  |  |  |     if (profile.support_explicit_workgroup_layout) { | 
					
						
							|  |  |  |         AddExtension("SPV_KHR_workgroup_memory_explicit_layout"); | 
					
						
							|  |  |  |         AddCapability(spv::Capability::WorkgroupMemoryExplicitLayoutKHR); | 
					
						
							|  |  |  |         if (program.info.uses_int8) { | 
					
						
							|  |  |  |             AddCapability(spv::Capability::WorkgroupMemoryExplicitLayout8BitAccessKHR); | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |             std::tie(shared_memory_u8, shared_u8, std::ignore) = make(U8, 1); | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  |         } | 
					
						
							|  |  |  |         if (program.info.uses_int16) { | 
					
						
							|  |  |  |             AddCapability(spv::Capability::WorkgroupMemoryExplicitLayout16BitAccessKHR); | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |             std::tie(shared_memory_u16, shared_u16, std::ignore) = make(U16, 2); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (program.info.uses_int64) { | 
					
						
							|  |  |  |             std::tie(shared_memory_u64, shared_u64, std::ignore) = make(U64, 8); | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         std::tie(shared_memory_u32, shared_u32, shared_memory_u32_type) = make(U32[1], 4); | 
					
						
							|  |  |  |         std::tie(shared_memory_u32x2, shared_u32x2, std::ignore) = make(U32[2], 8); | 
					
						
							|  |  |  |         std::tie(shared_memory_u32x4, shared_u32x4, std::ignore) = make(U32[4], 16); | 
					
						
							| 
									
										
										
										
											2021-04-02 21:59:58 +02:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  |     } | 
					
						
							|  |  |  |     const u32 num_elements{Common::DivCeil(program.shared_memory_size, 4U)}; | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |     const Id type{TypeArray(U32[1], Const(num_elements))}; | 
					
						
							| 
									
										
										
										
											2021-04-11 02:07:02 -04:00
										 |  |  |     shared_memory_u32_type = TypePointer(spv::StorageClass::Workgroup, type); | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  |     shared_u32 = TypePointer(spv::StorageClass::Workgroup, U32[1]); | 
					
						
							| 
									
										
										
										
											2021-04-11 02:07:02 -04:00
										 |  |  |     shared_memory_u32 = AddGlobalVariable(shared_memory_u32_type, spv::StorageClass::Workgroup); | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  |     interfaces.push_back(shared_memory_u32); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const Id func_type{TypeFunction(void_id, U32[1], U32[1])}; | 
					
						
							|  |  |  |     const auto make_function{[&](u32 mask, u32 size) { | 
					
						
							|  |  |  |         const Id loop_header{OpLabel()}; | 
					
						
							|  |  |  |         const Id continue_block{OpLabel()}; | 
					
						
							|  |  |  |         const Id merge_block{OpLabel()}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const Id func{OpFunction(void_id, spv::FunctionControlMask::MaskNone, func_type)}; | 
					
						
							|  |  |  |         const Id offset{OpFunctionParameter(U32[1])}; | 
					
						
							|  |  |  |         const Id insert_value{OpFunctionParameter(U32[1])}; | 
					
						
							|  |  |  |         AddLabel(); | 
					
						
							|  |  |  |         OpBranch(loop_header); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         AddLabel(loop_header); | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |         const Id word_offset{OpShiftRightArithmetic(U32[1], offset, Const(2U))}; | 
					
						
							|  |  |  |         const Id shift_offset{OpShiftLeftLogical(U32[1], offset, Const(3U))}; | 
					
						
							|  |  |  |         const Id bit_offset{OpBitwiseAnd(U32[1], shift_offset, Const(mask))}; | 
					
						
							|  |  |  |         const Id count{Const(size)}; | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  |         OpLoopMerge(merge_block, continue_block, spv::LoopControlMask::MaskNone); | 
					
						
							|  |  |  |         OpBranch(continue_block); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         AddLabel(continue_block); | 
					
						
							|  |  |  |         const Id word_pointer{OpAccessChain(shared_u32, shared_memory_u32, word_offset)}; | 
					
						
							|  |  |  |         const Id old_value{OpLoad(U32[1], word_pointer)}; | 
					
						
							|  |  |  |         const Id new_value{OpBitFieldInsert(U32[1], old_value, insert_value, bit_offset, count)}; | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |         const Id atomic_res{OpAtomicCompareExchange(U32[1], word_pointer, Const(1U), u32_zero_value, | 
					
						
							|  |  |  |                                                     u32_zero_value, new_value, old_value)}; | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  |         const Id success{OpIEqual(U1, atomic_res, old_value)}; | 
					
						
							|  |  |  |         OpBranchConditional(success, merge_block, loop_header); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         AddLabel(merge_block); | 
					
						
							|  |  |  |         OpReturn(); | 
					
						
							|  |  |  |         OpFunctionEnd(); | 
					
						
							|  |  |  |         return func; | 
					
						
							|  |  |  |     }}; | 
					
						
							|  |  |  |     if (program.info.uses_int8) { | 
					
						
							|  |  |  |         shared_store_u8_func = make_function(24, 8); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (program.info.uses_int16) { | 
					
						
							|  |  |  |         shared_store_u16_func = make_function(16, 16); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitContext::DefineSharedMemoryFunctions(const IR::Program& program) { | 
					
						
							| 
									
										
										
										
											2021-04-11 02:07:02 -04:00
										 |  |  |     if (program.info.uses_shared_increment) { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         increment_cas_shared = CasLoop(*this, Operation::Increment, shared_memory_u32_type, | 
					
						
							|  |  |  |                                        shared_u32, U32[1], U32[1], spv::Scope::Workgroup); | 
					
						
							| 
									
										
										
										
											2021-04-11 02:07:02 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (program.info.uses_shared_decrement) { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         decrement_cas_shared = CasLoop(*this, Operation::Decrement, shared_memory_u32_type, | 
					
						
							|  |  |  |                                        shared_u32, U32[1], U32[1], spv::Scope::Workgroup); | 
					
						
							| 
									
										
										
										
											2021-04-11 02:07:02 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  | void EmitContext::DefineAttributeMemAccess(const Info& info) { | 
					
						
							| 
									
										
										
										
											2021-04-05 04:03:12 +02:00
										 |  |  |     const auto make_load{[&] { | 
					
						
							| 
									
										
										
										
											2021-04-12 19:41:22 -03:00
										 |  |  |         const bool is_array{stage == Stage::Geometry}; | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |         const Id end_block{OpLabel()}; | 
					
						
							|  |  |  |         const Id default_label{OpLabel()}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-12 19:41:22 -03:00
										 |  |  |         const Id func_type_load{is_array ? TypeFunction(F32[1], U32[1], U32[1]) | 
					
						
							|  |  |  |                                          : TypeFunction(F32[1], U32[1])}; | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |         const Id func{OpFunction(F32[1], spv::FunctionControlMask::MaskNone, func_type_load)}; | 
					
						
							|  |  |  |         const Id offset{OpFunctionParameter(U32[1])}; | 
					
						
							| 
									
										
										
										
											2021-04-12 19:41:22 -03:00
										 |  |  |         const Id vertex{is_array ? OpFunctionParameter(U32[1]) : Id{}}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |         AddLabel(); | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |         const Id base_index{OpShiftRightArithmetic(U32[1], offset, Const(2U))}; | 
					
						
							|  |  |  |         const Id masked_index{OpBitwiseAnd(U32[1], base_index, Const(3U))}; | 
					
						
							|  |  |  |         const Id compare_index{OpShiftRightArithmetic(U32[1], base_index, Const(2U))}; | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |         std::vector<Sirit::Literal> literals; | 
					
						
							|  |  |  |         std::vector<Id> labels; | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |         if (info.loads.AnyComponent(IR::Attribute::PositionX)) { | 
					
						
							| 
									
										
										
										
											2021-04-04 09:38:15 +02:00
										 |  |  |             literals.push_back(static_cast<u32>(IR::Attribute::PositionX) >> 2); | 
					
						
							|  |  |  |             labels.push_back(OpLabel()); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |         const u32 base_attribute_value = static_cast<u32>(IR::Attribute::Generic0X) >> 2; | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |         for (u32 index = 0; index < static_cast<u32>(IR::NUM_GENERICS); ++index) { | 
					
						
							|  |  |  |             if (!info.loads.Generic(index)) { | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |             literals.push_back(base_attribute_value + index); | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |             labels.push_back(OpLabel()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         OpSelectionMerge(end_block, spv::SelectionControlMask::MaskNone); | 
					
						
							|  |  |  |         OpSwitch(compare_index, default_label, literals, labels); | 
					
						
							|  |  |  |         AddLabel(default_label); | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |         OpReturnValue(Const(0.0f)); | 
					
						
							| 
									
										
										
										
											2021-04-05 04:03:12 +02:00
										 |  |  |         size_t label_index{0}; | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |         if (info.loads.AnyComponent(IR::Attribute::PositionX)) { | 
					
						
							| 
									
										
										
										
											2021-04-04 09:38:15 +02:00
										 |  |  |             AddLabel(labels[label_index]); | 
					
						
							| 
									
										
										
										
											2022-08-02 17:41:41 +01:00
										 |  |  |             const Id pointer{[&]() { | 
					
						
							|  |  |  |                 if (need_input_position_indirect) { | 
					
						
							|  |  |  |                     if (is_array) | 
					
						
							|  |  |  |                         return OpAccessChain(input_f32, input_position, vertex, u32_zero_value, | 
					
						
							|  |  |  |                                              masked_index); | 
					
						
							|  |  |  |                     else | 
					
						
							|  |  |  |                         return OpAccessChain(input_f32, input_position, u32_zero_value, | 
					
						
							|  |  |  |                                              masked_index); | 
					
						
							| 
									
										
										
										
											2023-01-05 22:10:21 +00:00
										 |  |  |                 } else { | 
					
						
							| 
									
										
										
										
											2022-08-02 17:41:41 +01:00
										 |  |  |                     if (is_array) | 
					
						
							|  |  |  |                         return OpAccessChain(input_f32, input_position, vertex, masked_index); | 
					
						
							|  |  |  |                     else | 
					
						
							|  |  |  |                         return OpAccessChain(input_f32, input_position, masked_index); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             }()}; | 
					
						
							| 
									
										
										
										
											2021-04-12 19:41:22 -03:00
										 |  |  |             const Id result{OpLoad(F32[1], pointer)}; | 
					
						
							| 
									
										
										
										
											2021-04-04 09:38:15 +02:00
										 |  |  |             OpReturnValue(result); | 
					
						
							| 
									
										
										
										
											2021-04-05 04:03:12 +02:00
										 |  |  |             ++label_index; | 
					
						
							| 
									
										
										
										
											2021-04-04 09:38:15 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |         for (size_t index = 0; index < IR::NUM_GENERICS; ++index) { | 
					
						
							|  |  |  |             if (!info.loads.Generic(index)) { | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             AddLabel(labels[label_index]); | 
					
						
							| 
									
										
										
										
											2023-02-18 18:23:36 +00:00
										 |  |  |             const auto& generic{input_generics.at(index)}; | 
					
						
							|  |  |  |             const Id generic_id{generic.id}; | 
					
						
							|  |  |  |             if (!ValidId(generic_id)) { | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |                 OpReturnValue(Const(0.0f)); | 
					
						
							| 
									
										
										
										
											2021-04-05 04:03:12 +02:00
										 |  |  |                 ++label_index; | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2023-02-18 18:23:36 +00:00
										 |  |  |             const Id pointer{ | 
					
						
							|  |  |  |                 is_array ? OpAccessChain(generic.pointer_type, generic_id, vertex, masked_index) | 
					
						
							|  |  |  |                          : OpAccessChain(generic.pointer_type, generic_id, masked_index)}; | 
					
						
							|  |  |  |             const Id value{OpLoad(generic.component_type, pointer)}; | 
					
						
							|  |  |  |             const Id result{[this, generic, value]() { | 
					
						
							|  |  |  |                 switch (generic.load_op) { | 
					
						
							|  |  |  |                 case InputGenericLoadOp::Bitcast: | 
					
						
							|  |  |  |                     return OpBitcast(F32[1], value); | 
					
						
							|  |  |  |                 case InputGenericLoadOp::SToF: | 
					
						
							|  |  |  |                     return OpConvertSToF(F32[1], value); | 
					
						
							|  |  |  |                 case InputGenericLoadOp::UToF: | 
					
						
							|  |  |  |                     return OpConvertUToF(F32[1], value); | 
					
						
							|  |  |  |                 default: | 
					
						
							|  |  |  |                     return value; | 
					
						
							|  |  |  |                 }; | 
					
						
							|  |  |  |             }()}; | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |             OpReturnValue(result); | 
					
						
							| 
									
										
										
										
											2021-04-05 04:03:12 +02:00
										 |  |  |             ++label_index; | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         AddLabel(end_block); | 
					
						
							|  |  |  |         OpUnreachable(); | 
					
						
							|  |  |  |         OpFunctionEnd(); | 
					
						
							|  |  |  |         return func; | 
					
						
							|  |  |  |     }}; | 
					
						
							| 
									
										
										
										
											2021-04-05 04:03:12 +02:00
										 |  |  |     const auto make_store{[&] { | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |         const Id end_block{OpLabel()}; | 
					
						
							|  |  |  |         const Id default_label{OpLabel()}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const Id func_type_store{TypeFunction(void_id, U32[1], F32[1])}; | 
					
						
							|  |  |  |         const Id func{OpFunction(void_id, spv::FunctionControlMask::MaskNone, func_type_store)}; | 
					
						
							|  |  |  |         const Id offset{OpFunctionParameter(U32[1])}; | 
					
						
							|  |  |  |         const Id store_value{OpFunctionParameter(F32[1])}; | 
					
						
							|  |  |  |         AddLabel(); | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |         const Id base_index{OpShiftRightArithmetic(U32[1], offset, Const(2U))}; | 
					
						
							|  |  |  |         const Id masked_index{OpBitwiseAnd(U32[1], base_index, Const(3U))}; | 
					
						
							|  |  |  |         const Id compare_index{OpShiftRightArithmetic(U32[1], base_index, Const(2U))}; | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |         std::vector<Sirit::Literal> literals; | 
					
						
							|  |  |  |         std::vector<Id> labels; | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |         if (info.stores.AnyComponent(IR::Attribute::PositionX)) { | 
					
						
							| 
									
										
										
										
											2021-04-04 09:38:15 +02:00
										 |  |  |             literals.push_back(static_cast<u32>(IR::Attribute::PositionX) >> 2); | 
					
						
							|  |  |  |             labels.push_back(OpLabel()); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |         const u32 base_attribute_value = static_cast<u32>(IR::Attribute::Generic0X) >> 2; | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |         for (size_t index = 0; index < IR::NUM_GENERICS; ++index) { | 
					
						
							|  |  |  |             if (!info.stores.Generic(index)) { | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |             literals.push_back(base_attribute_value + static_cast<u32>(index)); | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |             labels.push_back(OpLabel()); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |         if (info.stores.ClipDistances()) { | 
					
						
							| 
									
										
										
										
											2021-04-04 09:38:15 +02:00
										 |  |  |             literals.push_back(static_cast<u32>(IR::Attribute::ClipDistance0) >> 2); | 
					
						
							|  |  |  |             labels.push_back(OpLabel()); | 
					
						
							|  |  |  |             literals.push_back(static_cast<u32>(IR::Attribute::ClipDistance4) >> 2); | 
					
						
							|  |  |  |             labels.push_back(OpLabel()); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |         OpSelectionMerge(end_block, spv::SelectionControlMask::MaskNone); | 
					
						
							|  |  |  |         OpSwitch(compare_index, default_label, literals, labels); | 
					
						
							|  |  |  |         AddLabel(default_label); | 
					
						
							|  |  |  |         OpReturn(); | 
					
						
							| 
									
										
										
										
											2021-04-05 04:03:12 +02:00
										 |  |  |         size_t label_index{0}; | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |         if (info.stores.AnyComponent(IR::Attribute::PositionX)) { | 
					
						
							| 
									
										
										
										
											2021-04-04 09:38:15 +02:00
										 |  |  |             AddLabel(labels[label_index]); | 
					
						
							|  |  |  |             const Id pointer{OpAccessChain(output_f32, output_position, masked_index)}; | 
					
						
							|  |  |  |             OpStore(pointer, store_value); | 
					
						
							|  |  |  |             OpReturn(); | 
					
						
							| 
									
										
										
										
											2021-04-05 04:03:12 +02:00
										 |  |  |             ++label_index; | 
					
						
							| 
									
										
										
										
											2021-04-04 09:38:15 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |         for (size_t index = 0; index < IR::NUM_GENERICS; ++index) { | 
					
						
							|  |  |  |             if (!info.stores.Generic(index)) { | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |             if (output_generics[index][0].num_components != 4) { | 
					
						
							| 
									
										
										
										
											2021-04-14 01:04:59 -03:00
										 |  |  |                 throw NotImplementedException("Physical stores and transform feedbacks"); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |             AddLabel(labels[label_index]); | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |             const Id generic_id{output_generics[index][0].id}; | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |             const Id pointer{OpAccessChain(output_f32, generic_id, masked_index)}; | 
					
						
							|  |  |  |             OpStore(pointer, store_value); | 
					
						
							|  |  |  |             OpReturn(); | 
					
						
							| 
									
										
										
										
											2021-04-05 04:03:12 +02:00
										 |  |  |             ++label_index; | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |         if (info.stores.ClipDistances()) { | 
					
						
							| 
									
										
										
										
											2021-04-04 09:38:15 +02:00
										 |  |  |             AddLabel(labels[label_index]); | 
					
						
							|  |  |  |             const Id pointer{OpAccessChain(output_f32, clip_distances, masked_index)}; | 
					
						
							|  |  |  |             OpStore(pointer, store_value); | 
					
						
							|  |  |  |             OpReturn(); | 
					
						
							| 
									
										
										
										
											2021-04-05 04:03:12 +02:00
										 |  |  |             ++label_index; | 
					
						
							| 
									
										
										
										
											2021-04-04 09:38:15 +02:00
										 |  |  |             AddLabel(labels[label_index]); | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |             const Id fixed_index{OpIAdd(U32[1], masked_index, Const(4U))}; | 
					
						
							| 
									
										
										
										
											2021-04-04 09:38:15 +02:00
										 |  |  |             const Id pointer2{OpAccessChain(output_f32, clip_distances, fixed_index)}; | 
					
						
							|  |  |  |             OpStore(pointer2, store_value); | 
					
						
							|  |  |  |             OpReturn(); | 
					
						
							| 
									
										
										
										
											2021-04-05 04:03:12 +02:00
										 |  |  |             ++label_index; | 
					
						
							| 
									
										
										
										
											2021-04-04 09:38:15 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-04 06:47:14 +02:00
										 |  |  |         AddLabel(end_block); | 
					
						
							|  |  |  |         OpUnreachable(); | 
					
						
							|  |  |  |         OpFunctionEnd(); | 
					
						
							|  |  |  |         return func; | 
					
						
							|  |  |  |     }}; | 
					
						
							|  |  |  |     if (info.loads_indexed_attributes) { | 
					
						
							|  |  |  |         indexed_load_func = make_load(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (info.stores_indexed_attributes) { | 
					
						
							|  |  |  |         indexed_store_func = make_store(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-19 16:33:23 -03:00
										 |  |  | void EmitContext::DefineGlobalMemoryFunctions(const Info& info) { | 
					
						
							| 
									
										
										
										
											2021-07-04 00:34:53 -04:00
										 |  |  |     if (!info.uses_global_memory || !profile.support_int64) { | 
					
						
							| 
									
										
										
										
											2021-04-19 16:33:23 -03:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     using DefPtr = Id StorageDefinitions::*; | 
					
						
							|  |  |  |     const Id zero{u32_zero_value}; | 
					
						
							|  |  |  |     const auto define_body{[&](DefPtr ssbo_member, Id addr, Id element_pointer, u32 shift, | 
					
						
							|  |  |  |                                auto&& callback) { | 
					
						
							|  |  |  |         AddLabel(); | 
					
						
							|  |  |  |         const size_t num_buffers{info.storage_buffers_descriptors.size()}; | 
					
						
							|  |  |  |         for (size_t index = 0; index < num_buffers; ++index) { | 
					
						
							| 
									
										
										
										
											2021-05-02 01:50:27 +02:00
										 |  |  |             if (!info.nvn_buffer_used[index]) { | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-04-19 16:33:23 -03:00
										 |  |  |             const auto& ssbo{info.storage_buffers_descriptors[index]}; | 
					
						
							|  |  |  |             const Id ssbo_addr_cbuf_offset{Const(ssbo.cbuf_offset / 8)}; | 
					
						
							|  |  |  |             const Id ssbo_size_cbuf_offset{Const(ssbo.cbuf_offset / 4 + 2)}; | 
					
						
							|  |  |  |             const Id ssbo_addr_pointer{OpAccessChain( | 
					
						
							|  |  |  |                 uniform_types.U32x2, cbufs[ssbo.cbuf_index].U32x2, zero, ssbo_addr_cbuf_offset)}; | 
					
						
							|  |  |  |             const Id ssbo_size_pointer{OpAccessChain(uniform_types.U32, cbufs[ssbo.cbuf_index].U32, | 
					
						
							|  |  |  |                                                      zero, ssbo_size_cbuf_offset)}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-31 20:02:11 -04:00
										 |  |  |             const u64 ssbo_align_mask{~(profile.min_ssbo_alignment - 1U)}; | 
					
						
							|  |  |  |             const Id unaligned_addr{OpBitcast(U64, OpLoad(U32[2], ssbo_addr_pointer))}; | 
					
						
							|  |  |  |             const Id ssbo_addr{OpBitwiseAnd(U64, unaligned_addr, Constant(U64, ssbo_align_mask))}; | 
					
						
							| 
									
										
										
										
											2021-04-19 16:33:23 -03:00
										 |  |  |             const Id ssbo_size{OpUConvert(U64, OpLoad(U32[1], ssbo_size_pointer))}; | 
					
						
							|  |  |  |             const Id ssbo_end{OpIAdd(U64, ssbo_addr, ssbo_size)}; | 
					
						
							|  |  |  |             const Id cond{OpLogicalAnd(U1, OpUGreaterThanEqual(U1, addr, ssbo_addr), | 
					
						
							|  |  |  |                                        OpULessThan(U1, addr, ssbo_end))}; | 
					
						
							|  |  |  |             const Id then_label{OpLabel()}; | 
					
						
							|  |  |  |             const Id else_label{OpLabel()}; | 
					
						
							|  |  |  |             OpSelectionMerge(else_label, spv::SelectionControlMask::MaskNone); | 
					
						
							|  |  |  |             OpBranchConditional(cond, then_label, else_label); | 
					
						
							|  |  |  |             AddLabel(then_label); | 
					
						
							|  |  |  |             const Id ssbo_id{ssbos[index].*ssbo_member}; | 
					
						
							|  |  |  |             const Id ssbo_offset{OpUConvert(U32[1], OpISub(U64, addr, ssbo_addr))}; | 
					
						
							|  |  |  |             const Id ssbo_index{OpShiftRightLogical(U32[1], ssbo_offset, Const(shift))}; | 
					
						
							|  |  |  |             const Id ssbo_pointer{OpAccessChain(element_pointer, ssbo_id, zero, ssbo_index)}; | 
					
						
							|  |  |  |             callback(ssbo_pointer); | 
					
						
							|  |  |  |             AddLabel(else_label); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }}; | 
					
						
							|  |  |  |     const auto define_load{[&](DefPtr ssbo_member, Id element_pointer, Id type, u32 shift) { | 
					
						
							|  |  |  |         const Id function_type{TypeFunction(type, U64)}; | 
					
						
							|  |  |  |         const Id func_id{OpFunction(type, spv::FunctionControlMask::MaskNone, function_type)}; | 
					
						
							|  |  |  |         const Id addr{OpFunctionParameter(U64)}; | 
					
						
							|  |  |  |         define_body(ssbo_member, addr, element_pointer, shift, | 
					
						
							|  |  |  |                     [&](Id ssbo_pointer) { OpReturnValue(OpLoad(type, ssbo_pointer)); }); | 
					
						
							|  |  |  |         OpReturnValue(ConstantNull(type)); | 
					
						
							|  |  |  |         OpFunctionEnd(); | 
					
						
							|  |  |  |         return func_id; | 
					
						
							|  |  |  |     }}; | 
					
						
							|  |  |  |     const auto define_write{[&](DefPtr ssbo_member, Id element_pointer, Id type, u32 shift) { | 
					
						
							|  |  |  |         const Id function_type{TypeFunction(void_id, U64, type)}; | 
					
						
							|  |  |  |         const Id func_id{OpFunction(void_id, spv::FunctionControlMask::MaskNone, function_type)}; | 
					
						
							|  |  |  |         const Id addr{OpFunctionParameter(U64)}; | 
					
						
							|  |  |  |         const Id data{OpFunctionParameter(type)}; | 
					
						
							|  |  |  |         define_body(ssbo_member, addr, element_pointer, shift, [&](Id ssbo_pointer) { | 
					
						
							|  |  |  |             OpStore(ssbo_pointer, data); | 
					
						
							|  |  |  |             OpReturn(); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |         OpReturn(); | 
					
						
							|  |  |  |         OpFunctionEnd(); | 
					
						
							|  |  |  |         return func_id; | 
					
						
							|  |  |  |     }}; | 
					
						
							|  |  |  |     const auto define{ | 
					
						
							|  |  |  |         [&](DefPtr ssbo_member, const StorageTypeDefinition& type_def, Id type, size_t size) { | 
					
						
							|  |  |  |             const Id element_type{type_def.element}; | 
					
						
							|  |  |  |             const u32 shift{static_cast<u32>(std::countr_zero(size))}; | 
					
						
							|  |  |  |             const Id load_func{define_load(ssbo_member, element_type, type, shift)}; | 
					
						
							|  |  |  |             const Id write_func{define_write(ssbo_member, element_type, type, shift)}; | 
					
						
							|  |  |  |             return std::make_pair(load_func, write_func); | 
					
						
							|  |  |  |         }}; | 
					
						
							|  |  |  |     std::tie(load_global_func_u32, write_global_func_u32) = | 
					
						
							|  |  |  |         define(&StorageDefinitions::U32, storage_types.U32, U32[1], sizeof(u32)); | 
					
						
							|  |  |  |     std::tie(load_global_func_u32x2, write_global_func_u32x2) = | 
					
						
							|  |  |  |         define(&StorageDefinitions::U32x2, storage_types.U32x2, U32[2], sizeof(u32[2])); | 
					
						
							|  |  |  |     std::tie(load_global_func_u32x4, write_global_func_u32x4) = | 
					
						
							|  |  |  |         define(&StorageDefinitions::U32x4, storage_types.U32x4, U32[4], sizeof(u32[4])); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-07-25 22:26:23 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | void EmitContext::DefineRescalingInput(const Info& info) { | 
					
						
							|  |  |  |     if (!info.uses_rescaling_uniform) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-08-01 18:57:45 -03:00
										 |  |  |     if (profile.unified_descriptor_binding) { | 
					
						
							| 
									
										
										
										
											2021-10-16 00:30:43 -04:00
										 |  |  |         DefineRescalingInputPushConstant(); | 
					
						
							| 
									
										
										
										
											2021-08-01 18:57:45 -03:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         DefineRescalingInputUniformConstant(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-16 00:30:43 -04:00
										 |  |  | void EmitContext::DefineRescalingInputPushConstant() { | 
					
						
							|  |  |  |     boost::container::static_vector<Id, 3> members{}; | 
					
						
							| 
									
										
										
										
											2021-07-25 22:26:23 -03:00
										 |  |  |     u32 member_index{0}; | 
					
						
							| 
									
										
										
										
											2021-10-16 00:30:43 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     rescaling_textures_type = TypeArray(U32[1], Const(4u)); | 
					
						
							|  |  |  |     Decorate(rescaling_textures_type, spv::Decoration::ArrayStride, 4u); | 
					
						
							|  |  |  |     members.push_back(rescaling_textures_type); | 
					
						
							|  |  |  |     rescaling_textures_member_index = member_index++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     rescaling_images_type = TypeArray(U32[1], Const(NUM_IMAGE_SCALING_WORDS)); | 
					
						
							|  |  |  |     Decorate(rescaling_images_type, spv::Decoration::ArrayStride, 4u); | 
					
						
							|  |  |  |     members.push_back(rescaling_images_type); | 
					
						
							|  |  |  |     rescaling_images_member_index = member_index++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (stage != Stage::Compute) { | 
					
						
							|  |  |  |         members.push_back(F32[1]); | 
					
						
							|  |  |  |         rescaling_downfactor_member_index = member_index++; | 
					
						
							| 
									
										
										
										
											2021-08-01 18:57:45 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-07-25 22:26:23 -03:00
										 |  |  |     const Id push_constant_struct{TypeStruct(std::span(members.data(), members.size()))}; | 
					
						
							|  |  |  |     Decorate(push_constant_struct, spv::Decoration::Block); | 
					
						
							|  |  |  |     Name(push_constant_struct, "ResolutionInfo"); | 
					
						
							| 
									
										
										
										
											2021-08-01 18:57:45 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-16 00:30:43 -04:00
										 |  |  |     MemberDecorate(push_constant_struct, rescaling_textures_member_index, spv::Decoration::Offset, | 
					
						
							|  |  |  |                    static_cast<u32>(offsetof(RescalingLayout, rescaling_textures))); | 
					
						
							|  |  |  |     MemberName(push_constant_struct, rescaling_textures_member_index, "rescaling_textures"); | 
					
						
							| 
									
										
										
										
											2021-08-01 18:57:45 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-16 00:30:43 -04:00
										 |  |  |     MemberDecorate(push_constant_struct, rescaling_images_member_index, spv::Decoration::Offset, | 
					
						
							|  |  |  |                    static_cast<u32>(offsetof(RescalingLayout, rescaling_images))); | 
					
						
							|  |  |  |     MemberName(push_constant_struct, rescaling_images_member_index, "rescaling_images"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (stage != Stage::Compute) { | 
					
						
							|  |  |  |         MemberDecorate(push_constant_struct, rescaling_downfactor_member_index, | 
					
						
							|  |  |  |                        spv::Decoration::Offset, | 
					
						
							|  |  |  |                        static_cast<u32>(offsetof(RescalingLayout, down_factor))); | 
					
						
							|  |  |  |         MemberName(push_constant_struct, rescaling_downfactor_member_index, "down_factor"); | 
					
						
							| 
									
										
										
										
											2021-08-01 18:57:45 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-07-25 22:26:23 -03:00
										 |  |  |     const Id pointer_type{TypePointer(spv::StorageClass::PushConstant, push_constant_struct)}; | 
					
						
							|  |  |  |     rescaling_push_constants = AddGlobalVariable(pointer_type, spv::StorageClass::PushConstant); | 
					
						
							|  |  |  |     Name(rescaling_push_constants, "rescaling_push_constants"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (profile.supported_spirv >= 0x00010400) { | 
					
						
							|  |  |  |         interfaces.push_back(rescaling_push_constants); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-19 16:33:23 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-01 18:57:45 -03:00
										 |  |  | void EmitContext::DefineRescalingInputUniformConstant() { | 
					
						
							|  |  |  |     const Id pointer_type{TypePointer(spv::StorageClass::UniformConstant, F32[4])}; | 
					
						
							|  |  |  |     rescaling_uniform_constant = | 
					
						
							|  |  |  |         AddGlobalVariable(pointer_type, spv::StorageClass::UniformConstant); | 
					
						
							|  |  |  |     Decorate(rescaling_uniform_constant, spv::Decoration::Location, 0u); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (profile.supported_spirv >= 0x00010400) { | 
					
						
							|  |  |  |         interfaces.push_back(rescaling_uniform_constant); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-01 22:05:11 +08:00
										 |  |  | void EmitContext::DefineRenderArea(const Info& info) { | 
					
						
							|  |  |  |     if (!info.uses_render_area) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (profile.unified_descriptor_binding) { | 
					
						
							|  |  |  |         boost::container::static_vector<Id, 1> members{}; | 
					
						
							|  |  |  |         u32 member_index{0}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         members.push_back(F32[4]); | 
					
						
							|  |  |  |         render_are_member_index = member_index++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const Id push_constant_struct{TypeStruct(std::span(members.data(), members.size()))}; | 
					
						
							|  |  |  |         Decorate(push_constant_struct, spv::Decoration::Block); | 
					
						
							|  |  |  |         Name(push_constant_struct, "RenderAreaInfo"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         MemberDecorate(push_constant_struct, render_are_member_index, spv::Decoration::Offset, 0); | 
					
						
							|  |  |  |         MemberName(push_constant_struct, render_are_member_index, "render_area"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const Id pointer_type{TypePointer(spv::StorageClass::PushConstant, push_constant_struct)}; | 
					
						
							|  |  |  |         render_area_push_constant = | 
					
						
							|  |  |  |             AddGlobalVariable(pointer_type, spv::StorageClass::PushConstant); | 
					
						
							|  |  |  |         Name(render_area_push_constant, "render_area_push_constants"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (profile.supported_spirv >= 0x00010400) { | 
					
						
							|  |  |  |             interfaces.push_back(render_area_push_constant); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-19 18:10:18 -03:00
										 |  |  | void EmitContext::DefineConstantBuffers(const Info& info, u32& binding) { | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     if (info.constant_buffer_descriptors.empty()) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-17 20:34:58 -03:00
										 |  |  |     if (!profile.support_descriptor_aliasing) { | 
					
						
							|  |  |  |         DefineConstBuffers(*this, info, &UniformDefinitions::U32x4, binding, U32[4], 'u', | 
					
						
							|  |  |  |                            sizeof(u32[4])); | 
					
						
							|  |  |  |         for (const ConstantBufferDescriptor& desc : info.constant_buffer_descriptors) { | 
					
						
							|  |  |  |             binding += desc.count; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-25 04:38:55 -03:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2021-06-17 20:34:58 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-17 14:45:38 -04:00
										 |  |  |     IR::Type types{info.used_constant_buffer_types | info.used_indirect_cbuf_types}; | 
					
						
							| 
									
										
										
										
											2021-06-17 20:34:58 -03:00
										 |  |  |     if (True(types & IR::Type::U8)) { | 
					
						
							|  |  |  |         if (profile.support_int8) { | 
					
						
							| 
									
										
										
										
											2021-05-23 03:58:11 -03:00
										 |  |  |             DefineConstBuffers(*this, info, &UniformDefinitions::U8, binding, U8, 'u', sizeof(u8)); | 
					
						
							|  |  |  |             DefineConstBuffers(*this, info, &UniformDefinitions::S8, binding, S8, 's', sizeof(s8)); | 
					
						
							| 
									
										
										
										
											2021-06-17 20:34:58 -03:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             types |= IR::Type::U32; | 
					
						
							| 
									
										
										
										
											2021-05-23 03:58:11 -03:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-17 20:34:58 -03:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (True(types & IR::Type::U16)) { | 
					
						
							|  |  |  |         if (profile.support_int16) { | 
					
						
							| 
									
										
										
										
											2021-05-23 03:58:11 -03:00
										 |  |  |             DefineConstBuffers(*this, info, &UniformDefinitions::U16, binding, U16, 'u', | 
					
						
							|  |  |  |                                sizeof(u16)); | 
					
						
							|  |  |  |             DefineConstBuffers(*this, info, &UniformDefinitions::S16, binding, S16, 's', | 
					
						
							|  |  |  |                                sizeof(s16)); | 
					
						
							| 
									
										
										
										
											2021-06-17 20:34:58 -03:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             types |= IR::Type::U32; | 
					
						
							| 
									
										
										
										
											2021-05-23 03:58:11 -03:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-03-09 17:14:57 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-17 20:34:58 -03:00
										 |  |  |     if (True(types & IR::Type::U32)) { | 
					
						
							|  |  |  |         DefineConstBuffers(*this, info, &UniformDefinitions::U32, binding, U32[1], 'u', | 
					
						
							|  |  |  |                            sizeof(u32)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (True(types & IR::Type::F32)) { | 
					
						
							|  |  |  |         DefineConstBuffers(*this, info, &UniformDefinitions::F32, binding, F32[1], 'f', | 
					
						
							|  |  |  |                            sizeof(f32)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (True(types & IR::Type::U32x2)) { | 
					
						
							|  |  |  |         DefineConstBuffers(*this, info, &UniformDefinitions::U32x2, binding, U32[2], 'u', | 
					
						
							|  |  |  |                            sizeof(u32[2])); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     binding += static_cast<u32>(info.constant_buffer_descriptors.size()); | 
					
						
							| 
									
										
										
										
											2021-03-09 17:14:57 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-17 13:30:21 -04:00
										 |  |  | void EmitContext::DefineConstantBufferIndirectFunctions(const Info& info) { | 
					
						
							|  |  |  |     if (!info.uses_cbuf_indirect) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const auto make_accessor{[&](Id buffer_type, Id UniformDefinitions::*member_ptr) { | 
					
						
							|  |  |  |         const Id func_type{TypeFunction(buffer_type, U32[1], U32[1])}; | 
					
						
							|  |  |  |         const Id func{OpFunction(buffer_type, spv::FunctionControlMask::MaskNone, func_type)}; | 
					
						
							|  |  |  |         const Id binding{OpFunctionParameter(U32[1])}; | 
					
						
							|  |  |  |         const Id offset{OpFunctionParameter(U32[1])}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         AddLabel(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const Id merge_label{OpLabel()}; | 
					
						
							|  |  |  |         const Id uniform_type{uniform_types.*member_ptr}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-01 11:17:54 -04:00
										 |  |  |         std::array<Id, Info::MAX_INDIRECT_CBUFS> buf_labels; | 
					
						
							|  |  |  |         std::array<Sirit::Literal, Info::MAX_INDIRECT_CBUFS> buf_literals; | 
					
						
							|  |  |  |         for (u32 i = 0; i < Info::MAX_INDIRECT_CBUFS; i++) { | 
					
						
							| 
									
										
										
										
											2022-03-17 13:30:21 -04:00
										 |  |  |             buf_labels[i] = OpLabel(); | 
					
						
							|  |  |  |             buf_literals[i] = Sirit::Literal{i}; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         OpSelectionMerge(merge_label, spv::SelectionControlMask::MaskNone); | 
					
						
							|  |  |  |         OpSwitch(binding, buf_labels[0], buf_literals, buf_labels); | 
					
						
							| 
									
										
										
										
											2022-04-01 11:17:54 -04:00
										 |  |  |         for (u32 i = 0; i < Info::MAX_INDIRECT_CBUFS; i++) { | 
					
						
							| 
									
										
										
										
											2022-03-17 13:30:21 -04:00
										 |  |  |             AddLabel(buf_labels[i]); | 
					
						
							|  |  |  |             const Id cbuf{cbufs[i].*member_ptr}; | 
					
						
							|  |  |  |             const Id access_chain{OpAccessChain(uniform_type, cbuf, u32_zero_value, offset)}; | 
					
						
							|  |  |  |             const Id result{OpLoad(buffer_type, access_chain)}; | 
					
						
							|  |  |  |             OpReturnValue(result); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         AddLabel(merge_label); | 
					
						
							|  |  |  |         OpUnreachable(); | 
					
						
							|  |  |  |         OpFunctionEnd(); | 
					
						
							|  |  |  |         return func; | 
					
						
							|  |  |  |     }}; | 
					
						
							| 
									
										
										
										
											2022-03-17 14:45:38 -04:00
										 |  |  |     IR::Type types{info.used_indirect_cbuf_types}; | 
					
						
							| 
									
										
										
										
											2022-04-01 11:17:54 -04:00
										 |  |  |     bool supports_aliasing = profile.support_descriptor_aliasing; | 
					
						
							|  |  |  |     if (supports_aliasing && True(types & IR::Type::U8)) { | 
					
						
							| 
									
										
										
										
											2022-03-17 13:30:21 -04:00
										 |  |  |         load_const_func_u8 = make_accessor(U8, &UniformDefinitions::U8); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-04-01 11:17:54 -04:00
										 |  |  |     if (supports_aliasing && True(types & IR::Type::U16)) { | 
					
						
							| 
									
										
										
										
											2022-03-17 13:30:21 -04:00
										 |  |  |         load_const_func_u16 = make_accessor(U16, &UniformDefinitions::U16); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-04-01 11:17:54 -04:00
										 |  |  |     if (supports_aliasing && True(types & IR::Type::F32)) { | 
					
						
							| 
									
										
										
										
											2022-03-17 13:30:21 -04:00
										 |  |  |         load_const_func_f32 = make_accessor(F32[1], &UniformDefinitions::F32); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-04-01 11:17:54 -04:00
										 |  |  |     if (supports_aliasing && True(types & IR::Type::U32)) { | 
					
						
							| 
									
										
										
										
											2022-03-17 13:30:21 -04:00
										 |  |  |         load_const_func_u32 = make_accessor(U32[1], &UniformDefinitions::U32); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-04-01 11:17:54 -04:00
										 |  |  |     if (supports_aliasing && True(types & IR::Type::U32x2)) { | 
					
						
							| 
									
										
										
										
											2022-03-17 13:30:21 -04:00
										 |  |  |         load_const_func_u32x2 = make_accessor(U32[2], &UniformDefinitions::U32x2); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-04-01 11:17:54 -04:00
										 |  |  |     if (!supports_aliasing || True(types & IR::Type::U32x4)) { | 
					
						
							| 
									
										
										
										
											2022-03-17 13:30:21 -04:00
										 |  |  |         load_const_func_u32x4 = make_accessor(U32[4], &UniformDefinitions::U32x4); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-19 18:10:18 -03:00
										 |  |  | void EmitContext::DefineStorageBuffers(const Info& info, u32& binding) { | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |     if (info.storage_buffers_descriptors.empty()) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     AddExtension("SPV_KHR_storage_buffer_storage_class"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-23 03:58:11 -03:00
										 |  |  |     const IR::Type used_types{profile.support_descriptor_aliasing ? info.used_storage_buffer_types | 
					
						
							|  |  |  |                                                                   : IR::Type::U32}; | 
					
						
							| 
									
										
										
										
											2021-06-16 22:00:31 -03:00
										 |  |  |     if (profile.support_int8 && True(used_types & IR::Type::U8)) { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         DefineSsbos(*this, storage_types.U8, &StorageDefinitions::U8, info, binding, U8, | 
					
						
							|  |  |  |                     sizeof(u8)); | 
					
						
							|  |  |  |         DefineSsbos(*this, storage_types.S8, &StorageDefinitions::S8, info, binding, S8, | 
					
						
							|  |  |  |                     sizeof(u8)); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-16 22:00:31 -03:00
										 |  |  |     if (profile.support_int16 && True(used_types & IR::Type::U16)) { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         DefineSsbos(*this, storage_types.U16, &StorageDefinitions::U16, info, binding, U16, | 
					
						
							|  |  |  |                     sizeof(u16)); | 
					
						
							|  |  |  |         DefineSsbos(*this, storage_types.S16, &StorageDefinitions::S16, info, binding, S16, | 
					
						
							|  |  |  |                     sizeof(u16)); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-23 03:58:11 -03:00
										 |  |  |     if (True(used_types & IR::Type::U32)) { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         DefineSsbos(*this, storage_types.U32, &StorageDefinitions::U32, info, binding, U32[1], | 
					
						
							|  |  |  |                     sizeof(u32)); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-23 03:58:11 -03:00
										 |  |  |     if (True(used_types & IR::Type::F32)) { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         DefineSsbos(*this, storage_types.F32, &StorageDefinitions::F32, info, binding, F32[1], | 
					
						
							|  |  |  |                     sizeof(f32)); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-23 03:58:11 -03:00
										 |  |  |     if (True(used_types & IR::Type::U64)) { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         DefineSsbos(*this, storage_types.U64, &StorageDefinitions::U64, info, binding, U64, | 
					
						
							|  |  |  |                     sizeof(u64)); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-23 03:58:11 -03:00
										 |  |  |     if (True(used_types & IR::Type::U32x2)) { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         DefineSsbos(*this, storage_types.U32x2, &StorageDefinitions::U32x2, info, binding, U32[2], | 
					
						
							|  |  |  |                     sizeof(u32[2])); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-05-23 03:58:11 -03:00
										 |  |  |     if (True(used_types & IR::Type::U32x4)) { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         DefineSsbos(*this, storage_types.U32x4, &StorageDefinitions::U32x4, info, binding, U32[4], | 
					
						
							|  |  |  |                     sizeof(u32[4])); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     for (const StorageBufferDescriptor& desc : info.storage_buffers_descriptors) { | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  |         binding += desc.count; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |     const bool needs_function{ | 
					
						
							|  |  |  |         info.uses_global_increment || info.uses_global_decrement || info.uses_atomic_f32_add || | 
					
						
							|  |  |  |         info.uses_atomic_f16x2_add || info.uses_atomic_f16x2_min || info.uses_atomic_f16x2_max || | 
					
						
							|  |  |  |         info.uses_atomic_f32x2_add || info.uses_atomic_f32x2_min || info.uses_atomic_f32x2_max}; | 
					
						
							|  |  |  |     if (needs_function) { | 
					
						
							| 
									
										
										
										
											2021-04-11 02:07:02 -04:00
										 |  |  |         AddCapability(spv::Capability::VariablePointersStorageBuffer); | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (info.uses_global_increment) { | 
					
						
							|  |  |  |         increment_cas_ssbo = CasLoop(*this, Operation::Increment, storage_types.U32.array, | 
					
						
							|  |  |  |                                      storage_types.U32.element, U32[1], U32[1], spv::Scope::Device); | 
					
						
							| 
									
										
										
										
											2021-04-11 02:07:02 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (info.uses_global_decrement) { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         decrement_cas_ssbo = CasLoop(*this, Operation::Decrement, storage_types.U32.array, | 
					
						
							|  |  |  |                                      storage_types.U32.element, U32[1], U32[1], spv::Scope::Device); | 
					
						
							| 
									
										
										
										
											2021-04-11 02:07:02 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (info.uses_atomic_f32_add) { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         f32_add_cas = CasLoop(*this, Operation::FPAdd, storage_types.U32.array, | 
					
						
							|  |  |  |                               storage_types.U32.element, F32[1], U32[1], spv::Scope::Device); | 
					
						
							| 
									
										
										
										
											2021-04-11 02:07:02 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (info.uses_atomic_f16x2_add) { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         f16x2_add_cas = CasLoop(*this, Operation::FPAdd, storage_types.U32.array, | 
					
						
							|  |  |  |                                 storage_types.U32.element, F16[2], F16[2], spv::Scope::Device); | 
					
						
							| 
									
										
										
										
											2021-04-11 02:07:02 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (info.uses_atomic_f16x2_min) { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         f16x2_min_cas = CasLoop(*this, Operation::FPMin, storage_types.U32.array, | 
					
						
							|  |  |  |                                 storage_types.U32.element, F16[2], F16[2], spv::Scope::Device); | 
					
						
							| 
									
										
										
										
											2021-04-11 02:07:02 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (info.uses_atomic_f16x2_max) { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         f16x2_max_cas = CasLoop(*this, Operation::FPMax, storage_types.U32.array, | 
					
						
							|  |  |  |                                 storage_types.U32.element, F16[2], F16[2], spv::Scope::Device); | 
					
						
							| 
									
										
										
										
											2021-04-11 02:07:02 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (info.uses_atomic_f32x2_add) { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         f32x2_add_cas = CasLoop(*this, Operation::FPAdd, storage_types.U32.array, | 
					
						
							|  |  |  |                                 storage_types.U32.element, F32[2], F32[2], spv::Scope::Device); | 
					
						
							| 
									
										
										
										
											2021-04-11 02:07:02 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (info.uses_atomic_f32x2_min) { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         f32x2_min_cas = CasLoop(*this, Operation::FPMin, storage_types.U32.array, | 
					
						
							|  |  |  |                                 storage_types.U32.element, F32[2], F32[2], spv::Scope::Device); | 
					
						
							| 
									
										
										
										
											2021-04-11 02:07:02 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (info.uses_atomic_f32x2_max) { | 
					
						
							| 
									
										
										
										
											2021-04-13 05:32:21 -03:00
										 |  |  |         f32x2_max_cas = CasLoop(*this, Operation::FPMax, storage_types.U32.array, | 
					
						
							|  |  |  |                                 storage_types.U32.element, F32[2], F32[2], spv::Scope::Device); | 
					
						
							| 
									
										
										
										
											2021-04-11 02:07:02 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-06 20:14:55 -03:00
										 |  |  | void EmitContext::DefineTextureBuffers(const Info& info, u32& binding) { | 
					
						
							|  |  |  |     if (info.texture_buffer_descriptors.empty()) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const spv::ImageFormat format{spv::ImageFormat::Unknown}; | 
					
						
							|  |  |  |     image_buffer_type = TypeImage(F32[1], spv::Dim::Buffer, 0U, false, false, 1, format); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-14 09:23:50 -04:00
										 |  |  |     const Id type{TypePointer(spv::StorageClass::UniformConstant, image_buffer_type)}; | 
					
						
							| 
									
										
										
										
											2021-04-06 20:14:55 -03:00
										 |  |  |     texture_buffers.reserve(info.texture_buffer_descriptors.size()); | 
					
						
							|  |  |  |     for (const TextureBufferDescriptor& desc : info.texture_buffer_descriptors) { | 
					
						
							|  |  |  |         if (desc.count != 1) { | 
					
						
							|  |  |  |             throw NotImplementedException("Array of texture buffers"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const Id id{AddGlobalVariable(type, spv::StorageClass::UniformConstant)}; | 
					
						
							|  |  |  |         Decorate(id, spv::Decoration::Binding, binding); | 
					
						
							|  |  |  |         Decorate(id, spv::Decoration::DescriptorSet, 0U); | 
					
						
							| 
									
										
										
										
											2021-05-23 03:52:54 -03:00
										 |  |  |         Name(id, NameOf(stage, desc, "texbuf")); | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |         texture_buffers.push_back({ | 
					
						
							|  |  |  |             .id = id, | 
					
						
							|  |  |  |             .count = desc.count, | 
					
						
							|  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2021-04-06 20:14:55 -03:00
										 |  |  |         if (profile.supported_spirv >= 0x00010400) { | 
					
						
							|  |  |  |             interfaces.push_back(id); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |         ++binding; | 
					
						
							| 
									
										
										
										
											2021-04-06 20:14:55 -03:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 21:36:36 -03:00
										 |  |  | void EmitContext::DefineImageBuffers(const Info& info, u32& binding) { | 
					
						
							|  |  |  |     image_buffers.reserve(info.image_buffer_descriptors.size()); | 
					
						
							|  |  |  |     for (const ImageBufferDescriptor& desc : info.image_buffer_descriptors) { | 
					
						
							|  |  |  |         if (desc.count != 1) { | 
					
						
							|  |  |  |             throw NotImplementedException("Array of image buffers"); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-09-21 23:12:45 -04:00
										 |  |  |         const spv::ImageFormat format{GetImageFormat(desc.format)}; | 
					
						
							| 
									
										
										
										
											2021-05-23 03:52:54 -03:00
										 |  |  |         const Id image_type{TypeImage(U32[1], spv::Dim::Buffer, false, false, false, 2, format)}; | 
					
						
							| 
									
										
										
										
											2021-04-14 21:36:36 -03:00
										 |  |  |         const Id pointer_type{TypePointer(spv::StorageClass::UniformConstant, image_type)}; | 
					
						
							|  |  |  |         const Id id{AddGlobalVariable(pointer_type, spv::StorageClass::UniformConstant)}; | 
					
						
							|  |  |  |         Decorate(id, spv::Decoration::Binding, binding); | 
					
						
							|  |  |  |         Decorate(id, spv::Decoration::DescriptorSet, 0U); | 
					
						
							| 
									
										
										
										
											2021-05-23 03:52:54 -03:00
										 |  |  |         Name(id, NameOf(stage, desc, "imgbuf")); | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |         image_buffers.push_back({ | 
					
						
							| 
									
										
										
										
											2021-04-14 21:36:36 -03:00
										 |  |  |             .id = id, | 
					
						
							|  |  |  |             .image_type = image_type, | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |             .count = desc.count, | 
					
						
							|  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2021-04-14 21:36:36 -03:00
										 |  |  |         if (profile.supported_spirv >= 0x00010400) { | 
					
						
							|  |  |  |             interfaces.push_back(id); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |         ++binding; | 
					
						
							| 
									
										
										
										
											2021-04-14 21:36:36 -03:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-01 18:57:45 -03:00
										 |  |  | void EmitContext::DefineTextures(const Info& info, u32& binding, u32& scaling_index) { | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     textures.reserve(info.texture_descriptors.size()); | 
					
						
							|  |  |  |     for (const TextureDescriptor& desc : info.texture_descriptors) { | 
					
						
							| 
									
										
										
										
											2021-03-26 18:45:38 -03:00
										 |  |  |         const Id image_type{ImageType(*this, desc)}; | 
					
						
							|  |  |  |         const Id sampled_type{TypeSampledImage(image_type)}; | 
					
						
							|  |  |  |         const Id pointer_type{TypePointer(spv::StorageClass::UniformConstant, sampled_type)}; | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |         const Id desc_type{DescType(*this, sampled_type, pointer_type, desc.count)}; | 
					
						
							|  |  |  |         const Id id{AddGlobalVariable(desc_type, spv::StorageClass::UniformConstant)}; | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |         Decorate(id, spv::Decoration::Binding, binding); | 
					
						
							|  |  |  |         Decorate(id, spv::Decoration::DescriptorSet, 0U); | 
					
						
							| 
									
										
										
										
											2021-05-23 03:52:54 -03:00
										 |  |  |         Name(id, NameOf(stage, desc, "tex")); | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |         textures.push_back({ | 
					
						
							|  |  |  |             .id = id, | 
					
						
							|  |  |  |             .sampled_type = sampled_type, | 
					
						
							|  |  |  |             .pointer_type = pointer_type, | 
					
						
							|  |  |  |             .image_type = image_type, | 
					
						
							|  |  |  |             .count = desc.count, | 
					
						
							| 
									
										
										
										
											2023-01-29 13:47:30 -05:00
										 |  |  |             .is_multisample = desc.is_multisample, | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |         }); | 
					
						
							| 
									
										
										
										
											2021-03-28 19:53:34 -03:00
										 |  |  |         if (profile.supported_spirv >= 0x00010400) { | 
					
						
							|  |  |  |             interfaces.push_back(id); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |         ++binding; | 
					
						
							| 
									
										
										
										
											2021-08-01 18:57:45 -03:00
										 |  |  |         ++scaling_index; | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-23 17:47:54 -04:00
										 |  |  |     if (info.uses_atomic_image_u32) { | 
					
						
							|  |  |  |         image_u32 = TypePointer(spv::StorageClass::Image, U32[1]); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-01 18:57:45 -03:00
										 |  |  | void EmitContext::DefineImages(const Info& info, u32& binding, u32& scaling_index) { | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  |     images.reserve(info.image_descriptors.size()); | 
					
						
							|  |  |  |     for (const ImageDescriptor& desc : info.image_descriptors) { | 
					
						
							|  |  |  |         if (desc.count != 1) { | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |             throw NotImplementedException("Array of images"); | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  |         } | 
					
						
							|  |  |  |         const Id image_type{ImageType(*this, desc)}; | 
					
						
							|  |  |  |         const Id pointer_type{TypePointer(spv::StorageClass::UniformConstant, image_type)}; | 
					
						
							|  |  |  |         const Id id{AddGlobalVariable(pointer_type, spv::StorageClass::UniformConstant)}; | 
					
						
							|  |  |  |         Decorate(id, spv::Decoration::Binding, binding); | 
					
						
							|  |  |  |         Decorate(id, spv::Decoration::DescriptorSet, 0U); | 
					
						
							| 
									
										
										
										
											2021-05-23 03:52:54 -03:00
										 |  |  |         Name(id, NameOf(stage, desc, "img")); | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |         images.push_back({ | 
					
						
							|  |  |  |             .id = id, | 
					
						
							|  |  |  |             .image_type = image_type, | 
					
						
							|  |  |  |             .count = desc.count, | 
					
						
							|  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  |         if (profile.supported_spirv >= 0x00010400) { | 
					
						
							|  |  |  |             interfaces.push_back(id); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |         ++binding; | 
					
						
							| 
									
										
										
										
											2021-08-01 18:57:45 -03:00
										 |  |  |         ++scaling_index; | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  | void EmitContext::DefineInputs(const IR::Program& program) { | 
					
						
							|  |  |  |     const Info& info{program.info}; | 
					
						
							|  |  |  |     const VaryingState loads{info.loads.mask | info.passthrough.mask}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |     if (info.uses_workgroup_id) { | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         workgroup_id = DefineInput(*this, U32[3], false, spv::BuiltIn::WorkgroupId); | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (info.uses_local_invocation_id) { | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         local_invocation_id = DefineInput(*this, U32[3], false, spv::BuiltIn::LocalInvocationId); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (info.uses_invocation_id) { | 
					
						
							|  |  |  |         invocation_id = DefineInput(*this, U32[1], false, spv::BuiltIn::InvocationId); | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-10-30 19:59:11 +08:00
										 |  |  |     if (info.uses_invocation_info && | 
					
						
							|  |  |  |         (stage == Shader::Stage::TessellationControl || stage == Shader::Stage::TessellationEval)) { | 
					
						
							|  |  |  |         patch_vertices_in = DefineInput(*this, U32[1], false, spv::BuiltIn::PatchVertices); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-16 17:22:59 -03:00
										 |  |  |     if (info.uses_sample_id) { | 
					
						
							|  |  |  |         sample_id = DefineInput(*this, U32[1], false, spv::BuiltIn::SampleId); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-11 19:16:12 -03:00
										 |  |  |     if (info.uses_is_helper_invocation) { | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         is_helper_invocation = DefineInput(*this, U1, false, spv::BuiltIn::HelperInvocation); | 
					
						
							| 
									
										
										
										
											2021-04-11 19:16:12 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-04 05:17:17 -03:00
										 |  |  |     if (info.uses_subgroup_mask) { | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         subgroup_mask_eq = DefineInput(*this, U32[4], false, spv::BuiltIn::SubgroupEqMaskKHR); | 
					
						
							|  |  |  |         subgroup_mask_lt = DefineInput(*this, U32[4], false, spv::BuiltIn::SubgroupLtMaskKHR); | 
					
						
							|  |  |  |         subgroup_mask_le = DefineInput(*this, U32[4], false, spv::BuiltIn::SubgroupLeMaskKHR); | 
					
						
							|  |  |  |         subgroup_mask_gt = DefineInput(*this, U32[4], false, spv::BuiltIn::SubgroupGtMaskKHR); | 
					
						
							|  |  |  |         subgroup_mask_ge = DefineInput(*this, U32[4], false, spv::BuiltIn::SubgroupGeMaskKHR); | 
					
						
							| 
									
										
										
										
											2021-04-04 05:17:17 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-07-29 04:00:21 +01:00
										 |  |  |     if (info.uses_fswzadd || info.uses_subgroup_invocation_id || info.uses_subgroup_shuffles || | 
					
						
							| 
									
										
										
										
											2021-04-04 05:17:17 -03:00
										 |  |  |         (profile.warp_size_potentially_larger_than_guest && | 
					
						
							|  |  |  |          (info.uses_subgroup_vote || info.uses_subgroup_mask))) { | 
					
						
							| 
									
										
										
										
											2022-12-13 17:35:46 -05:00
										 |  |  |         AddCapability(spv::Capability::GroupNonUniform); | 
					
						
							| 
									
										
										
										
											2021-03-23 20:27:17 -04:00
										 |  |  |         subgroup_local_invocation_id = | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |             DefineInput(*this, U32[1], false, spv::BuiltIn::SubgroupLocalInvocationId); | 
					
						
							| 
									
										
										
										
											2022-12-13 17:35:46 -05:00
										 |  |  |         Decorate(subgroup_local_invocation_id, spv::Decoration::Flat); | 
					
						
							| 
									
										
										
										
											2021-03-23 20:27:17 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-28 22:23:45 -04:00
										 |  |  |     if (info.uses_fswzadd) { | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |         const Id f32_one{Const(1.0f)}; | 
					
						
							|  |  |  |         const Id f32_minus_one{Const(-1.0f)}; | 
					
						
							|  |  |  |         const Id f32_zero{Const(0.0f)}; | 
					
						
							| 
									
										
										
										
											2021-03-28 22:23:45 -04:00
										 |  |  |         fswzadd_lut_a = ConstantComposite(F32[4], f32_minus_one, f32_one, f32_minus_one, f32_zero); | 
					
						
							|  |  |  |         fswzadd_lut_b = | 
					
						
							|  |  |  |             ConstantComposite(F32[4], f32_minus_one, f32_minus_one, f32_one, f32_minus_one); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |     if (loads[IR::Attribute::PrimitiveId]) { | 
					
						
							| 
									
										
										
										
											2021-04-16 01:55:06 -03:00
										 |  |  |         primitive_id = DefineInput(*this, U32[1], false, spv::BuiltIn::PrimitiveId); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-11-11 00:25:03 +08:00
										 |  |  |     if (loads[IR::Attribute::Layer]) { | 
					
						
							|  |  |  |         AddCapability(spv::Capability::Geometry); | 
					
						
							|  |  |  |         layer = DefineInput(*this, U32[1], false, spv::BuiltIn::Layer); | 
					
						
							| 
									
										
										
										
											2022-11-18 22:05:28 -05:00
										 |  |  |         Decorate(layer, spv::Decoration::Flat); | 
					
						
							| 
									
										
										
										
											2022-11-11 00:25:03 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |     if (loads.AnyComponent(IR::Attribute::PositionX)) { | 
					
						
							| 
									
										
										
										
											2022-08-02 17:41:41 +01:00
										 |  |  |         const bool is_fragment{stage == Stage::Fragment}; | 
					
						
							|  |  |  |         if (!is_fragment && profile.has_broken_spirv_position_input) { | 
					
						
							|  |  |  |             need_input_position_indirect = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             const Id input_position_struct = TypeStruct(F32[4]); | 
					
						
							|  |  |  |             input_position = DefineInput(*this, input_position_struct, true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             MemberDecorate(input_position_struct, 0, spv::Decoration::BuiltIn, | 
					
						
							|  |  |  |                            static_cast<unsigned>(spv::BuiltIn::Position)); | 
					
						
							|  |  |  |             Decorate(input_position_struct, spv::Decoration::Block); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2023-01-05 22:10:21 +00:00
										 |  |  |             const spv::BuiltIn built_in{is_fragment ? spv::BuiltIn::FragCoord | 
					
						
							|  |  |  |                                                     : spv::BuiltIn::Position}; | 
					
						
							| 
									
										
										
										
											2022-08-02 17:41:41 +01:00
										 |  |  |             input_position = DefineInput(*this, F32[4], true, built_in); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (profile.support_geometry_shader_passthrough) { | 
					
						
							|  |  |  |                 if (info.passthrough.AnyComponent(IR::Attribute::PositionX)) { | 
					
						
							|  |  |  |                     Decorate(input_position, spv::Decoration::PassthroughNV); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |     if (loads[IR::Attribute::InstanceId]) { | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |         if (profile.support_vertex_instance_id) { | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |             instance_id = DefineInput(*this, U32[1], true, spv::BuiltIn::InstanceId); | 
					
						
							| 
									
										
										
										
											2022-11-09 17:58:10 +01:00
										 |  |  |             if (loads[IR::Attribute::BaseInstance]) { | 
					
						
							|  |  |  |                 base_instance = DefineInput(*this, U32[1], true, spv::BuiltIn::BaseVertex); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |             instance_index = DefineInput(*this, U32[1], true, spv::BuiltIn::InstanceIndex); | 
					
						
							|  |  |  |             base_instance = DefineInput(*this, U32[1], true, spv::BuiltIn::BaseInstance); | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-11-09 17:58:10 +01:00
										 |  |  |     } else if (loads[IR::Attribute::BaseInstance]) { | 
					
						
							|  |  |  |         base_instance = DefineInput(*this, U32[1], true, spv::BuiltIn::BaseInstance); | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |     if (loads[IR::Attribute::VertexId]) { | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |         if (profile.support_vertex_instance_id) { | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |             vertex_id = DefineInput(*this, U32[1], true, spv::BuiltIn::VertexId); | 
					
						
							| 
									
										
										
										
											2022-11-09 17:58:10 +01:00
										 |  |  |             if (loads[IR::Attribute::BaseVertex]) { | 
					
						
							|  |  |  |                 base_vertex = DefineInput(*this, U32[1], true, spv::BuiltIn::BaseVertex); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |             vertex_index = DefineInput(*this, U32[1], true, spv::BuiltIn::VertexIndex); | 
					
						
							|  |  |  |             base_vertex = DefineInput(*this, U32[1], true, spv::BuiltIn::BaseVertex); | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-11-09 17:58:10 +01:00
										 |  |  |     } else if (loads[IR::Attribute::BaseVertex]) { | 
					
						
							|  |  |  |         base_vertex = DefineInput(*this, U32[1], true, spv::BuiltIn::BaseVertex); | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-01-03 10:01:25 -05:00
										 |  |  |     if (loads[IR::Attribute::DrawID]) { | 
					
						
							|  |  |  |         draw_index = DefineInput(*this, U32[1], true, spv::BuiltIn::DrawIndex); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |     if (loads[IR::Attribute::FrontFace]) { | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         front_face = DefineInput(*this, U1, true, spv::BuiltIn::FrontFacing); | 
					
						
							| 
									
										
										
										
											2021-03-27 02:55:37 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |     if (loads[IR::Attribute::PointSpriteS] || loads[IR::Attribute::PointSpriteT]) { | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         point_coord = DefineInput(*this, F32[2], true, spv::BuiltIn::PointCoord); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |     if (loads[IR::Attribute::TessellationEvaluationPointU] || | 
					
						
							|  |  |  |         loads[IR::Attribute::TessellationEvaluationPointV]) { | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         tess_coord = DefineInput(*this, F32[3], false, spv::BuiltIn::TessCoord); | 
					
						
							| 
									
										
										
										
											2021-03-29 20:05:38 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |     for (size_t index = 0; index < IR::NUM_GENERICS; ++index) { | 
					
						
							|  |  |  |         const AttributeType input_type{runtime_info.generic_input_types[index]}; | 
					
						
							| 
									
										
										
										
											2021-09-07 13:21:17 +08:00
										 |  |  |         if (!runtime_info.previous_stage_stores.Generic(index)) { | 
					
						
							| 
									
										
										
										
											2021-03-27 04:59:58 -03:00
										 |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-07 13:21:17 +08:00
										 |  |  |         if (!loads.Generic(index)) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (input_type == AttributeType::Disabled) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-03-27 04:59:58 -03:00
										 |  |  |         const Id type{GetAttributeType(*this, input_type)}; | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         const Id id{DefineInput(*this, type, true)}; | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |         Decorate(id, spv::Decoration::Location, static_cast<u32>(index)); | 
					
						
							|  |  |  |         Name(id, fmt::format("in_attr{}", index)); | 
					
						
							| 
									
										
										
										
											2023-02-18 18:23:36 +00:00
										 |  |  |         input_generics[index] = GetAttributeInfo(*this, input_type, id); | 
					
						
							| 
									
										
										
										
											2021-03-27 04:59:58 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |         if (info.passthrough.Generic(index) && profile.support_geometry_shader_passthrough) { | 
					
						
							|  |  |  |             Decorate(id, spv::Decoration::PassthroughNV); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-03-27 04:59:58 -03:00
										 |  |  |         if (stage != Stage::Fragment) { | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |         switch (info.interpolation[index]) { | 
					
						
							| 
									
										
										
										
											2021-03-27 04:59:58 -03:00
										 |  |  |         case Interpolation::Smooth: | 
					
						
							|  |  |  |             // Default
 | 
					
						
							|  |  |  |             // Decorate(id, spv::Decoration::Smooth);
 | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case Interpolation::NoPerspective: | 
					
						
							|  |  |  |             Decorate(id, spv::Decoration::NoPerspective); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         case Interpolation::Flat: | 
					
						
							|  |  |  |             Decorate(id, spv::Decoration::Flat); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |     if (stage == Stage::TessellationEval) { | 
					
						
							|  |  |  |         for (size_t index = 0; index < info.uses_patches.size(); ++index) { | 
					
						
							|  |  |  |             if (!info.uses_patches[index]) { | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             const Id id{DefineInput(*this, F32[4], false)}; | 
					
						
							|  |  |  |             Decorate(id, spv::Decoration::Patch); | 
					
						
							|  |  |  |             Decorate(id, spv::Decoration::Location, static_cast<u32>(index)); | 
					
						
							|  |  |  |             patches[index] = id; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  | void EmitContext::DefineOutputs(const IR::Program& program) { | 
					
						
							|  |  |  |     const Info& info{program.info}; | 
					
						
							|  |  |  |     const std::optional<u32> invocations{program.invocations}; | 
					
						
							| 
									
										
										
										
											2022-07-29 04:00:21 +01:00
										 |  |  |     if (runtime_info.convert_depth_mode || info.stores.AnyComponent(IR::Attribute::PositionX) || | 
					
						
							|  |  |  |         stage == Stage::VertexB) { | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         output_position = DefineOutput(*this, F32[4], invocations, spv::BuiltIn::Position); | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |     if (info.stores[IR::Attribute::PointSize] || runtime_info.fixed_state_point_size) { | 
					
						
							| 
									
										
										
										
											2021-03-26 18:52:06 -04:00
										 |  |  |         if (stage == Stage::Fragment) { | 
					
						
							| 
									
										
										
										
											2021-04-12 19:41:22 -03:00
										 |  |  |             throw NotImplementedException("Storing PointSize in fragment stage"); | 
					
						
							| 
									
										
										
										
											2021-03-26 18:52:06 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         output_point_size = DefineOutput(*this, F32[1], invocations, spv::BuiltIn::PointSize); | 
					
						
							| 
									
										
										
										
											2021-03-26 18:52:06 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |     if (info.stores.ClipDistances()) { | 
					
						
							| 
									
										
										
										
											2021-03-30 21:52:06 +02:00
										 |  |  |         if (stage == Stage::Fragment) { | 
					
						
							| 
									
										
										
										
											2021-04-12 19:41:22 -03:00
										 |  |  |             throw NotImplementedException("Storing ClipDistance in fragment stage"); | 
					
						
							| 
									
										
										
										
											2021-03-30 21:52:06 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-12-18 22:15:44 -05:00
										 |  |  |         const Id type{TypeArray(F32[1], Const(std::min(8U, profile.max_user_clip_distances)))}; | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         clip_distances = DefineOutput(*this, type, invocations, spv::BuiltIn::ClipDistance); | 
					
						
							| 
									
										
										
										
											2021-03-30 21:52:06 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |     if (info.stores[IR::Attribute::Layer] && | 
					
						
							| 
									
										
										
										
											2021-04-14 18:09:18 -03:00
										 |  |  |         (profile.support_viewport_index_layer_non_geometry || stage == Stage::Geometry)) { | 
					
						
							|  |  |  |         if (stage == Stage::Fragment) { | 
					
						
							|  |  |  |             throw NotImplementedException("Storing Layer in fragment stage"); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         layer = DefineOutput(*this, U32[1], invocations, spv::BuiltIn::Layer); | 
					
						
							| 
									
										
										
										
											2021-04-14 18:09:18 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |     if (info.stores[IR::Attribute::ViewportIndex] && | 
					
						
							| 
									
										
										
										
											2021-04-14 18:09:18 -03:00
										 |  |  |         (profile.support_viewport_index_layer_non_geometry || stage == Stage::Geometry)) { | 
					
						
							| 
									
										
										
										
											2021-04-01 08:34:45 +02:00
										 |  |  |         if (stage == Stage::Fragment) { | 
					
						
							| 
									
										
										
										
											2021-04-12 19:41:22 -03:00
										 |  |  |             throw NotImplementedException("Storing ViewportIndex in fragment stage"); | 
					
						
							| 
									
										
										
										
											2021-04-01 08:34:45 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         viewport_index = DefineOutput(*this, U32[1], invocations, spv::BuiltIn::ViewportIndex); | 
					
						
							| 
									
										
										
										
											2021-04-01 08:34:45 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-24 02:41:09 -03:00
										 |  |  |     if (info.stores[IR::Attribute::ViewportMask] && profile.support_viewport_mask) { | 
					
						
							| 
									
										
										
										
											2021-04-19 16:23:52 -03:00
										 |  |  |         viewport_mask = DefineOutput(*this, TypeArray(U32[1], Const(1u)), std::nullopt, | 
					
						
							|  |  |  |                                      spv::BuiltIn::ViewportMaskNV); | 
					
						
							| 
									
										
										
										
											2021-04-16 16:31:15 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-06 10:46:03 +08:00
										 |  |  |     for (size_t index = 0; index < IR::NUM_GENERICS; ++index) { | 
					
						
							|  |  |  |         if (info.stores.Generic(index)) { | 
					
						
							|  |  |  |             DefineGenericOutput(*this, index, invocations); | 
					
						
							| 
									
										
										
										
											2021-09-04 00:12:06 +08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-01 00:07:25 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |     switch (stage) { | 
					
						
							|  |  |  |     case Stage::TessellationControl: | 
					
						
							|  |  |  |         if (info.stores_tess_level_outer) { | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |             const Id type{TypeArray(F32[1], Const(4U))}; | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |             output_tess_level_outer = | 
					
						
							|  |  |  |                 DefineOutput(*this, type, std::nullopt, spv::BuiltIn::TessLevelOuter); | 
					
						
							|  |  |  |             Decorate(output_tess_level_outer, spv::Decoration::Patch); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (info.stores_tess_level_inner) { | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |             const Id type{TypeArray(F32[1], Const(2U))}; | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |             output_tess_level_inner = | 
					
						
							|  |  |  |                 DefineOutput(*this, type, std::nullopt, spv::BuiltIn::TessLevelInner); | 
					
						
							|  |  |  |             Decorate(output_tess_level_inner, spv::Decoration::Patch); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         for (size_t index = 0; index < info.uses_patches.size(); ++index) { | 
					
						
							|  |  |  |             if (!info.uses_patches[index]) { | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             const Id id{DefineOutput(*this, F32[4], std::nullopt)}; | 
					
						
							|  |  |  |             Decorate(id, spv::Decoration::Patch); | 
					
						
							|  |  |  |             Decorate(id, spv::Decoration::Location, static_cast<u32>(index)); | 
					
						
							|  |  |  |             patches[index] = id; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case Stage::Fragment: | 
					
						
							| 
									
										
										
										
											2021-03-26 16:46:07 -03:00
										 |  |  |         for (u32 index = 0; index < 8; ++index) { | 
					
						
							| 
									
										
										
										
											2021-05-30 03:40:19 -03:00
										 |  |  |             if (!info.stores_frag_color[index] && !profile.need_declared_frag_colors) { | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |             frag_color[index] = DefineOutput(*this, F32[4], std::nullopt); | 
					
						
							| 
									
										
										
										
											2021-03-26 16:46:07 -03:00
										 |  |  |             Decorate(frag_color[index], spv::Decoration::Location, index); | 
					
						
							|  |  |  |             Name(frag_color[index], fmt::format("frag_color{}", index)); | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-03-26 16:46:07 -03:00
										 |  |  |         if (info.stores_frag_depth) { | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |             frag_depth = DefineOutput(*this, F32[1], std::nullopt); | 
					
						
							| 
									
										
										
										
											2021-03-26 16:46:07 -03:00
										 |  |  |             Decorate(frag_depth, spv::Decoration::BuiltIn, spv::BuiltIn::FragDepth); | 
					
						
							| 
									
										
										
										
											2021-04-16 18:47:26 -03:00
										 |  |  |         } | 
					
						
							|  |  |  |         if (info.stores_sample_mask) { | 
					
						
							| 
									
										
										
										
											2023-04-30 18:37:37 +01:00
										 |  |  |             const Id array_type{TypeArray(U32[1], Const(1U))}; | 
					
						
							|  |  |  |             sample_mask = DefineOutput(*this, array_type, std::nullopt); | 
					
						
							| 
									
										
										
										
											2021-04-16 18:47:26 -03:00
										 |  |  |             Decorate(sample_mask, spv::Decoration::BuiltIn, spv::BuiltIn::SampleMask); | 
					
						
							| 
									
										
										
										
											2021-03-25 20:27:09 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-15 22:46:11 -03:00
										 |  |  |         break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2021-03-20 19:11:56 -03:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-16 04:10:22 -03:00
										 |  |  | } // namespace Shader::Backend::SPIRV
 |