aboutsummaryrefslogtreecommitdiff
path: root/externals/dynarmic/src/dynarmic/frontend/A64/translate/impl/a64_branch.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'externals/dynarmic/src/dynarmic/frontend/A64/translate/impl/a64_branch.cpp')
-rw-r--r--externals/dynarmic/src/dynarmic/frontend/A64/translate/impl/a64_branch.cpp128
1 files changed, 128 insertions, 0 deletions
diff --git a/externals/dynarmic/src/dynarmic/frontend/A64/translate/impl/a64_branch.cpp b/externals/dynarmic/src/dynarmic/frontend/A64/translate/impl/a64_branch.cpp
new file mode 100644
index 0000000000..01cc1390c7
--- /dev/null
+++ b/externals/dynarmic/src/dynarmic/frontend/A64/translate/impl/a64_branch.cpp
@@ -0,0 +1,128 @@
+/* This file is part of the dynarmic project.
+ * Copyright (c) 2018 MerryMage
+ * SPDX-License-Identifier: 0BSD
+ */
+
+#include "dynarmic/frontend/A64/translate/impl/impl.h"
+
+namespace Dynarmic::A64 {
+
+bool TranslatorVisitor::B_cond(Imm<19> imm19, Cond cond) {
+ const s64 offset = concatenate(imm19, Imm<2>{0}).SignExtend<s64>();
+ const u64 target = ir.PC() + offset;
+
+ const auto cond_pass = IR::Term::LinkBlock{ir.current_location->SetPC(target)};
+ const auto cond_fail = IR::Term::LinkBlock{ir.current_location->AdvancePC(4)};
+ ir.SetTerm(IR::Term::If{cond, cond_pass, cond_fail});
+ return false;
+}
+
+bool TranslatorVisitor::B_uncond(Imm<26> imm26) {
+ const s64 offset = concatenate(imm26, Imm<2>{0}).SignExtend<s64>();
+ const u64 target = ir.PC() + offset;
+
+ ir.SetTerm(IR::Term::LinkBlock{ir.current_location->SetPC(target)});
+ return false;
+}
+
+bool TranslatorVisitor::BL(Imm<26> imm26) {
+ const s64 offset = concatenate(imm26, Imm<2>{0}).SignExtend<s64>();
+
+ X(64, Reg::R30, ir.Imm64(ir.PC() + 4));
+ ir.PushRSB(ir.current_location->AdvancePC(4));
+
+ const u64 target = ir.PC() + offset;
+ ir.SetTerm(IR::Term::LinkBlock{ir.current_location->SetPC(target)});
+ return false;
+}
+
+bool TranslatorVisitor::BLR(Reg Rn) {
+ const auto target = X(64, Rn);
+
+ X(64, Reg::R30, ir.Imm64(ir.PC() + 4));
+ ir.PushRSB(ir.current_location->AdvancePC(4));
+
+ ir.SetPC(target);
+ ir.SetTerm(IR::Term::FastDispatchHint{});
+ return false;
+}
+
+bool TranslatorVisitor::BR(Reg Rn) {
+ const auto target = X(64, Rn);
+
+ ir.SetPC(target);
+ ir.SetTerm(IR::Term::FastDispatchHint{});
+ return false;
+}
+
+bool TranslatorVisitor::RET(Reg Rn) {
+ const auto target = X(64, Rn);
+
+ ir.SetPC(target);
+ ir.SetTerm(IR::Term::PopRSBHint{});
+ return false;
+}
+
+bool TranslatorVisitor::CBZ(bool sf, Imm<19> imm19, Reg Rt) {
+ const size_t datasize = sf ? 64 : 32;
+ const s64 offset = concatenate(imm19, Imm<2>{0}).SignExtend<s64>();
+
+ const IR::U32U64 operand1 = X(datasize, Rt);
+
+ ir.SetCheckBit(ir.IsZero(operand1));
+
+ const u64 target = ir.PC() + offset;
+ const auto cond_pass = IR::Term::LinkBlock{ir.current_location->SetPC(target)};
+ const auto cond_fail = IR::Term::LinkBlock{ir.current_location->AdvancePC(4)};
+ ir.SetTerm(IR::Term::CheckBit{cond_pass, cond_fail});
+ return false;
+}
+
+bool TranslatorVisitor::CBNZ(bool sf, Imm<19> imm19, Reg Rt) {
+ const size_t datasize = sf ? 64 : 32;
+ const s64 offset = concatenate(imm19, Imm<2>{0}).SignExtend<s64>();
+
+ const IR::U32U64 operand1 = X(datasize, Rt);
+
+ ir.SetCheckBit(ir.IsZero(operand1));
+
+ const u64 target = ir.PC() + offset;
+ const auto cond_pass = IR::Term::LinkBlock{ir.current_location->AdvancePC(4)};
+ const auto cond_fail = IR::Term::LinkBlock{ir.current_location->SetPC(target)};
+ ir.SetTerm(IR::Term::CheckBit{cond_pass, cond_fail});
+ return false;
+}
+
+bool TranslatorVisitor::TBZ(Imm<1> b5, Imm<5> b40, Imm<14> imm14, Reg Rt) {
+ const size_t datasize = b5 == 1 ? 64 : 32;
+ const u8 bit_pos = concatenate(b5, b40).ZeroExtend<u8>();
+ const s64 offset = concatenate(imm14, Imm<2>{0}).SignExtend<s64>();
+
+ const auto operand = X(datasize, Rt);
+
+ ir.SetCheckBit(ir.TestBit(operand, ir.Imm8(bit_pos)));
+
+ const u64 target = ir.PC() + offset;
+ const auto cond_1 = IR::Term::LinkBlock{ir.current_location->AdvancePC(4)};
+ const auto cond_0 = IR::Term::LinkBlock{ir.current_location->SetPC(target)};
+ ir.SetTerm(IR::Term::CheckBit{cond_1, cond_0});
+ return false;
+}
+
+bool TranslatorVisitor::TBNZ(Imm<1> b5, Imm<5> b40, Imm<14> imm14, Reg Rt) {
+ const size_t datasize = b5 == 1 ? 64 : 32;
+ const u8 bit_pos = concatenate(b5, b40).ZeroExtend<u8>();
+ const s64 offset = concatenate(imm14, Imm<2>{0}).SignExtend<s64>();
+
+ const auto operand = X(datasize, Rt);
+
+ ir.SetCheckBit(ir.TestBit(operand, ir.Imm8(bit_pos)));
+
+ const u64 target = ir.PC() + offset;
+ const auto cond_1 = IR::Term::LinkBlock{ir.current_location->SetPC(target)};
+ const auto cond_0 = IR::Term::LinkBlock{ir.current_location->AdvancePC(4)};
+ ir.SetTerm(IR::Term::CheckBit{cond_1, cond_0});
+ return false;
+}
+
+} // namespace Dynarmic::A64