mirror of
https://git.suyu.dev/suyu/suyu
synced 2025-01-09 16:03:21 +00:00
51 lines
1.9 KiB
C++
51 lines
1.9 KiB
C++
// Copyright 2021 yuzu Emulator Project
|
|
// Licensed under GPLv2 or any later version
|
|
// Refer to the license.txt file included.
|
|
|
|
#include "shader_recompiler/environment.h"
|
|
#include "shader_recompiler/frontend/ir/basic_block.h"
|
|
#include "shader_recompiler/frontend/maxwell/decode.h"
|
|
#include "shader_recompiler/frontend/maxwell/location.h"
|
|
#include "shader_recompiler/frontend/maxwell/translate/impl/impl.h"
|
|
#include "shader_recompiler/frontend/maxwell/translate/translate.h"
|
|
|
|
namespace Shader::Maxwell {
|
|
|
|
template <auto method>
|
|
static void Invoke(TranslatorVisitor& visitor, Location pc, u64 insn) {
|
|
using MethodType = decltype(method);
|
|
if constexpr (std::is_invocable_r_v<void, MethodType, TranslatorVisitor&, Location, u64>) {
|
|
(visitor.*method)(pc, insn);
|
|
} else if constexpr (std::is_invocable_r_v<void, MethodType, TranslatorVisitor&, u64>) {
|
|
(visitor.*method)(insn);
|
|
} else {
|
|
(visitor.*method)();
|
|
}
|
|
}
|
|
|
|
IR::Block Translate(ObjectPool<IR::Inst>& inst_pool, Environment& env,
|
|
const Flow::Block& flow_block) {
|
|
IR::Block block{inst_pool, flow_block.begin.Offset(), flow_block.end.Offset()};
|
|
TranslatorVisitor visitor{env, block};
|
|
|
|
const Location pc_end{flow_block.end};
|
|
Location pc{flow_block.begin};
|
|
while (pc != pc_end) {
|
|
const u64 insn{env.ReadInstruction(pc.Offset())};
|
|
const Opcode opcode{Decode(insn)};
|
|
switch (opcode) {
|
|
#define INST(name, cute, mask) \
|
|
case Opcode::name: \
|
|
Invoke<&TranslatorVisitor::name>(visitor, pc, insn); \
|
|
break;
|
|
#include "shader_recompiler/frontend/maxwell/maxwell.inc"
|
|
#undef OPCODE
|
|
default:
|
|
throw LogicError("Invalid opcode {}", opcode);
|
|
}
|
|
++pc;
|
|
}
|
|
return block;
|
|
}
|
|
|
|
} // namespace Shader::Maxwell
|