| 
									
										
										
										
											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"
 | 
					
						
							|  |  |  | #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, | 
					
						
							|  |  |  |                            Id lod, Id offset) { | 
					
						
							|  |  |  |         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); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (Sirit::ValidId(offset)) { | 
					
						
							|  |  |  |             Add(spv::ImageOperandsMask::Offset, offset); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         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-04-01 08:25:55 +02:00
										 |  |  |             // LOG_WARNING("Not all arguments in PTP are immediate, STUBBING");
 | 
					
						
							|  |  |  |             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-05 22:25:22 -04:00
										 |  |  |         auto read{[&](unsigned int a, unsigned int b) { | 
					
						
							|  |  |  |             return ctx.Constant(ctx.U32[1], values[a]->Arg(b).U32()); | 
					
						
							|  |  |  |         }}; | 
					
						
							| 
									
										
										
										
											2021-03-26 16:46:07 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         const Id offsets{ | 
					
						
							|  |  |  |             ctx.ConstantComposite(ctx.TypeArray(ctx.U32[2], ctx.Constant(ctx.U32[1], 4)), | 
					
						
							|  |  |  |                                   ctx.ConstantComposite(ctx.U32[2], read(0, 0), read(0, 1)), | 
					
						
							|  |  |  |                                   ctx.ConstantComposite(ctx.U32[2], read(0, 2), read(0, 3)), | 
					
						
							|  |  |  |                                   ctx.ConstantComposite(ctx.U32[2], read(1, 0), read(1, 1)), | 
					
						
							|  |  |  |                                   ctx.ConstantComposite(ctx.U32[2], read(1, 2), read(1, 3)))}; | 
					
						
							|  |  |  |         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-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); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     std::span<const Id> Span() const noexcept { | 
					
						
							|  |  |  |         return std::span{operands.data(), operands.size()}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     spv::ImageOperandsMask Mask() const noexcept { | 
					
						
							|  |  |  |         return mask; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2021-03-29 02:00:43 +02:00
										 |  |  |     boost::container::static_vector<Id, 4> operands; | 
					
						
							| 
									
										
										
										
											2021-03-08 18:31:53 -03:00
										 |  |  |     spv::ImageOperandsMask mask{}; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id Texture(EmitContext& ctx, const IR::Value& index) { | 
					
						
							|  |  |  |     if (index.IsImmediate()) { | 
					
						
							|  |  |  |         const TextureDefinition def{ctx.textures.at(index.U32())}; | 
					
						
							| 
									
										
										
										
											2021-03-26 18:45:38 -03:00
										 |  |  |         return ctx.OpLoad(def.sampled_type, def.id); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     throw NotImplementedException("Indirect texture sample"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-06 02:56:15 -03:00
										 |  |  | Id TextureImage(EmitContext& ctx, const IR::Value& index, IR::TextureInstInfo info) { | 
					
						
							|  |  |  |     if (!index.IsImmediate()) { | 
					
						
							|  |  |  |         throw NotImplementedException("Indirect texture sample"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (info.type == TextureType::Buffer) { | 
					
						
							|  |  |  |         const Id sampler_id{ctx.texture_buffers.at(index.U32())}; | 
					
						
							|  |  |  |         const Id id{ctx.OpLoad(ctx.sampled_texture_buffer_type, sampler_id)}; | 
					
						
							|  |  |  |         return ctx.OpImage(ctx.image_buffer_type, id); | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2021-03-26 18:45:38 -03:00
										 |  |  |         const TextureDefinition def{ctx.textures.at(index.U32())}; | 
					
						
							|  |  |  |         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) { | 
					
						
							|  |  |  |     if (!index.IsImmediate()) { | 
					
						
							|  |  |  |         throw NotImplementedException("Indirect image indexing"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (info.type == TextureType::Buffer) { | 
					
						
							|  |  |  |         throw NotImplementedException("Image buffer"); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         const ImageDefinition def{ctx.images.at(index.U32())}; | 
					
						
							|  |  |  |         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); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | } // 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, | 
					
						
							|  |  |  |                               Id bias_lc, Id offset) { | 
					
						
							|  |  |  |     const auto info{inst->Flags<IR::TextureInstInfo>()}; | 
					
						
							|  |  |  |     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], Texture(ctx, index), | 
					
						
							|  |  |  |                 coords, operands.Mask(), operands.Span()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitImageSampleExplicitLod(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id coords, | 
					
						
							|  |  |  |                               Id lod_lc, Id offset) { | 
					
						
							|  |  |  |     const auto info{inst->Flags<IR::TextureInstInfo>()}; | 
					
						
							|  |  |  |     const ImageOperands operands(ctx, false, true, info.has_lod_clamp != 0, lod_lc, offset); | 
					
						
							|  |  |  |     return Emit(&EmitContext::OpImageSparseSampleExplicitLod, | 
					
						
							|  |  |  |                 &EmitContext::OpImageSampleExplicitLod, ctx, inst, ctx.F32[4], Texture(ctx, index), | 
					
						
							|  |  |  |                 coords, operands.Mask(), operands.Span()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitImageSampleDrefImplicitLod(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, | 
					
						
							|  |  |  |                                   Id coords, Id dref, Id bias_lc, Id offset) { | 
					
						
							|  |  |  |     const auto info{inst->Flags<IR::TextureInstInfo>()}; | 
					
						
							|  |  |  |     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, index), coords, dref, operands.Mask(), operands.Span()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Id EmitImageSampleDrefExplicitLod(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, | 
					
						
							|  |  |  |                                   Id coords, Id dref, Id lod_lc, Id offset) { | 
					
						
							|  |  |  |     const auto info{inst->Flags<IR::TextureInstInfo>()}; | 
					
						
							|  |  |  |     const ImageOperands operands(ctx, false, true, info.has_lod_clamp != 0, lod_lc, offset); | 
					
						
							|  |  |  |     return Emit(&EmitContext::OpImageSparseSampleDrefExplicitLod, | 
					
						
							|  |  |  |                 &EmitContext::OpImageSampleDrefExplicitLod, ctx, inst, ctx.F32[1], | 
					
						
							|  |  |  |                 Texture(ctx, index), coords, dref, operands.Mask(), operands.Span()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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, | 
					
						
							|  |  |  |                 ctx.F32[4], Texture(ctx, index), coords, | 
					
						
							|  |  |  |                 ctx.Constant(ctx.U32[1], info.gather_component.Value()), operands.Mask(), | 
					
						
							|  |  |  |                 operands.Span()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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-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, | 
					
						
							|  |  |  |                 ctx.F32[4], Texture(ctx, index), coords, dref, operands.Mask(), operands.Span()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-04-06 02:56:15 -03:00
										 |  |  |                 TextureImage(ctx, index, info), coords, operands.Mask(), 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-06 02:56:15 -03:00
										 |  |  |     const Id image{TextureImage(ctx, index, info)}; | 
					
						
							| 
									
										
										
										
											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-03-28 19:47:52 +02:00
										 |  |  | Id EmitImageQueryLod(EmitContext& ctx, IR::Inst*, const IR::Value& index, Id coords) { | 
					
						
							|  |  |  |     const Id zero{ctx.f32_zero_value}; | 
					
						
							| 
									
										
										
										
											2021-03-28 21:25:08 +02:00
										 |  |  |     const Id sampler{Texture(ctx, index)}; | 
					
						
							|  |  |  |     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, | 
					
						
							|  |  |  |                 &EmitContext::OpImageSampleExplicitLod, ctx, inst, ctx.F32[4], Texture(ctx, index), | 
					
						
							|  |  |  |                 coords, operands.Mask(), operands.Span()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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>()}; | 
					
						
							|  |  |  |     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-03-08 18:31:53 -03:00
										 |  |  | } // namespace Shader::Backend::SPIRV
 |