mirror of
https://git.suyu.dev/suyu/suyu.git
synced 2025-01-14 23:34:07 +00:00
shader_decode: Implement TEXS.F16
This commit is contained in:
parent
c68c13e1aa
commit
5af82a8ed4
3 changed files with 57 additions and 15 deletions
|
@ -219,8 +219,7 @@ u32 ShaderIR::DecodeMemory(BasicBlock& bb, u32 pc) {
|
||||||
if (instr.texs.fp32_flag) {
|
if (instr.texs.fp32_flag) {
|
||||||
WriteTexsInstructionFloat(bb, instr, texture);
|
WriteTexsInstructionFloat(bb, instr, texture);
|
||||||
} else {
|
} else {
|
||||||
UNIMPLEMENTED();
|
WriteTexsInstructionHalfFloat(bb, instr, texture);
|
||||||
// WriteTexsInstructionHalfFloat(bb, instr, texture);
|
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -416,39 +415,52 @@ const Sampler& ShaderIR::GetSampler(const Tegra::Shader::Sampler& sampler, Textu
|
||||||
return *used_samplers.emplace(entry).first;
|
return *used_samplers.emplace(entry).first;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ShaderIR::WriteTexsInstructionFloat(BasicBlock& bb, Tegra::Shader::Instruction instr,
|
void ShaderIR::WriteTexsInstructionFloat(BasicBlock& bb, Instruction instr, Node texture) {
|
||||||
Node texture) {
|
|
||||||
// TEXS has two destination registers and a swizzle. The first two elements in the swizzle
|
// TEXS has two destination registers and a swizzle. The first two elements in the swizzle
|
||||||
// go into gpr0+0 and gpr0+1, and the rest goes into gpr28+0 and gpr28+1
|
// go into gpr0+0 and gpr0+1, and the rest goes into gpr28+0 and gpr28+1
|
||||||
|
|
||||||
MetaComponents meta;
|
MetaComponents meta;
|
||||||
std::array<Node, 4> dest;
|
std::array<Node, 4> dest;
|
||||||
|
|
||||||
std::size_t written_components = 0;
|
|
||||||
for (u32 component = 0; component < 4; ++component) {
|
for (u32 component = 0; component < 4; ++component) {
|
||||||
if (!instr.texs.IsComponentEnabled(component)) {
|
if (!instr.texs.IsComponentEnabled(component)) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
meta.components_map[written_components] = static_cast<u32>(component);
|
meta.components_map[meta.count] = component;
|
||||||
|
|
||||||
if (written_components < 2) {
|
if (meta.count < 2) {
|
||||||
// Write the first two swizzle components to gpr0 and gpr0+1
|
// Write the first two swizzle components to gpr0 and gpr0+1
|
||||||
dest[written_components] = GetRegister(instr.gpr0.Value() + written_components % 2);
|
dest[meta.count] = GetRegister(instr.gpr0.Value() + meta.count % 2);
|
||||||
} else {
|
} else {
|
||||||
ASSERT(instr.texs.HasTwoDestinations());
|
ASSERT(instr.texs.HasTwoDestinations());
|
||||||
// Write the rest of the swizzle components to gpr28 and gpr28+1
|
// Write the rest of the swizzle components to gpr28 and gpr28+1
|
||||||
dest[written_components] = GetRegister(instr.gpr28.Value() + written_components % 2);
|
dest[meta.count] = GetRegister(instr.gpr28.Value() + meta.count % 2);
|
||||||
}
|
}
|
||||||
|
++meta.count;
|
||||||
++written_components;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
std::generate(dest.begin() + written_components, dest.end(), [&]() { return GetRegister(RZ); });
|
std::generate(dest.begin() + meta.count, dest.end(), [&]() { return GetRegister(RZ); });
|
||||||
|
|
||||||
bb.push_back(Operation(OperationCode::AssignComposite, meta, texture, dest[0], dest[1], dest[2],
|
bb.push_back(Operation(OperationCode::AssignComposite, meta, texture, dest[0], dest[1], dest[2],
|
||||||
dest[3]));
|
dest[3]));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ShaderIR::WriteTexsInstructionHalfFloat(BasicBlock& bb, Instruction instr, Node texture) {
|
||||||
|
// TEXS.F16 destionation registers are packed in two registers in pairs (just like any half
|
||||||
|
// float instruction).
|
||||||
|
|
||||||
|
MetaComponents meta;
|
||||||
|
for (u32 component = 0; component < 4; ++component) {
|
||||||
|
if (!instr.texs.IsComponentEnabled(component))
|
||||||
|
continue;
|
||||||
|
meta.components_map[meta.count++] = component;
|
||||||
|
}
|
||||||
|
if (meta.count == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
bb.push_back(Operation(OperationCode::AssignCompositeHalf, meta, texture,
|
||||||
|
GetRegister(instr.gpr0), GetRegister(instr.gpr28)));
|
||||||
|
}
|
||||||
|
|
||||||
Node ShaderIR::GetTextureCode(Instruction instr, TextureType texture_type,
|
Node ShaderIR::GetTextureCode(Instruction instr, TextureType texture_type,
|
||||||
TextureProcessMode process_mode, bool depth_compare, bool is_array,
|
TextureProcessMode process_mode, bool depth_compare, bool is_array,
|
||||||
std::size_t array_offset, std::size_t bias_offset,
|
std::size_t array_offset, std::size_t bias_offset,
|
||||||
|
|
|
@ -785,6 +785,31 @@ private:
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::string AssignCompositeHalf(Operation operation) {
|
||||||
|
const auto& meta = std::get<MetaComponents>(operation.GetMeta());
|
||||||
|
|
||||||
|
const std::string composite = code.GenerateTemporal();
|
||||||
|
code.AddLine("vec4 " + composite + " = " + Visit(operation[0]) + ';');
|
||||||
|
|
||||||
|
const auto ReadComponent = [&](u32 component) {
|
||||||
|
if (component < meta.count) {
|
||||||
|
return composite + '[' + std::to_string(meta.GetSourceComponent(component)) + ']';
|
||||||
|
}
|
||||||
|
return std::string("0");
|
||||||
|
};
|
||||||
|
|
||||||
|
const auto dst1 = std::get<GprNode>(*operation[1]).GetIndex();
|
||||||
|
const std::string src1 = "vec2(" + ReadComponent(0) + ", " + ReadComponent(1) + ')';
|
||||||
|
code.AddLine(GetRegister(dst1) + " = utof(packHalf2x16(" + src1 + "))");
|
||||||
|
|
||||||
|
if (meta.count > 2) {
|
||||||
|
const auto dst2 = std::get<GprNode>(*operation[2]).GetIndex();
|
||||||
|
const std::string src2 = "vec2(" + ReadComponent(2) + ", " + ReadComponent(3) + ')';
|
||||||
|
code.AddLine(GetRegister(dst2) + " = utof(packHalf2x16(" + src2 + "))");
|
||||||
|
}
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
std::string Composite(Operation operation) {
|
std::string Composite(Operation operation) {
|
||||||
std::string value = "vec4(";
|
std::string value = "vec4(";
|
||||||
for (std::size_t i = 0; i < 4; ++i) {
|
for (std::size_t i = 0; i < 4; ++i) {
|
||||||
|
@ -1302,6 +1327,7 @@ private:
|
||||||
static constexpr OperationDecompilersArray operation_decompilers = {
|
static constexpr OperationDecompilersArray operation_decompilers = {
|
||||||
&GLSLDecompiler::Assign,
|
&GLSLDecompiler::Assign,
|
||||||
&GLSLDecompiler::AssignComposite,
|
&GLSLDecompiler::AssignComposite,
|
||||||
|
&GLSLDecompiler::AssignCompositeHalf,
|
||||||
|
|
||||||
&GLSLDecompiler::Composite,
|
&GLSLDecompiler::Composite,
|
||||||
&GLSLDecompiler::Select,
|
&GLSLDecompiler::Select,
|
||||||
|
|
|
@ -44,8 +44,9 @@ constexpr u32 MAX_PROGRAM_LENGTH = 0x1000;
|
||||||
constexpr u32 RZ = 0xff;
|
constexpr u32 RZ = 0xff;
|
||||||
|
|
||||||
enum class OperationCode {
|
enum class OperationCode {
|
||||||
Assign, /// (float& dest, float src) -> void
|
Assign, /// (float& dest, float src) -> void
|
||||||
AssignComposite, /// (MetaComponents, float4 src, float&[4] dst) -> void
|
AssignComposite, /// (MetaComponents, float4 src, float&[4] dst) -> void
|
||||||
|
AssignCompositeHalf, /// (MetaComponents, float4 src, float&[2] dst) -> void
|
||||||
|
|
||||||
Composite, /// (float[4] values) -> float4
|
Composite, /// (float[4] values) -> float4
|
||||||
Select, /// (MetaArithmetic, bool pred, float a, float b) -> float
|
Select, /// (MetaArithmetic, bool pred, float a, float b) -> float
|
||||||
|
@ -279,6 +280,7 @@ struct MetaTexture {
|
||||||
|
|
||||||
struct MetaComponents {
|
struct MetaComponents {
|
||||||
std::array<u32, 4> components_map{};
|
std::array<u32, 4> components_map{};
|
||||||
|
u32 count{};
|
||||||
|
|
||||||
u32 GetSourceComponent(u32 dest_index) const {
|
u32 GetSourceComponent(u32 dest_index) const {
|
||||||
return components_map[dest_index];
|
return components_map[dest_index];
|
||||||
|
@ -692,6 +694,8 @@ private:
|
||||||
Tegra::Shader::TextureType type, bool is_array, bool is_shadow);
|
Tegra::Shader::TextureType type, bool is_array, bool is_shadow);
|
||||||
|
|
||||||
void WriteTexsInstructionFloat(BasicBlock& bb, Tegra::Shader::Instruction instr, Node texture);
|
void WriteTexsInstructionFloat(BasicBlock& bb, Tegra::Shader::Instruction instr, Node texture);
|
||||||
|
void WriteTexsInstructionHalfFloat(BasicBlock& bb, Tegra::Shader::Instruction instr,
|
||||||
|
Node texture);
|
||||||
|
|
||||||
Node GetTexCode(Tegra::Shader::Instruction instr, Tegra::Shader::TextureType texture_type,
|
Node GetTexCode(Tegra::Shader::Instruction instr, Tegra::Shader::TextureType texture_type,
|
||||||
Tegra::Shader::TextureProcessMode process_mode, bool depth_compare,
|
Tegra::Shader::TextureProcessMode process_mode, bool depth_compare,
|
||||||
|
|
Loading…
Reference in a new issue