mirror of
https://git.suyu.dev/suyu/suyu
synced 2025-01-09 16:03:21 +00:00
gl_shader_decompiler: Implement integer size conversions for I2I/I2F/F2I.
This commit is contained in:
parent
fb5bd0920d
commit
d383043e07
2 changed files with 43 additions and 14 deletions
|
@ -343,7 +343,8 @@ union Instruction {
|
||||||
} iset;
|
} iset;
|
||||||
|
|
||||||
union {
|
union {
|
||||||
BitField<10, 2, Register::Size> size;
|
BitField<8, 2, Register::Size> dest_size;
|
||||||
|
BitField<10, 2, Register::Size> src_size;
|
||||||
BitField<12, 1, u64> is_output_signed;
|
BitField<12, 1, u64> is_output_signed;
|
||||||
BitField<13, 1, u64> is_input_signed;
|
BitField<13, 1, u64> is_input_signed;
|
||||||
BitField<41, 2, u64> selector;
|
BitField<41, 2, u64> selector;
|
||||||
|
|
|
@ -265,6 +265,27 @@ public:
|
||||||
BuildRegisterList();
|
BuildRegisterList();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns code that does an integer size conversion for the specified size.
|
||||||
|
* @param value Value to perform integer size conversion on.
|
||||||
|
* @param size Register size to use for conversion instructions.
|
||||||
|
* @returns GLSL string corresponding to the value converted to the specified size.
|
||||||
|
*/
|
||||||
|
static std::string ConvertIntegerSize(const std::string& value, Register::Size size) {
|
||||||
|
switch (size) {
|
||||||
|
case Register::Size::Byte:
|
||||||
|
return "((" + value + " << 24) >> 24)";
|
||||||
|
case Register::Size::Short:
|
||||||
|
return "((" + value + " << 16) >> 16)";
|
||||||
|
case Register::Size::Word:
|
||||||
|
// Default - do nothing
|
||||||
|
return value;
|
||||||
|
default:
|
||||||
|
NGLOG_CRITICAL(HW_GPU, "Unimplemented conversion size {}", static_cast<u32>(size));
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets a register as an float.
|
* Gets a register as an float.
|
||||||
* @param reg The register to get.
|
* @param reg The register to get.
|
||||||
|
@ -281,15 +302,18 @@ public:
|
||||||
* @param reg The register to get.
|
* @param reg The register to get.
|
||||||
* @param elem The element to use for the operation.
|
* @param elem The element to use for the operation.
|
||||||
* @param is_signed Whether to get the register as a signed (or unsigned) integer.
|
* @param is_signed Whether to get the register as a signed (or unsigned) integer.
|
||||||
|
* @param size Register size to use for conversion instructions.
|
||||||
* @returns GLSL string corresponding to the register as an integer.
|
* @returns GLSL string corresponding to the register as an integer.
|
||||||
*/
|
*/
|
||||||
std::string GetRegisterAsInteger(const Register& reg, unsigned elem = 0,
|
std::string GetRegisterAsInteger(const Register& reg, unsigned elem = 0, bool is_signed = true,
|
||||||
bool is_signed = true) {
|
Register::Size size = Register::Size::Word) {
|
||||||
const std::string func = GetGLSLConversionFunc(
|
const std::string func = GetGLSLConversionFunc(
|
||||||
GLSLRegister::Type::Float,
|
GLSLRegister::Type::Float,
|
||||||
is_signed ? GLSLRegister::Type::Integer : GLSLRegister::Type::UnsignedInteger);
|
is_signed ? GLSLRegister::Type::Integer : GLSLRegister::Type::UnsignedInteger);
|
||||||
|
|
||||||
return func + '(' + GetRegister(reg, elem) + ')';
|
std::string value = func + '(' + GetRegister(reg, elem) + ')';
|
||||||
|
|
||||||
|
return ConvertIntegerSize(value, size);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -319,19 +343,20 @@ public:
|
||||||
* @param value_num_components Number of components in the value.
|
* @param value_num_components Number of components in the value.
|
||||||
* @param is_saturated Optional, when True, saturates the provided value.
|
* @param is_saturated Optional, when True, saturates the provided value.
|
||||||
* @param dest_elem Optional, the destination element to use for the operation.
|
* @param dest_elem Optional, the destination element to use for the operation.
|
||||||
|
* @param size Register size to use for conversion instructions.
|
||||||
*/
|
*/
|
||||||
void SetRegisterToInteger(const Register& reg, bool is_signed, u64 elem,
|
void SetRegisterToInteger(const Register& reg, bool is_signed, u64 elem,
|
||||||
const std::string& value, u64 dest_num_components,
|
const std::string& value, u64 dest_num_components,
|
||||||
u64 value_num_components, bool is_saturated = false,
|
u64 value_num_components, bool is_saturated = false,
|
||||||
u64 dest_elem = 0) {
|
u64 dest_elem = 0, Register::Size size = Register::Size::Word) {
|
||||||
ASSERT_MSG(!is_saturated, "Unimplemented");
|
ASSERT_MSG(!is_saturated, "Unimplemented");
|
||||||
|
|
||||||
const std::string func = GetGLSLConversionFunc(
|
const std::string func = GetGLSLConversionFunc(
|
||||||
is_signed ? GLSLRegister::Type::Integer : GLSLRegister::Type::UnsignedInteger,
|
is_signed ? GLSLRegister::Type::Integer : GLSLRegister::Type::UnsignedInteger,
|
||||||
GLSLRegister::Type::Float);
|
GLSLRegister::Type::Float);
|
||||||
|
|
||||||
SetRegister(reg, elem, func + '(' + value + ')', dest_num_components, value_num_components,
|
SetRegister(reg, elem, func + '(' + ConvertIntegerSize(value, size) + ')',
|
||||||
dest_elem);
|
dest_num_components, value_num_components, dest_elem);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1114,28 +1139,28 @@ private:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case OpCode::Type::Conversion: {
|
case OpCode::Type::Conversion: {
|
||||||
ASSERT_MSG(instr.conversion.size == Register::Size::Word, "Unimplemented");
|
|
||||||
ASSERT_MSG(!instr.conversion.negate_a, "Unimplemented");
|
ASSERT_MSG(!instr.conversion.negate_a, "Unimplemented");
|
||||||
|
|
||||||
switch (opcode->GetId()) {
|
switch (opcode->GetId()) {
|
||||||
case OpCode::Id::I2I_R: {
|
case OpCode::Id::I2I_R: {
|
||||||
ASSERT_MSG(!instr.conversion.selector, "Unimplemented");
|
ASSERT_MSG(!instr.conversion.selector, "Unimplemented");
|
||||||
|
|
||||||
std::string op_a =
|
std::string op_a = regs.GetRegisterAsInteger(
|
||||||
regs.GetRegisterAsInteger(instr.gpr20, 0, instr.conversion.is_input_signed);
|
instr.gpr20, 0, instr.conversion.is_input_signed, instr.conversion.src_size);
|
||||||
|
|
||||||
if (instr.conversion.abs_a) {
|
if (instr.conversion.abs_a) {
|
||||||
op_a = "abs(" + op_a + ')';
|
op_a = "abs(" + op_a + ')';
|
||||||
}
|
}
|
||||||
|
|
||||||
regs.SetRegisterToInteger(instr.gpr0, instr.conversion.is_output_signed, 0, op_a, 1,
|
regs.SetRegisterToInteger(instr.gpr0, instr.conversion.is_output_signed, 0, op_a, 1,
|
||||||
1, instr.alu.saturate_d);
|
1, instr.alu.saturate_d, 0, instr.conversion.dest_size);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case OpCode::Id::I2F_R: {
|
case OpCode::Id::I2F_R: {
|
||||||
|
ASSERT_MSG(instr.conversion.dest_size == Register::Size::Word, "Unimplemented");
|
||||||
ASSERT_MSG(!instr.conversion.selector, "Unimplemented");
|
ASSERT_MSG(!instr.conversion.selector, "Unimplemented");
|
||||||
std::string op_a =
|
std::string op_a = regs.GetRegisterAsInteger(
|
||||||
regs.GetRegisterAsInteger(instr.gpr20, 0, instr.conversion.is_input_signed);
|
instr.gpr20, 0, instr.conversion.is_input_signed, instr.conversion.src_size);
|
||||||
|
|
||||||
if (instr.conversion.abs_a) {
|
if (instr.conversion.abs_a) {
|
||||||
op_a = "abs(" + op_a + ')';
|
op_a = "abs(" + op_a + ')';
|
||||||
|
@ -1145,6 +1170,8 @@ private:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case OpCode::Id::F2F_R: {
|
case OpCode::Id::F2F_R: {
|
||||||
|
ASSERT_MSG(instr.conversion.dest_size == Register::Size::Word, "Unimplemented");
|
||||||
|
ASSERT_MSG(instr.conversion.src_size == Register::Size::Word, "Unimplemented");
|
||||||
std::string op_a = regs.GetRegisterAsFloat(instr.gpr20);
|
std::string op_a = regs.GetRegisterAsFloat(instr.gpr20);
|
||||||
|
|
||||||
switch (instr.conversion.f2f.rounding) {
|
switch (instr.conversion.f2f.rounding) {
|
||||||
|
@ -1174,6 +1201,7 @@ private:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case OpCode::Id::F2I_R: {
|
case OpCode::Id::F2I_R: {
|
||||||
|
ASSERT_MSG(instr.conversion.src_size == Register::Size::Word, "Unimplemented");
|
||||||
std::string op_a = regs.GetRegisterAsFloat(instr.gpr20);
|
std::string op_a = regs.GetRegisterAsFloat(instr.gpr20);
|
||||||
|
|
||||||
if (instr.conversion.abs_a) {
|
if (instr.conversion.abs_a) {
|
||||||
|
@ -1206,7 +1234,7 @@ private:
|
||||||
}
|
}
|
||||||
|
|
||||||
regs.SetRegisterToInteger(instr.gpr0, instr.conversion.is_output_signed, 0, op_a, 1,
|
regs.SetRegisterToInteger(instr.gpr0, instr.conversion.is_output_signed, 0, op_a, 1,
|
||||||
1);
|
1, false, 0, instr.conversion.dest_size);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
default: {
|
default: {
|
||||||
|
|
Loading…
Reference in a new issue