mirror of
https://git.suyu.dev/suyu/dynarmic.git
synced 2026-01-07 23:18:10 +01:00
A64: Implement compare and branch
This commit is contained in:
parent
e8bcf72ee5
commit
cb481a3a48
20 changed files with 249 additions and 21 deletions
|
|
@ -117,10 +117,10 @@ std::vector<Matcher<V>> GetDecodeTable() {
|
|||
INST(&V::BL, "BL", "100101iiiiiiiiiiiiiiiiiiiiiiiiii"),
|
||||
|
||||
// Compare and branch (immediate)
|
||||
//INST(&V::CBZ, "CBZ", "z0110100iiiiiiiiiiiiiiiiiiittttt"),
|
||||
//INST(&V::CBNZ, "CBNZ", "z0110101iiiiiiiiiiiiiiiiiiittttt"),
|
||||
//INST(&V::TBZ, "TBZ", "b0110110bbbbbiiiiiiiiiiiiiittttt"),
|
||||
//INST(&V::TBNZ, "TBNZ", "b0110111bbbbbiiiiiiiiiiiiiittttt"),
|
||||
INST(&V::CBZ, "CBZ", "z0110100iiiiiiiiiiiiiiiiiiittttt"),
|
||||
INST(&V::CBNZ, "CBNZ", "z0110101iiiiiiiiiiiiiiiiiiittttt"),
|
||||
INST(&V::TBZ, "TBZ", "b0110110bbbbbiiiiiiiiiiiiiittttt"),
|
||||
INST(&V::TBNZ, "TBNZ", "b0110111bbbbbiiiiiiiiiiiiiittttt"),
|
||||
|
||||
// Loads and stores - Advanced SIMD Load/Store multiple structures
|
||||
//INST(&V::ST4_mult_1, "ST4 (multiple structures)", "0Q001100000000000000zznnnnnttttt"),
|
||||
|
|
|
|||
|
|
@ -22,6 +22,10 @@ u64 IREmitter::AlignPC(size_t alignment) {
|
|||
return static_cast<u64>(pc - pc % alignment);
|
||||
}
|
||||
|
||||
void IREmitter::SetCheckBit(const IR::U1& value) {
|
||||
Inst(Opcode::A64SetCheckBit, value);
|
||||
}
|
||||
|
||||
IR::U1 IREmitter::GetCFlag() {
|
||||
return Inst<IR::U1>(Opcode::A64GetCFlag);
|
||||
}
|
||||
|
|
|
|||
|
|
@ -31,6 +31,7 @@ public:
|
|||
u64 PC();
|
||||
u64 AlignPC(size_t alignment);
|
||||
|
||||
void SetCheckBit(const IR::U1& value);
|
||||
IR::U1 GetCFlag();
|
||||
void SetNZCV(const IR::NZCV& nzcv);
|
||||
|
||||
|
|
|
|||
|
|
@ -62,5 +62,67 @@ bool TranslatorVisitor::RET(Reg Rn) {
|
|||
return false;
|
||||
}
|
||||
|
||||
bool TranslatorVisitor::CBZ(bool sf, Imm<19> imm19, Reg Rt) {
|
||||
size_t datasize = sf ? 64 : 32;
|
||||
s64 offset = concatenate(imm19, Imm<2>{0}).SignExtend<s64>();
|
||||
|
||||
auto operand1 = X(datasize, Rt);
|
||||
|
||||
ir.SetCheckBit(ir.IsZero(operand1));
|
||||
|
||||
u64 target = ir.PC() + offset;
|
||||
auto cond_pass = IR::Term::LinkBlock{ir.current_location.SetPC(target)};
|
||||
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) {
|
||||
size_t datasize = sf ? 64 : 32;
|
||||
s64 offset = concatenate(imm19, Imm<2>{0}).SignExtend<s64>();
|
||||
|
||||
auto operand1 = X(datasize, Rt);
|
||||
|
||||
ir.SetCheckBit(ir.IsZero(operand1));
|
||||
|
||||
u64 target = ir.PC() + offset;
|
||||
auto cond_pass = IR::Term::LinkBlock{ir.current_location.AdvancePC(4)};
|
||||
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) {
|
||||
size_t datasize = b5 == 1 ? 64 : 32;
|
||||
size_t bit_pos = concatenate(b5, b40).ZeroExtend<size_t>();
|
||||
s64 offset = concatenate(imm14, Imm<2>{0}).SignExtend<s64>();
|
||||
|
||||
auto operand = X(datasize, Rt);
|
||||
|
||||
ir.SetCheckBit(ir.TestBit(operand, ir.Imm8(bit_pos)));
|
||||
|
||||
u64 target = ir.PC() + offset;
|
||||
auto cond_1 = IR::Term::LinkBlock{ir.current_location.AdvancePC(4)};
|
||||
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) {
|
||||
size_t datasize = b5 == 1 ? 64 : 32;
|
||||
size_t bit_pos = concatenate(b5, b40).ZeroExtend<size_t>();
|
||||
s64 offset = concatenate(imm14, Imm<2>{0}).SignExtend<s64>();
|
||||
|
||||
auto operand = X(datasize, Rt);
|
||||
|
||||
ir.SetCheckBit(ir.TestBit(operand, ir.Imm8(bit_pos)));
|
||||
|
||||
u64 target = ir.PC() + offset;
|
||||
auto cond_1 = IR::Term::LinkBlock{ir.current_location.SetPC(target)};
|
||||
auto cond_0 = IR::Term::LinkBlock{ir.current_location.AdvancePC(4)};
|
||||
ir.SetTerm(IR::Term::CheckBit{cond_1, cond_0});
|
||||
return false;
|
||||
}
|
||||
|
||||
} // namespace A64
|
||||
} // namespace Dynarmic
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue