diff options
| author | 2020-01-04 14:05:17 -0500 | |
|---|---|---|
| committer | 2020-01-04 14:05:17 -0500 | |
| commit | cd0a7dfdbc3ed3ce83a2ad5bb143f46e4f92dfb8 (patch) | |
| tree | e60c9786f929a2b35c7f23ee7a7dbebcc14eb7a7 /src | |
| parent | Merge pull request #3247 from FernandoS27/remap-fix (diff) | |
| parent | Shader_IR: Address Feedback (diff) | |
| download | yuzu-cd0a7dfdbc3ed3ce83a2ad5bb143f46e4f92dfb8.tar.gz yuzu-cd0a7dfdbc3ed3ce83a2ad5bb143f46e4f92dfb8.tar.xz yuzu-cd0a7dfdbc3ed3ce83a2ad5bb143f46e4f92dfb8.zip | |
Merge pull request #3258 from FernandoS27/shader-amend
Shader_IR: add the ability to amend code in the shader ir.
Diffstat (limited to 'src')
| -rw-r--r-- | src/video_core/renderer_opengl/gl_shader_decompiler.cpp | 6 | ||||
| -rw-r--r-- | src/video_core/renderer_vulkan/vk_shader_decompiler.cpp | 8 | ||||
| -rw-r--r-- | src/video_core/shader/node.h | 26 | ||||
| -rw-r--r-- | src/video_core/shader/shader_ir.cpp | 6 | ||||
| -rw-r--r-- | src/video_core/shader/shader_ir.h | 8 |
5 files changed, 52 insertions, 2 deletions
diff --git a/src/video_core/renderer_opengl/gl_shader_decompiler.cpp b/src/video_core/renderer_opengl/gl_shader_decompiler.cpp index a311dbcfe..f9f7a97b5 100644 --- a/src/video_core/renderer_opengl/gl_shader_decompiler.cpp +++ b/src/video_core/renderer_opengl/gl_shader_decompiler.cpp | |||
| @@ -751,6 +751,9 @@ private: | |||
| 751 | 751 | ||
| 752 | Expression Visit(const Node& node) { | 752 | Expression Visit(const Node& node) { |
| 753 | if (const auto operation = std::get_if<OperationNode>(&*node)) { | 753 | if (const auto operation = std::get_if<OperationNode>(&*node)) { |
| 754 | if (const auto amend_index = operation->GetAmendIndex()) { | ||
| 755 | Visit(ir.GetAmendNode(*amend_index)).CheckVoid(); | ||
| 756 | } | ||
| 754 | const auto operation_index = static_cast<std::size_t>(operation->GetCode()); | 757 | const auto operation_index = static_cast<std::size_t>(operation->GetCode()); |
| 755 | if (operation_index >= operation_decompilers.size()) { | 758 | if (operation_index >= operation_decompilers.size()) { |
| 756 | UNREACHABLE_MSG("Out of bounds operation: {}", operation_index); | 759 | UNREACHABLE_MSG("Out of bounds operation: {}", operation_index); |
| @@ -872,6 +875,9 @@ private: | |||
| 872 | } | 875 | } |
| 873 | 876 | ||
| 874 | if (const auto conditional = std::get_if<ConditionalNode>(&*node)) { | 877 | if (const auto conditional = std::get_if<ConditionalNode>(&*node)) { |
| 878 | if (const auto amend_index = conditional->GetAmendIndex()) { | ||
| 879 | Visit(ir.GetAmendNode(*amend_index)).CheckVoid(); | ||
| 880 | } | ||
| 875 | // It's invalid to call conditional on nested nodes, use an operation instead | 881 | // It's invalid to call conditional on nested nodes, use an operation instead |
| 876 | code.AddLine("if ({}) {{", Visit(conditional->GetCondition()).AsBool()); | 882 | code.AddLine("if ({}) {{", Visit(conditional->GetCondition()).AsBool()); |
| 877 | ++code.scope; | 883 | ++code.scope; |
diff --git a/src/video_core/renderer_vulkan/vk_shader_decompiler.cpp b/src/video_core/renderer_vulkan/vk_shader_decompiler.cpp index a8baf91de..8fe852ce8 100644 --- a/src/video_core/renderer_vulkan/vk_shader_decompiler.cpp +++ b/src/video_core/renderer_vulkan/vk_shader_decompiler.cpp | |||
| @@ -954,6 +954,10 @@ private: | |||
| 954 | 954 | ||
| 955 | Expression Visit(const Node& node) { | 955 | Expression Visit(const Node& node) { |
| 956 | if (const auto operation = std::get_if<OperationNode>(&*node)) { | 956 | if (const auto operation = std::get_if<OperationNode>(&*node)) { |
| 957 | if (const auto amend_index = operation->GetAmendIndex()) { | ||
| 958 | [[maybe_unused]] const Type type = Visit(ir.GetAmendNode(*amend_index)).type; | ||
| 959 | ASSERT(type == Type::Void); | ||
| 960 | } | ||
| 957 | const auto operation_index = static_cast<std::size_t>(operation->GetCode()); | 961 | const auto operation_index = static_cast<std::size_t>(operation->GetCode()); |
| 958 | const auto decompiler = operation_decompilers[operation_index]; | 962 | const auto decompiler = operation_decompilers[operation_index]; |
| 959 | if (decompiler == nullptr) { | 963 | if (decompiler == nullptr) { |
| @@ -1142,6 +1146,10 @@ private: | |||
| 1142 | } | 1146 | } |
| 1143 | 1147 | ||
| 1144 | if (const auto conditional = std::get_if<ConditionalNode>(&*node)) { | 1148 | if (const auto conditional = std::get_if<ConditionalNode>(&*node)) { |
| 1149 | if (const auto amend_index = conditional->GetAmendIndex()) { | ||
| 1150 | [[maybe_unused]] const Type type = Visit(ir.GetAmendNode(*amend_index)).type; | ||
| 1151 | ASSERT(type == Type::Void); | ||
| 1152 | } | ||
| 1145 | // It's invalid to call conditional on nested nodes, use an operation instead | 1153 | // It's invalid to call conditional on nested nodes, use an operation instead |
| 1146 | const Id true_label = OpLabel(); | 1154 | const Id true_label = OpLabel(); |
| 1147 | const Id skip_label = OpLabel(); | 1155 | const Id skip_label = OpLabel(); |
diff --git a/src/video_core/shader/node.h b/src/video_core/shader/node.h index 4d2f4d6a8..4e155542a 100644 --- a/src/video_core/shader/node.h +++ b/src/video_core/shader/node.h | |||
| @@ -392,8 +392,30 @@ struct MetaImage { | |||
| 392 | using Meta = | 392 | using Meta = |
| 393 | std::variant<MetaArithmetic, MetaTexture, MetaImage, MetaStackClass, Tegra::Shader::HalfType>; | 393 | std::variant<MetaArithmetic, MetaTexture, MetaImage, MetaStackClass, Tegra::Shader::HalfType>; |
| 394 | 394 | ||
| 395 | class AmendNode { | ||
| 396 | public: | ||
| 397 | std::optional<std::size_t> GetAmendIndex() const { | ||
| 398 | if (amend_index == amend_null_index) { | ||
| 399 | return std::nullopt; | ||
| 400 | } | ||
| 401 | return {amend_index}; | ||
| 402 | } | ||
| 403 | |||
| 404 | void SetAmendIndex(std::size_t index) { | ||
| 405 | amend_index = index; | ||
| 406 | } | ||
| 407 | |||
| 408 | void ClearAmend() { | ||
| 409 | amend_index = amend_null_index; | ||
| 410 | } | ||
| 411 | |||
| 412 | private: | ||
| 413 | static constexpr std::size_t amend_null_index = 0xFFFFFFFFFFFFFFFFULL; | ||
| 414 | std::size_t amend_index{amend_null_index}; | ||
| 415 | }; | ||
| 416 | |||
| 395 | /// Holds any kind of operation that can be done in the IR | 417 | /// Holds any kind of operation that can be done in the IR |
| 396 | class OperationNode final { | 418 | class OperationNode final : public AmendNode { |
| 397 | public: | 419 | public: |
| 398 | explicit OperationNode(OperationCode code) : OperationNode(code, Meta{}) {} | 420 | explicit OperationNode(OperationCode code) : OperationNode(code, Meta{}) {} |
| 399 | 421 | ||
| @@ -433,7 +455,7 @@ private: | |||
| 433 | }; | 455 | }; |
| 434 | 456 | ||
| 435 | /// Encloses inside any kind of node that returns a boolean conditionally-executed code | 457 | /// Encloses inside any kind of node that returns a boolean conditionally-executed code |
| 436 | class ConditionalNode final { | 458 | class ConditionalNode final : public AmendNode { |
| 437 | public: | 459 | public: |
| 438 | explicit ConditionalNode(Node condition, std::vector<Node>&& code) | 460 | explicit ConditionalNode(Node condition, std::vector<Node>&& code) |
| 439 | : condition{std::move(condition)}, code{std::move(code)} {} | 461 | : condition{std::move(condition)}, code{std::move(code)} {} |
diff --git a/src/video_core/shader/shader_ir.cpp b/src/video_core/shader/shader_ir.cpp index 1d9825c76..31eecb3f4 100644 --- a/src/video_core/shader/shader_ir.cpp +++ b/src/video_core/shader/shader_ir.cpp | |||
| @@ -446,4 +446,10 @@ Node ShaderIR::BitfieldInsert(Node base, Node insert, u32 offset, u32 bits) { | |||
| 446 | Immediate(bits)); | 446 | Immediate(bits)); |
| 447 | } | 447 | } |
| 448 | 448 | ||
| 449 | std::size_t ShaderIR::DeclareAmend(Node new_amend) { | ||
| 450 | const std::size_t id = amend_code.size(); | ||
| 451 | amend_code.push_back(new_amend); | ||
| 452 | return id; | ||
| 453 | } | ||
| 454 | |||
| 449 | } // namespace VideoCommon::Shader | 455 | } // namespace VideoCommon::Shader |
diff --git a/src/video_core/shader/shader_ir.h b/src/video_core/shader/shader_ir.h index baed06ccd..aacd0a0da 100644 --- a/src/video_core/shader/shader_ir.h +++ b/src/video_core/shader/shader_ir.h | |||
| @@ -176,6 +176,10 @@ public: | |||
| 176 | /// Returns a condition code evaluated from internal flags | 176 | /// Returns a condition code evaluated from internal flags |
| 177 | Node GetConditionCode(Tegra::Shader::ConditionCode cc) const; | 177 | Node GetConditionCode(Tegra::Shader::ConditionCode cc) const; |
| 178 | 178 | ||
| 179 | const Node& GetAmendNode(std::size_t index) const { | ||
| 180 | return amend_code[index]; | ||
| 181 | } | ||
| 182 | |||
| 179 | private: | 183 | private: |
| 180 | friend class ASTDecoder; | 184 | friend class ASTDecoder; |
| 181 | 185 | ||
| @@ -392,6 +396,9 @@ private: | |||
| 392 | Tegra::Shader::Instruction instr, | 396 | Tegra::Shader::Instruction instr, |
| 393 | bool is_write); | 397 | bool is_write); |
| 394 | 398 | ||
| 399 | /// Register new amending code and obtain the reference id. | ||
| 400 | std::size_t DeclareAmend(Node new_amend); | ||
| 401 | |||
| 395 | const ProgramCode& program_code; | 402 | const ProgramCode& program_code; |
| 396 | const u32 main_offset; | 403 | const u32 main_offset; |
| 397 | const CompilerSettings settings; | 404 | const CompilerSettings settings; |
| @@ -406,6 +413,7 @@ private: | |||
| 406 | std::map<u32, NodeBlock> basic_blocks; | 413 | std::map<u32, NodeBlock> basic_blocks; |
| 407 | NodeBlock global_code; | 414 | NodeBlock global_code; |
| 408 | ASTManager program_manager{true, true}; | 415 | ASTManager program_manager{true, true}; |
| 416 | std::vector<Node> amend_code; | ||
| 409 | 417 | ||
| 410 | std::set<u32> used_registers; | 418 | std::set<u32> used_registers; |
| 411 | std::set<Tegra::Shader::Pred> used_predicates; | 419 | std::set<Tegra::Shader::Pred> used_predicates; |