[D3D12] HLSL shader translator loads
This commit is contained in:
parent
d60c8b29ca
commit
11e4ae3f90
|
@ -9,6 +9,8 @@
|
||||||
|
|
||||||
#include "xenia/gpu/hlsl_shader_translator.h"
|
#include "xenia/gpu/hlsl_shader_translator.h"
|
||||||
|
|
||||||
|
#include "xenia/base/assert.h"
|
||||||
|
|
||||||
namespace xe {
|
namespace xe {
|
||||||
namespace gpu {
|
namespace gpu {
|
||||||
using namespace ucode;
|
using namespace ucode;
|
||||||
|
@ -69,6 +71,7 @@ void HlslShaderTranslator::StartTranslation() {
|
||||||
EmitSource(
|
EmitSource(
|
||||||
"cbuffer xe_system_constants : register(b0) {\n"
|
"cbuffer xe_system_constants : register(b0) {\n"
|
||||||
" float2 xe_viewport_inv_scale;\n"
|
" float2 xe_viewport_inv_scale;\n"
|
||||||
|
" uint xe_textures_are_3d;\n"
|
||||||
"}\n"
|
"}\n"
|
||||||
"\n"
|
"\n"
|
||||||
"struct XeFloatConstantPage {\n"
|
"struct XeFloatConstantPage {\n"
|
||||||
|
@ -158,8 +161,11 @@ void HlslShaderTranslator::StartTranslation() {
|
||||||
}
|
}
|
||||||
|
|
||||||
EmitSource(
|
EmitSource(
|
||||||
// Float constant index for dynamic indexing.
|
// Dynamic index for source operands (mainly for float and bool constants
|
||||||
" uint xe_float_constant_index;\n"
|
// since they are indexed in two parts).
|
||||||
|
" uint xe_src_index;\n"
|
||||||
|
// Sources for instructions.
|
||||||
|
" float4 xe_src0, xe_src1, xe_src2;\n"
|
||||||
// Previous vector result (used as a scratch).
|
// Previous vector result (used as a scratch).
|
||||||
" float4 xe_pv;\n"
|
" float4 xe_pv;\n"
|
||||||
// Previous scalar result (used for RETAIN_PREV).
|
// Previous scalar result (used for RETAIN_PREV).
|
||||||
|
@ -303,7 +309,7 @@ void HlslShaderTranslator::ProcessLoopEndInstruction(
|
||||||
Indent();
|
Indent();
|
||||||
|
|
||||||
// Still looping. Adjust index and jump back to body.
|
// Still looping. Adjust index and jump back to body.
|
||||||
EmitSourceDepth("xe_aL.x += int((xe_loop_constants[%u] << 8u) >> 24u);\n",
|
EmitSourceDepth("xe_aL.x += int(xe_loop_constants[%u] << 8u) >> 24;\n",
|
||||||
instr.loop_constant_index);
|
instr.loop_constant_index);
|
||||||
EmitSourceDepth("xe_pc = %u; // Loop back to body L%u\n",
|
EmitSourceDepth("xe_pc = %u; // Loop back to body L%u\n",
|
||||||
instr.loop_body_address, instr.loop_body_address);
|
instr.loop_body_address, instr.loop_body_address);
|
||||||
|
@ -374,6 +380,128 @@ void HlslShaderTranslator::ProcessAllocInstruction(
|
||||||
instr.Disassemble(&source_);
|
instr.Disassemble(&source_);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool HlslShaderTranslator::BeginPredicatedInstruction(
|
||||||
|
bool is_predicated, bool predicate_condition) {
|
||||||
|
if (is_predicated &&
|
||||||
|
(!cf_exec_pred_ || cf_exec_pred_cond_ != predicate_condition)) {
|
||||||
|
EmitSourceDepth("if (%cxe_p0) {\n", instr.predicate_condition ? ' ' : '!');
|
||||||
|
Indent();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void HlslShaderTranslator::EndPredicatedInstruction(bool conditional_emitted) {
|
||||||
|
if (conditional_emitted) {
|
||||||
|
Unindent();
|
||||||
|
EmitSourceDepth("}\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void HlslShaderTranslator::EmitLoadOperand(uint32_t src_index,
|
||||||
|
const InstructionOperand& op) {
|
||||||
|
// If indexing dynamically, emit the index because float and bool constants
|
||||||
|
// need to be indexed in two parts.
|
||||||
|
// Also verify we are not using vertex/texture fetch constants here.
|
||||||
|
uint32_t storage_index_max;
|
||||||
|
switch (op.storage_source) {
|
||||||
|
case InstructionStorageSource::kRegister:
|
||||||
|
index_max = 127;
|
||||||
|
break;
|
||||||
|
case InstructionStorageSource::kConstantFloat:
|
||||||
|
case InstructionStorageSource::kConstantBool:
|
||||||
|
index_max = 255;
|
||||||
|
break;
|
||||||
|
case InstructionStorageSource::kConstantInt:
|
||||||
|
index_max = 31;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert_always();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (op.storage_addressing_mode ==
|
||||||
|
InstructionStorageAddressingMode::kAddressAbsolute) {
|
||||||
|
EmitSourceDepth("xe_src_index = uint(%u + xe_a0) & %uu;\n",
|
||||||
|
op.storage_index, storage_index_max);
|
||||||
|
} else if (op.storage_addressing_mode ==
|
||||||
|
InstructionStorageAddressingMode::kAddressRelative) {
|
||||||
|
EmitSourceDepth("xe_src_index = uint(%u + xe_aL.x) & %uu;\n",
|
||||||
|
op.storage_index, storage_index_max);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Negation and abs are store modifiers, so they're applied after swizzling.
|
||||||
|
EmitSourceDepth("xe_src%u = ", src_index);
|
||||||
|
if (op.is_negated) {
|
||||||
|
EmitSource("-");
|
||||||
|
}
|
||||||
|
if (op.is_absolute_value) {
|
||||||
|
EmitSource("abs");
|
||||||
|
}
|
||||||
|
EmitSource("(");
|
||||||
|
|
||||||
|
if (op.storage_addressing_mode == InstructionStorageAddressingMode::kStatic) {
|
||||||
|
switch (op.storage_source) {
|
||||||
|
case InstructionStorageSource::kRegister:
|
||||||
|
EmitSource("xe_r[%u]", op.storage_index);
|
||||||
|
break;
|
||||||
|
case InstructionStorageSource::kConstantFloat:
|
||||||
|
EmitSource("xe_float_constants[%u].c[%u]", op.storage_index >> 4,
|
||||||
|
op.storage_index & 15);
|
||||||
|
break;
|
||||||
|
case InstructionStorageSource::kConstantInt:
|
||||||
|
EmitSource("xe_loop_constants[%u]", op.storage_index);
|
||||||
|
break;
|
||||||
|
case InstructionStorageSource::kConstantBool:
|
||||||
|
EmitSource("float((xe_bool_constants[%u] >> %uu) & 1u)",
|
||||||
|
op.storage_index >> 5, op.storage_index & 31);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert_always();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
switch (op.storage_source) {
|
||||||
|
case InstructionStorageSource::kRegister:
|
||||||
|
EmitSource("xe_r[xe_src_index]");
|
||||||
|
break;
|
||||||
|
case InstructionStorageSource::kConstantFloat:
|
||||||
|
EmitSource(
|
||||||
|
"xe_float_constants[xe_src_index >> 4u].c[xe_src_index & 15u]");
|
||||||
|
break;
|
||||||
|
case InstructionStorageSource::kConstantInt:
|
||||||
|
EmitSource("xe_loop_constants[xe_src_index]");
|
||||||
|
break;
|
||||||
|
case InstructionStorageSource::kConstantBool:
|
||||||
|
EmitSource("float((xe_bool_constants[xe_src_index >> 5u] >> "
|
||||||
|
"(xe_src_index & 31u)) & 1u)");
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert_always();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
EmitSource(")");
|
||||||
|
// Integer and bool constants are scalar, can't swizzle them.
|
||||||
|
if (op.storage_source == InstructionStorageSource::kConstantInt ||
|
||||||
|
op.storage_source == InstructionStorageSource::kConstantBool) {
|
||||||
|
EmitSource(".xxxx");
|
||||||
|
} else {
|
||||||
|
if (!op.is_standard_swizzle()) {
|
||||||
|
EmitSource(".");
|
||||||
|
// For 1 component stores it will be .aaaa, for 2 components it's .abbb.
|
||||||
|
for (int i = 0; i < op.component_count; ++i) {
|
||||||
|
EmitSource("%c", GetCharForSwizzle(op.components[i]));
|
||||||
|
}
|
||||||
|
for (int i = op.component_count; i < 4; ++i) {
|
||||||
|
EmitSource("%c",
|
||||||
|
GetCharForSwizzle(op.components[op.component_count - 1]));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
EmitSource(";\n");
|
||||||
|
}
|
||||||
|
|
||||||
uint32_t HlslShaderTranslator::AddSRVBinding(SRVType type,
|
uint32_t HlslShaderTranslator::AddSRVBinding(SRVType type,
|
||||||
uint32_t fetch_constant) {
|
uint32_t fetch_constant) {
|
||||||
for (uint32_t i = 0; i < srv_bindings_.size(); ++i) {
|
for (uint32_t i = 0; i < srv_bindings_.size(); ++i) {
|
||||||
|
|
|
@ -66,6 +66,12 @@ class HlslShaderTranslator : public ShaderTranslator {
|
||||||
void Indent();
|
void Indent();
|
||||||
void Unindent();
|
void Unindent();
|
||||||
|
|
||||||
|
// Returns whether a new conditional was emitted.
|
||||||
|
bool BeginPredicatedInstruction(bool is_predicated, bool predicate_condition);
|
||||||
|
void EndPredicatedInstruction(bool conditional_emitted);
|
||||||
|
|
||||||
|
void EmitLoadOperand(uint32_t src_index, const InstructionOperand& op);
|
||||||
|
|
||||||
StringBuffer source_;
|
StringBuffer source_;
|
||||||
uint32_t depth_ = 0;
|
uint32_t depth_ = 0;
|
||||||
char depth_prefix_[16] = {0};
|
char depth_prefix_[16] = {0};
|
||||||
|
|
Loading…
Reference in New Issue