| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  | // Copyright 2021 yuzu Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <boost/container/static_vector.hpp>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "shader_recompiler/backend/spirv/emit_spirv.h"
 | 
					
						
							| 
									
										
										
										
											2021-05-03 20:53:00 -03:00
										 |  |  | #include "shader_recompiler/backend/spirv/emit_spirv_instructions.h"
 | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  | #include "shader_recompiler/frontend/ir/modifiers.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Shader::Backend::SPIRV { | 
					
						
							|  |  |  | namespace { | 
					
						
							|  |  |  | class ImageOperands { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     explicit ImageOperands(EmitContext& ctx, bool has_bias, bool has_lod, bool has_lod_clamp, | 
					
						
							| 
									
										
										
										
											2021-04-17 02:59:54 -03:00
										 |  |  |                            Id lod, const IR::Value& offset) { | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |         if (has_bias) { | 
					
						
							|  |  |  |             const Id bias{has_lod_clamp ? ctx.OpCompositeExtract(ctx.F32[1], lod, 0) : lod}; | 
					
						
							|  |  |  |             Add(spv::ImageOperandsMask::Bias, bias); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (has_lod) { | 
					
						
							|  |  |  |             const Id lod_value{has_lod_clamp ? ctx.OpCompositeExtract(ctx.F32[1], lod, 0) : lod}; | 
					
						
							|  |  |  |             Add(spv::ImageOperandsMask::Lod, lod_value); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-17 02:59:54 -03:00
										 |  |  |         AddOffset(ctx, offset); | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |         if (has_lod_clamp) { | 
					
						
							|  |  |  |             const Id lod_clamp{has_bias ? ctx.OpCompositeExtract(ctx.F32[1], lod, 1) : lod}; | 
					
						
							|  |  |  |             Add(spv::ImageOperandsMask::MinLod, lod_clamp); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 16:46:07 -03:00
										 |  |  |     explicit ImageOperands(EmitContext& ctx, const IR::Value& offset, const IR::Value& offset2) { | 
					
						
							|  |  |  |         if (offset2.IsEmpty()) { | 
					
						
							|  |  |  |             if (offset.IsEmpty()) { | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             Add(spv::ImageOperandsMask::Offset, ctx.Def(offset)); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const std::array values{offset.InstRecursive(), offset2.InstRecursive()}; | 
					
						
							|  |  |  |         if (!values[0]->AreAllArgsImmediates() || !values[1]->AreAllArgsImmediates()) { | 
					
						
							| 
									
										
										
										
											2021-06-14 02:27:49 -03:00
										 |  |  |             LOG_WARNING(Shader_SPIRV, "Not all arguments in PTP are immediate, ignoring"); | 
					
						
							| 
									
										
										
										
											2021-04-01 08:25:55 +02:00
										 |  |  |             return; | 
					
						
							| 
									
										
										
										
											2021-03-24 23:41:55 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-05 22:25:22 -04:00
										 |  |  |         const IR::Opcode opcode{values[0]->GetOpcode()}; | 
					
						
							|  |  |  |         if (opcode != values[1]->GetOpcode() || opcode != IR::Opcode::CompositeConstructU32x4) { | 
					
						
							| 
									
										
										
										
											2021-03-26 16:46:07 -03:00
										 |  |  |             throw LogicError("Invalid PTP arguments"); | 
					
						
							| 
									
										
										
										
											2021-03-26 16:02:04 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-04-18 20:47:31 -04:00
										 |  |  |         auto read{[&](unsigned int a, unsigned int b) { return values[a]->Arg(b).U32(); }}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const Id offsets{ctx.ConstantComposite( | 
					
						
							|  |  |  |             ctx.TypeArray(ctx.U32[2], ctx.Const(4U)), ctx.Const(read(0, 0), read(0, 1)), | 
					
						
							|  |  |  |             ctx.Const(read(0, 2), read(0, 3)), ctx.Const(read(1, 0), read(1, 1)), | 
					
						
							|  |  |  |             ctx.Const(read(1, 2), read(1, 3)))}; | 
					
						
							| 
									
										
										
										
											2021-03-26 16:46:07 -03:00
										 |  |  |         Add(spv::ImageOperandsMask::ConstOffsets, offsets); | 
					
						
							| 
									
										
										
										
											2021-03-24 23:41:55 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 16:46:07 -03:00
										 |  |  |     explicit ImageOperands(Id offset, Id lod, Id ms) { | 
					
						
							| 
									
										
										
										
											2021-03-26 19:24:50 +01:00
										 |  |  |         if (Sirit::ValidId(lod)) { | 
					
						
							|  |  |  |             Add(spv::ImageOperandsMask::Lod, lod); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (Sirit::ValidId(offset)) { | 
					
						
							|  |  |  |             Add(spv::ImageOperandsMask::Offset, offset); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (Sirit::ValidId(ms)) { | 
					
						
							|  |  |  |             Add(spv::ImageOperandsMask::Sample, ms); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-29 02:00:43 +02:00
										 |  |  |     explicit ImageOperands(EmitContext& ctx, bool has_lod_clamp, Id derivates, u32 num_derivates, | 
					
						
							|  |  |  |                            Id offset, Id lod_clamp) { | 
					
						
							| 
									
										
										
										
											2021-03-30 19:20:59 +02:00
										 |  |  |         if (!Sirit::ValidId(derivates)) { | 
					
						
							| 
									
										
										
										
											2021-03-29 02:00:43 +02:00
										 |  |  |             throw LogicError("Derivates must be present"); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-03-30 08:41:21 +02:00
										 |  |  |         boost::container::static_vector<Id, 3> deriv_x_accum; | 
					
						
							|  |  |  |         boost::container::static_vector<Id, 3> deriv_y_accum; | 
					
						
							| 
									
										
										
										
											2021-04-06 05:52:41 -03:00
										 |  |  |         for (u32 i = 0; i < num_derivates; ++i) { | 
					
						
							| 
									
										
										
										
											2021-03-30 08:41:21 +02:00
										 |  |  |             deriv_x_accum.push_back(ctx.OpCompositeExtract(ctx.F32[1], derivates, i * 2)); | 
					
						
							|  |  |  |             deriv_y_accum.push_back(ctx.OpCompositeExtract(ctx.F32[1], derivates, i * 2 + 1)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const Id derivates_X{ctx.OpCompositeConstruct( | 
					
						
							|  |  |  |             ctx.F32[num_derivates], std::span{deriv_x_accum.data(), deriv_x_accum.size()})}; | 
					
						
							|  |  |  |         const Id derivates_Y{ctx.OpCompositeConstruct( | 
					
						
							|  |  |  |             ctx.F32[num_derivates], std::span{deriv_y_accum.data(), deriv_y_accum.size()})}; | 
					
						
							|  |  |  |         Add(spv::ImageOperandsMask::Grad, derivates_X, derivates_Y); | 
					
						
							| 
									
										
										
										
											2021-03-29 02:00:43 +02:00
										 |  |  |         if (Sirit::ValidId(offset)) { | 
					
						
							|  |  |  |             Add(spv::ImageOperandsMask::Offset, offset); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (has_lod_clamp) { | 
					
						
							|  |  |  |             Add(spv::ImageOperandsMask::MinLod, lod_clamp); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-17 02:59:54 -03:00
										 |  |  |     std::span<const Id> Span() const noexcept { | 
					
						
							|  |  |  |         return std::span{operands.data(), operands.size()}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-23 04:15:36 -03:00
										 |  |  |     std::optional<spv::ImageOperandsMask> MaskOptional() const noexcept { | 
					
						
							|  |  |  |         return mask != spv::ImageOperandsMask{} ? std::make_optional(mask) : std::nullopt; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-17 02:59:54 -03:00
										 |  |  |     spv::ImageOperandsMask Mask() const noexcept { | 
					
						
							|  |  |  |         return mask; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     void AddOffset(EmitContext& ctx, const IR::Value& offset) { | 
					
						
							|  |  |  |         if (offset.IsEmpty()) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (offset.IsImmediate()) { | 
					
						
							| 
									
										
										
										
											2021-07-11 22:10:38 -04:00
										 |  |  |             Add(spv::ImageOperandsMask::ConstOffset, ctx.SConst(static_cast<s32>(offset.U32()))); | 
					
						
							| 
									
										
										
										
											2021-04-17 02:59:54 -03:00
										 |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         IR::Inst* const inst{offset.InstRecursive()}; | 
					
						
							|  |  |  |         if (inst->AreAllArgsImmediates()) { | 
					
						
							|  |  |  |             switch (inst->GetOpcode()) { | 
					
						
							|  |  |  |             case IR::Opcode::CompositeConstructU32x2: | 
					
						
							|  |  |  |                 Add(spv::ImageOperandsMask::ConstOffset, | 
					
						
							| 
									
										
										
										
											2021-07-11 22:10:38 -04:00
										 |  |  |                     ctx.SConst(static_cast<s32>(inst->Arg(0).U32()), | 
					
						
							|  |  |  |                                static_cast<s32>(inst->Arg(1).U32()))); | 
					
						
							| 
									
										
										
										
											2021-04-17 02:59:54 -03:00
										 |  |  |                 return; | 
					
						
							|  |  |  |             case IR::Opcode::CompositeConstructU32x3: | 
					
						
							|  |  |  |                 Add(spv::ImageOperandsMask::ConstOffset, | 
					
						
							| 
									
										
										
										
											2021-07-11 22:10:38 -04:00
										 |  |  |                     ctx.SConst(static_cast<s32>(inst->Arg(0).U32()), | 
					
						
							|  |  |  |                                static_cast<s32>(inst->Arg(1).U32()), | 
					
						
							|  |  |  |                                static_cast<s32>(inst->Arg(2).U32()))); | 
					
						
							| 
									
										
										
										
											2021-04-17 02:59:54 -03:00
										 |  |  |                 return; | 
					
						
							|  |  |  |             case IR::Opcode::CompositeConstructU32x4: | 
					
						
							|  |  |  |                 Add(spv::ImageOperandsMask::ConstOffset, | 
					
						
							| 
									
										
										
										
											2021-07-11 22:10:38 -04:00
										 |  |  |                     ctx.SConst(static_cast<s32>(inst->Arg(0).U32()), | 
					
						
							|  |  |  |                                static_cast<s32>(inst->Arg(1).U32()), | 
					
						
							|  |  |  |                                static_cast<s32>(inst->Arg(2).U32()), | 
					
						
							|  |  |  |                                static_cast<s32>(inst->Arg(3).U32()))); | 
					
						
							| 
									
										
										
										
											2021-04-17 02:59:54 -03:00
										 |  |  |                 return; | 
					
						
							|  |  |  |             default: | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         Add(spv::ImageOperandsMask::Offset, ctx.Def(offset)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     void Add(spv::ImageOperandsMask new_mask, Id value) { | 
					
						
							|  |  |  |         mask = static_cast<spv::ImageOperandsMask>(static_cast<unsigned>(mask) | | 
					
						
							|  |  |  |                                                    static_cast<unsigned>(new_mask)); | 
					
						
							|  |  |  |         operands.push_back(value); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-30 08:41:21 +02:00
										 |  |  |     void Add(spv::ImageOperandsMask new_mask, Id value_1, Id value_2) { | 
					
						
							| 
									
										
										
										
											2021-03-29 02:00:43 +02:00
										 |  |  |         mask = static_cast<spv::ImageOperandsMask>(static_cast<unsigned>(mask) | | 
					
						
							|  |  |  |                                                    static_cast<unsigned>(new_mask)); | 
					
						
							| 
									
										
										
										
											2021-03-30 08:41:21 +02:00
										 |  |  |         operands.push_back(value_1); | 
					
						
							| 
									
										
										
										
											2021-03-29 02:00:43 +02:00
										 |  |  |         operands.push_back(value_2); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     boost::container::static_vector<Id, 4> operands; | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     spv::ImageOperandsMask mask{}; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  | Id Texture(EmitContext& ctx, IR::TextureInstInfo info, [[maybe_unused]] const IR::Value& index) { | 
					
						
							|  |  |  |     const TextureDefinition& def{ctx.textures.at(info.descriptor_index)}; | 
					
						
							|  |  |  |     if (def.count > 1) { | 
					
						
							|  |  |  |         const Id pointer{ctx.OpAccessChain(def.pointer_type, def.id, ctx.Def(index))}; | 
					
						
							|  |  |  |         return ctx.OpLoad(def.sampled_type, pointer); | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2021-03-26 18:45:38 -03:00
										 |  |  |         return ctx.OpLoad(def.sampled_type, def.id); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-29 02:49:40 -03:00
										 |  |  | Id TextureImage(EmitContext& ctx, IR::TextureInstInfo info, const IR::Value& index) { | 
					
						
							|  |  |  |     if (!index.IsImmediate() || index.U32() != 0) { | 
					
						
							|  |  |  |         throw NotImplementedException("Indirect image indexing"); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-06 02:56:15 -03:00
										 |  |  |     if (info.type == TextureType::Buffer) { | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |         const TextureBufferDefinition& def{ctx.texture_buffers.at(info.descriptor_index)}; | 
					
						
							|  |  |  |         if (def.count > 1) { | 
					
						
							|  |  |  |             throw NotImplementedException("Indirect texture sample"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const Id sampler_id{def.id}; | 
					
						
							| 
									
										
										
										
											2021-04-06 02:56:15 -03:00
										 |  |  |         const Id id{ctx.OpLoad(ctx.sampled_texture_buffer_type, sampler_id)}; | 
					
						
							|  |  |  |         return ctx.OpImage(ctx.image_buffer_type, id); | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |         const TextureDefinition& def{ctx.textures.at(info.descriptor_index)}; | 
					
						
							|  |  |  |         if (def.count > 1) { | 
					
						
							|  |  |  |             throw NotImplementedException("Indirect texture sample"); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-03-26 18:45:38 -03:00
										 |  |  |         return ctx.OpImage(def.image_type, ctx.OpLoad(def.sampled_type, def.id)); | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  | Id Image(EmitContext& ctx, const IR::Value& index, IR::TextureInstInfo info) { | 
					
						
							| 
									
										
										
										
											2021-05-29 02:49:40 -03:00
										 |  |  |     if (!index.IsImmediate() || index.U32() != 0) { | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  |         throw NotImplementedException("Indirect image indexing"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (info.type == TextureType::Buffer) { | 
					
						
							| 
									
										
										
										
											2021-05-29 02:49:40 -03:00
										 |  |  |         const ImageBufferDefinition def{ctx.image_buffers.at(info.descriptor_index)}; | 
					
						
							| 
									
										
										
										
											2021-04-14 21:36:36 -03:00
										 |  |  |         return ctx.OpLoad(def.image_type, def.id); | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2021-05-29 02:49:40 -03:00
										 |  |  |         const ImageDefinition def{ctx.images.at(info.descriptor_index)}; | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  |         return ctx.OpLoad(def.image_type, def.id); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-15 04:54:43 -03:00
										 |  |  | Id Decorate(EmitContext& ctx, IR::Inst* inst, Id sample) { | 
					
						
							|  |  |  |     const auto info{inst->Flags<IR::TextureInstInfo>()}; | 
					
						
							|  |  |  |     if (info.relaxed_precision != 0) { | 
					
						
							|  |  |  |         ctx.Decorate(sample, spv::Decoration::RelaxedPrecision); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return sample; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  | template <typename MethodPtrType, typename... Args> | 
					
						
							|  |  |  | Id Emit(MethodPtrType sparse_ptr, MethodPtrType non_sparse_ptr, EmitContext& ctx, IR::Inst* inst, | 
					
						
							|  |  |  |         Id result_type, Args&&... args) { | 
					
						
							|  |  |  |     IR::Inst* const sparse{inst->GetAssociatedPseudoOperation(IR::Opcode::GetSparseFromOp)}; | 
					
						
							|  |  |  |     if (!sparse) { | 
					
						
							| 
									
										
										
										
											2021-03-15 04:54:43 -03:00
										 |  |  |         return Decorate(ctx, inst, (ctx.*non_sparse_ptr)(result_type, std::forward<Args>(args)...)); | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     } | 
					
						
							|  |  |  |     const Id struct_type{ctx.TypeStruct(ctx.U32[1], result_type)}; | 
					
						
							|  |  |  |     const Id sample{(ctx.*sparse_ptr)(struct_type, std::forward<Args>(args)...)}; | 
					
						
							|  |  |  |     const Id resident_code{ctx.OpCompositeExtract(ctx.U32[1], sample, 0U)}; | 
					
						
							|  |  |  |     sparse->SetDefinition(ctx.OpImageSparseTexelsResident(ctx.U1, resident_code)); | 
					
						
							|  |  |  |     sparse->Invalidate(); | 
					
						
							| 
									
										
										
										
											2021-03-15 04:54:43 -03:00
										 |  |  |     Decorate(ctx, inst, sample); | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     return ctx.OpCompositeExtract(result_type, sample, 1U); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-08-01 18:57:45 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | Id IsScaled(EmitContext& ctx, const IR::Value& index, Id member_index, u32 base_index) { | 
					
						
							|  |  |  |     const Id push_constant_u32{ctx.TypePointer(spv::StorageClass::PushConstant, ctx.U32[1])}; | 
					
						
							|  |  |  |     Id bit{}; | 
					
						
							|  |  |  |     if (index.IsImmediate()) { | 
					
						
							|  |  |  |         // Use BitwiseAnd instead of BitfieldExtract for better codegen on Nvidia OpenGL.
 | 
					
						
							|  |  |  |         // LOP32I.NZ is used to set the predicate rather than BFE+ISETP.
 | 
					
						
							|  |  |  |         const u32 index_value{index.U32() + base_index}; | 
					
						
							|  |  |  |         const Id word_index{ctx.Const(index_value / 32)}; | 
					
						
							|  |  |  |         const Id bit_index_mask{ctx.Const(1u << (index_value % 32))}; | 
					
						
							|  |  |  |         const Id pointer{ctx.OpAccessChain(push_constant_u32, ctx.rescaling_push_constants, | 
					
						
							|  |  |  |                                            member_index, word_index)}; | 
					
						
							|  |  |  |         const Id word{ctx.OpLoad(ctx.U32[1], pointer)}; | 
					
						
							|  |  |  |         bit = ctx.OpBitwiseAnd(ctx.U32[1], word, bit_index_mask); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         Id index_value{ctx.Def(index)}; | 
					
						
							|  |  |  |         if (base_index != 0) { | 
					
						
							|  |  |  |             index_value = ctx.OpIAdd(ctx.U32[1], index_value, ctx.Const(base_index)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const Id bit_index{ctx.OpBitwiseAnd(ctx.U32[1], index_value, ctx.Const(31u))}; | 
					
						
							|  |  |  |         bit = ctx.OpBitFieldUExtract(ctx.U32[1], index_value, bit_index, ctx.Const(1u)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return ctx.OpINotEqual(ctx.U1, bit, ctx.u32_zero_value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id BitTest(EmitContext& ctx, Id mask, Id bit) { | 
					
						
							|  |  |  |     const Id shifted{ctx.OpShiftRightLogical(ctx.U32[1], mask, bit)}; | 
					
						
							|  |  |  |     const Id bit_value{ctx.OpBitwiseAnd(ctx.U32[1], shifted, ctx.Const(1u))}; | 
					
						
							|  |  |  |     return ctx.OpINotEqual(ctx.U1, bit_value, ctx.u32_zero_value); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  | } // Anonymous namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitBindlessImageSampleImplicitLod(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitBindlessImageSampleExplicitLod(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitBindlessImageSampleDrefImplicitLod(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitBindlessImageSampleDrefExplicitLod(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-24 23:41:55 +01:00
										 |  |  | Id EmitBindlessImageGather(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitBindlessImageGatherDref(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 19:24:50 +01:00
										 |  |  | Id EmitBindlessImageFetch(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 18:45:38 -03:00
										 |  |  | Id EmitBindlessImageQueryDimensions(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-28 19:47:52 +02:00
										 |  |  | Id EmitBindlessImageQueryLod(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-29 02:00:43 +02:00
										 |  |  | Id EmitBindlessImageGradient(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  | Id EmitBindlessImageRead(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitBindlessImageWrite(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  | Id EmitBoundImageSampleImplicitLod(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitBoundImageSampleExplicitLod(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitBoundImageSampleDrefImplicitLod(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitBoundImageSampleDrefExplicitLod(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-24 23:41:55 +01:00
										 |  |  | Id EmitBoundImageGather(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitBoundImageGatherDref(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 19:24:50 +01:00
										 |  |  | Id EmitBoundImageFetch(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 18:45:38 -03:00
										 |  |  | Id EmitBoundImageQueryDimensions(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-28 19:47:52 +02:00
										 |  |  | Id EmitBoundImageQueryLod(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-29 02:00:43 +02:00
										 |  |  | Id EmitBoundImageGradient(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  | Id EmitBoundImageRead(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitBoundImageWrite(EmitContext&) { | 
					
						
							|  |  |  |     throw LogicError("Unreachable instruction"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  | Id EmitImageSampleImplicitLod(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id coords, | 
					
						
							| 
									
										
										
										
											2021-04-17 02:59:54 -03:00
										 |  |  |                               Id bias_lc, const IR::Value& offset) { | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     const auto info{inst->Flags<IR::TextureInstInfo>()}; | 
					
						
							| 
									
										
										
										
											2021-04-17 03:07:31 -03:00
										 |  |  |     if (ctx.stage == Stage::Fragment) { | 
					
						
							|  |  |  |         const ImageOperands operands(ctx, info.has_bias != 0, false, info.has_lod_clamp != 0, | 
					
						
							|  |  |  |                                      bias_lc, offset); | 
					
						
							|  |  |  |         return Emit(&EmitContext::OpImageSparseSampleImplicitLod, | 
					
						
							|  |  |  |                     &EmitContext::OpImageSampleImplicitLod, ctx, inst, ctx.F32[4], | 
					
						
							| 
									
										
										
										
											2021-05-23 04:15:36 -03:00
										 |  |  |                     Texture(ctx, info, index), coords, operands.MaskOptional(), operands.Span()); | 
					
						
							| 
									
										
										
										
											2021-04-17 03:07:31 -03:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         // We can't use implicit lods on non-fragment stages on SPIR-V. Maxwell hardware behaves as
 | 
					
						
							|  |  |  |         // if the lod was explicitly zero.  This may change on Turing with implicit compute
 | 
					
						
							|  |  |  |         // derivatives
 | 
					
						
							| 
									
										
										
										
											2021-04-17 03:19:54 -03:00
										 |  |  |         const Id lod{ctx.Const(0.0f)}; | 
					
						
							| 
									
										
										
										
											2021-04-17 03:07:31 -03:00
										 |  |  |         const ImageOperands operands(ctx, false, true, info.has_lod_clamp != 0, lod, offset); | 
					
						
							|  |  |  |         return Emit(&EmitContext::OpImageSparseSampleExplicitLod, | 
					
						
							|  |  |  |                     &EmitContext::OpImageSampleExplicitLod, ctx, inst, ctx.F32[4], | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |                     Texture(ctx, info, index), coords, operands.Mask(), operands.Span()); | 
					
						
							| 
									
										
										
										
											2021-04-17 03:07:31 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitImageSampleExplicitLod(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id coords, | 
					
						
							| 
									
										
										
										
											2021-05-17 02:52:01 -03:00
										 |  |  |                               Id lod, const IR::Value& offset) { | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     const auto info{inst->Flags<IR::TextureInstInfo>()}; | 
					
						
							| 
									
										
										
										
											2021-05-17 02:52:01 -03:00
										 |  |  |     const ImageOperands operands(ctx, false, true, false, lod, offset); | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     return Emit(&EmitContext::OpImageSparseSampleExplicitLod, | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |                 &EmitContext::OpImageSampleExplicitLod, ctx, inst, ctx.F32[4], | 
					
						
							|  |  |  |                 Texture(ctx, info, index), coords, operands.Mask(), operands.Span()); | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitImageSampleDrefImplicitLod(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, | 
					
						
							| 
									
										
										
										
											2021-04-17 02:59:54 -03:00
										 |  |  |                                   Id coords, Id dref, Id bias_lc, const IR::Value& offset) { | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     const auto info{inst->Flags<IR::TextureInstInfo>()}; | 
					
						
							| 
									
										
										
										
											2021-09-24 20:38:43 -04:00
										 |  |  |     if (ctx.stage == Stage::Fragment) { | 
					
						
							|  |  |  |         const ImageOperands operands(ctx, info.has_bias != 0, false, info.has_lod_clamp != 0, | 
					
						
							|  |  |  |                                      bias_lc, offset); | 
					
						
							|  |  |  |         return Emit(&EmitContext::OpImageSparseSampleDrefImplicitLod, | 
					
						
							|  |  |  |                     &EmitContext::OpImageSampleDrefImplicitLod, ctx, inst, ctx.F32[1], | 
					
						
							|  |  |  |                     Texture(ctx, info, index), coords, dref, operands.MaskOptional(), | 
					
						
							|  |  |  |                     operands.Span()); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         // Implicit lods in compute behave on hardware as if sampling from LOD 0.
 | 
					
						
							|  |  |  |         // This check is to ensure all drivers behave this way.
 | 
					
						
							|  |  |  |         const Id lod{ctx.Const(0.0f)}; | 
					
						
							|  |  |  |         const ImageOperands operands(ctx, false, true, false, lod, offset); | 
					
						
							|  |  |  |         return Emit(&EmitContext::OpImageSparseSampleDrefExplicitLod, | 
					
						
							|  |  |  |                     &EmitContext::OpImageSampleDrefExplicitLod, ctx, inst, ctx.F32[1], | 
					
						
							|  |  |  |                     Texture(ctx, info, index), coords, dref, operands.Mask(), operands.Span()); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitImageSampleDrefExplicitLod(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, | 
					
						
							| 
									
										
										
										
											2021-05-17 02:52:01 -03:00
										 |  |  |                                   Id coords, Id dref, Id lod, const IR::Value& offset) { | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     const auto info{inst->Flags<IR::TextureInstInfo>()}; | 
					
						
							| 
									
										
										
										
											2021-05-17 02:52:01 -03:00
										 |  |  |     const ImageOperands operands(ctx, false, true, false, lod, offset); | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     return Emit(&EmitContext::OpImageSparseSampleDrefExplicitLod, | 
					
						
							|  |  |  |                 &EmitContext::OpImageSampleDrefExplicitLod, ctx, inst, ctx.F32[1], | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |                 Texture(ctx, info, index), coords, dref, operands.Mask(), operands.Span()); | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 16:46:07 -03:00
										 |  |  | Id EmitImageGather(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id coords, | 
					
						
							|  |  |  |                    const IR::Value& offset, const IR::Value& offset2) { | 
					
						
							| 
									
										
										
										
											2021-03-24 23:41:55 +01:00
										 |  |  |     const auto info{inst->Flags<IR::TextureInstInfo>()}; | 
					
						
							| 
									
										
										
										
											2021-03-26 16:02:04 +01:00
										 |  |  |     const ImageOperands operands(ctx, offset, offset2); | 
					
						
							| 
									
										
										
										
											2021-03-24 23:41:55 +01:00
										 |  |  |     return Emit(&EmitContext::OpImageSparseGather, &EmitContext::OpImageGather, ctx, inst, | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |                 ctx.F32[4], Texture(ctx, info, index), coords, ctx.Const(info.gather_component), | 
					
						
							| 
									
										
										
										
											2021-05-23 04:15:36 -03:00
										 |  |  |                 operands.MaskOptional(), operands.Span()); | 
					
						
							| 
									
										
										
										
											2021-03-24 23:41:55 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitImageGatherDref(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id coords, | 
					
						
							| 
									
										
										
										
											2021-03-26 16:46:07 -03:00
										 |  |  |                        const IR::Value& offset, const IR::Value& offset2, Id dref) { | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |     const auto info{inst->Flags<IR::TextureInstInfo>()}; | 
					
						
							| 
									
										
										
										
											2021-03-26 16:02:04 +01:00
										 |  |  |     const ImageOperands operands(ctx, offset, offset2); | 
					
						
							| 
									
										
										
										
											2021-03-24 23:41:55 +01:00
										 |  |  |     return Emit(&EmitContext::OpImageSparseDrefGather, &EmitContext::OpImageDrefGather, ctx, inst, | 
					
						
							| 
									
										
										
										
											2021-05-23 04:15:36 -03:00
										 |  |  |                 ctx.F32[4], Texture(ctx, info, index), coords, dref, operands.MaskOptional(), | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |                 operands.Span()); | 
					
						
							| 
									
										
										
										
											2021-03-24 23:41:55 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 19:24:50 +01:00
										 |  |  | Id EmitImageFetch(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id coords, Id offset, | 
					
						
							|  |  |  |                   Id lod, Id ms) { | 
					
						
							|  |  |  |     const auto info{inst->Flags<IR::TextureInstInfo>()}; | 
					
						
							| 
									
										
										
										
											2021-04-06 02:56:15 -03:00
										 |  |  |     if (info.type == TextureType::Buffer) { | 
					
						
							|  |  |  |         lod = Id{}; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-26 16:46:07 -03:00
										 |  |  |     const ImageOperands operands(offset, lod, ms); | 
					
						
							| 
									
										
										
										
											2021-03-26 19:24:50 +01:00
										 |  |  |     return Emit(&EmitContext::OpImageSparseFetch, &EmitContext::OpImageFetch, ctx, inst, ctx.F32[4], | 
					
						
							| 
									
										
										
										
											2021-05-23 04:15:36 -03:00
										 |  |  |                 TextureImage(ctx, info, index), coords, operands.MaskOptional(), operands.Span()); | 
					
						
							| 
									
										
										
										
											2021-03-26 19:24:50 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-26 18:45:38 -03:00
										 |  |  | Id EmitImageQueryDimensions(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id lod) { | 
					
						
							|  |  |  |     const auto info{inst->Flags<IR::TextureInstInfo>()}; | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |     const Id image{TextureImage(ctx, info, index)}; | 
					
						
							| 
									
										
										
										
											2021-03-26 18:45:38 -03:00
										 |  |  |     const Id zero{ctx.u32_zero_value}; | 
					
						
							|  |  |  |     const auto mips{[&] { return ctx.OpImageQueryLevels(ctx.U32[1], image); }}; | 
					
						
							|  |  |  |     switch (info.type) { | 
					
						
							|  |  |  |     case TextureType::Color1D: | 
					
						
							|  |  |  |         return ctx.OpCompositeConstruct(ctx.U32[4], ctx.OpImageQuerySizeLod(ctx.U32[1], image, lod), | 
					
						
							|  |  |  |                                         zero, zero, mips()); | 
					
						
							|  |  |  |     case TextureType::ColorArray1D: | 
					
						
							|  |  |  |     case TextureType::Color2D: | 
					
						
							|  |  |  |     case TextureType::ColorCube: | 
					
						
							|  |  |  |         return ctx.OpCompositeConstruct(ctx.U32[4], ctx.OpImageQuerySizeLod(ctx.U32[2], image, lod), | 
					
						
							|  |  |  |                                         zero, mips()); | 
					
						
							|  |  |  |     case TextureType::ColorArray2D: | 
					
						
							|  |  |  |     case TextureType::Color3D: | 
					
						
							|  |  |  |     case TextureType::ColorArrayCube: | 
					
						
							|  |  |  |         return ctx.OpCompositeConstruct(ctx.U32[4], ctx.OpImageQuerySizeLod(ctx.U32[3], image, lod), | 
					
						
							|  |  |  |                                         mips()); | 
					
						
							| 
									
										
										
										
											2021-04-06 02:56:15 -03:00
										 |  |  |     case TextureType::Buffer: | 
					
						
							|  |  |  |         return ctx.OpCompositeConstruct(ctx.U32[4], ctx.OpImageQuerySize(ctx.U32[1], image), zero, | 
					
						
							|  |  |  |                                         zero, mips()); | 
					
						
							| 
									
										
										
										
											2021-03-26 18:45:38 -03:00
										 |  |  |     } | 
					
						
							|  |  |  |     throw LogicError("Unspecified image type {}", info.type.Value()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  | Id EmitImageQueryLod(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id coords) { | 
					
						
							|  |  |  |     const auto info{inst->Flags<IR::TextureInstInfo>()}; | 
					
						
							| 
									
										
										
										
											2021-03-28 19:47:52 +02:00
										 |  |  |     const Id zero{ctx.f32_zero_value}; | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |     const Id sampler{Texture(ctx, info, index)}; | 
					
						
							| 
									
										
										
										
											2021-03-28 21:25:08 +02:00
										 |  |  |     return ctx.OpCompositeConstruct(ctx.F32[4], ctx.OpImageQueryLod(ctx.F32[2], sampler, coords), | 
					
						
							| 
									
										
										
										
											2021-03-28 19:47:52 +02:00
										 |  |  |                                     zero, zero); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-29 02:00:43 +02:00
										 |  |  | Id EmitImageGradient(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id coords, | 
					
						
							|  |  |  |                      Id derivates, Id offset, Id lod_clamp) { | 
					
						
							|  |  |  |     const auto info{inst->Flags<IR::TextureInstInfo>()}; | 
					
						
							| 
									
										
										
										
											2021-03-30 08:41:21 +02:00
										 |  |  |     const ImageOperands operands(ctx, info.has_lod_clamp != 0, derivates, info.num_derivates, | 
					
						
							|  |  |  |                                  offset, lod_clamp); | 
					
						
							| 
									
										
										
										
											2021-03-29 02:00:43 +02:00
										 |  |  |     return Emit(&EmitContext::OpImageSparseSampleExplicitLod, | 
					
						
							| 
									
										
										
										
											2021-04-22 16:17:59 -03:00
										 |  |  |                 &EmitContext::OpImageSampleExplicitLod, ctx, inst, ctx.F32[4], | 
					
						
							|  |  |  |                 Texture(ctx, info, index), coords, operands.Mask(), operands.Span()); | 
					
						
							| 
									
										
										
										
											2021-03-29 02:00:43 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  | Id EmitImageRead(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id coords) { | 
					
						
							|  |  |  |     const auto info{inst->Flags<IR::TextureInstInfo>()}; | 
					
						
							| 
									
										
										
										
											2021-04-11 02:37:03 -03:00
										 |  |  |     if (info.image_format == ImageFormat::Typeless && !ctx.profile.support_typeless_image_loads) { | 
					
						
							| 
									
										
										
										
											2021-06-14 02:27:49 -03:00
										 |  |  |         LOG_WARNING(Shader_SPIRV, "Typeless image read not supported by host"); | 
					
						
							| 
									
										
										
										
											2021-04-11 02:37:03 -03:00
										 |  |  |         return ctx.ConstantNull(ctx.U32[4]); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-09 01:45:39 -03:00
										 |  |  |     return Emit(&EmitContext::OpImageSparseRead, &EmitContext::OpImageRead, ctx, inst, ctx.U32[4], | 
					
						
							|  |  |  |                 Image(ctx, index, info), coords, std::nullopt, std::span<const Id>{}); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EmitImageWrite(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id coords, Id color) { | 
					
						
							|  |  |  |     const auto info{inst->Flags<IR::TextureInstInfo>()}; | 
					
						
							|  |  |  |     ctx.OpImageWrite(Image(ctx, index, info), coords, color); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-25 22:26:23 -03:00
										 |  |  | Id EmitIsTextureScaled(EmitContext& ctx, const IR::Value& index) { | 
					
						
							| 
									
										
										
										
											2021-08-01 18:57:45 -03:00
										 |  |  |     if (ctx.profile.unified_descriptor_binding) { | 
					
						
							|  |  |  |         const Id member_index{ctx.Const(ctx.rescaling_textures_member_index)}; | 
					
						
							|  |  |  |         return IsScaled(ctx, index, member_index, ctx.texture_rescaling_index); | 
					
						
							| 
									
										
										
										
											2021-07-25 22:26:23 -03:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2021-08-01 18:57:45 -03:00
										 |  |  |         const Id composite{ctx.OpLoad(ctx.F32[4], ctx.rescaling_uniform_constant)}; | 
					
						
							|  |  |  |         const Id mask_f32{ctx.OpCompositeExtract(ctx.F32[1], composite, 0u)}; | 
					
						
							|  |  |  |         const Id mask{ctx.OpBitcast(ctx.U32[1], mask_f32)}; | 
					
						
							|  |  |  |         return BitTest(ctx, mask, ctx.Def(index)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitIsImageScaled(EmitContext& ctx, const IR::Value& index) { | 
					
						
							|  |  |  |     if (ctx.profile.unified_descriptor_binding) { | 
					
						
							|  |  |  |         const Id member_index{ctx.Const(ctx.rescaling_images_member_index)}; | 
					
						
							|  |  |  |         return IsScaled(ctx, index, member_index, ctx.image_rescaling_index); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         const Id composite{ctx.OpLoad(ctx.F32[4], ctx.rescaling_uniform_constant)}; | 
					
						
							|  |  |  |         const Id mask_f32{ctx.OpCompositeExtract(ctx.F32[1], composite, 1u)}; | 
					
						
							|  |  |  |         const Id mask{ctx.OpBitcast(ctx.U32[1], mask_f32)}; | 
					
						
							|  |  |  |         return BitTest(ctx, mask, ctx.Def(index)); | 
					
						
							| 
									
										
										
										
											2021-07-25 22:26:23 -03:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-07-25 22:04:53 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  | } // namespace Shader::Backend::SPIRV
 |