summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/video_core/engines/shader_bytecode.h20
-rw-r--r--src/video_core/renderer_opengl/gl_shader_decompiler.cpp14
2 files changed, 31 insertions, 3 deletions
diff --git a/src/video_core/engines/shader_bytecode.h b/src/video_core/engines/shader_bytecode.h
index 3b70efeec..2bc1782ad 100644
--- a/src/video_core/engines/shader_bytecode.h
+++ b/src/video_core/engines/shader_bytecode.h
@@ -194,6 +194,13 @@ enum class UniformType : u64 {
194 Double = 5, 194 Double = 5,
195}; 195};
196 196
197enum class IMinMaxExchange : u64 {
198 None = 0,
199 XLo = 1,
200 XMed = 2,
201 XHi = 3,
202};
203
197union Instruction { 204union Instruction {
198 Instruction& operator=(const Instruction& instr) { 205 Instruction& operator=(const Instruction& instr) {
199 value = instr.value; 206 value = instr.value;
@@ -279,6 +286,13 @@ union Instruction {
279 } alu_integer; 286 } alu_integer;
280 287
281 union { 288 union {
289 BitField<39, 3, u64> pred;
290 BitField<42, 1, u64> negate_pred;
291 BitField<43, 2, IMinMaxExchange> exchange;
292 BitField<48, 1, u64> is_signed;
293 } imnmx;
294
295 union {
282 BitField<54, 1, u64> saturate; 296 BitField<54, 1, u64> saturate;
283 BitField<56, 1, u64> negate_a; 297 BitField<56, 1, u64> negate_a;
284 } iadd32i; 298 } iadd32i;
@@ -700,9 +714,9 @@ private:
700 INST("0100110001100---", Id::FMNMX_C, Type::Arithmetic, "FMNMX_C"), 714 INST("0100110001100---", Id::FMNMX_C, Type::Arithmetic, "FMNMX_C"),
701 INST("0101110001100---", Id::FMNMX_R, Type::Arithmetic, "FMNMX_R"), 715 INST("0101110001100---", Id::FMNMX_R, Type::Arithmetic, "FMNMX_R"),
702 INST("0011100-01100---", Id::FMNMX_IMM, Type::Arithmetic, "FMNMX_IMM"), 716 INST("0011100-01100---", Id::FMNMX_IMM, Type::Arithmetic, "FMNMX_IMM"),
703 INST("0100110000100---", Id::IMNMX_C, Type::Arithmetic, "FMNMX_IMM"), 717 INST("0100110000100---", Id::IMNMX_C, Type::ArithmeticInteger, "IMNMX_C"),
704 INST("0101110000100---", Id::IMNMX_R, Type::Arithmetic, "FMNMX_IMM"), 718 INST("0101110000100---", Id::IMNMX_R, Type::ArithmeticInteger, "IMNMX_R"),
705 INST("0011100-00100---", Id::IMNMX_IMM, Type::Arithmetic, "FMNMX_IMM"), 719 INST("0011100-00100---", Id::IMNMX_IMM, Type::ArithmeticInteger, "IMNMX_IMM"),
706 INST("0100110000000---", Id::BFE_C, Type::Bfe, "BFE_C"), 720 INST("0100110000000---", Id::BFE_C, Type::Bfe, "BFE_C"),
707 INST("0101110000000---", Id::BFE_R, Type::Bfe, "BFE_R"), 721 INST("0101110000000---", Id::BFE_R, Type::Bfe, "BFE_R"),
708 INST("0011100-00000---", Id::BFE_IMM, Type::Bfe, "BFE_IMM"), 722 INST("0011100-00000---", Id::BFE_IMM, Type::Bfe, "BFE_IMM"),
diff --git a/src/video_core/renderer_opengl/gl_shader_decompiler.cpp b/src/video_core/renderer_opengl/gl_shader_decompiler.cpp
index e817aca5a..5914077e8 100644
--- a/src/video_core/renderer_opengl/gl_shader_decompiler.cpp
+++ b/src/video_core/renderer_opengl/gl_shader_decompiler.cpp
@@ -1127,6 +1127,20 @@ private:
1127 WriteLogicOperation(instr.gpr0, instr.alu.lop.operation, op_a, op_b); 1127 WriteLogicOperation(instr.gpr0, instr.alu.lop.operation, op_a, op_b);
1128 break; 1128 break;
1129 } 1129 }
1130 case OpCode::Id::IMNMX_C:
1131 case OpCode::Id::IMNMX_R:
1132 case OpCode::Id::IMNMX_IMM: {
1133 ASSERT_MSG(instr.imnmx.exchange == Tegra::Shader::IMinMaxExchange::None,
1134 "Unimplemented");
1135 std::string condition =
1136 GetPredicateCondition(instr.imnmx.pred, instr.imnmx.negate_pred != 0);
1137 std::string parameters = op_a + ',' + op_b;
1138 regs.SetRegisterToInteger(instr.gpr0, instr.imnmx.is_signed, 0,
1139 '(' + condition + ") ? min(" + parameters + ") : max(" +
1140 parameters + ')',
1141 1, 1);
1142 break;
1143 }
1130 default: { 1144 default: {
1131 LOG_CRITICAL(HW_GPU, "Unhandled ArithmeticInteger instruction: {}", 1145 LOG_CRITICAL(HW_GPU, "Unhandled ArithmeticInteger instruction: {}",
1132 opcode->GetName()); 1146 opcode->GetName());