1
0
mirror of https://github.com/0O0o0oOoO00/Alas.git synced 2026-05-20 03:29:29 +08:00
Files
Alas/blcrack/cracker/Dobby/source/InstructionRelocation/arm/InstructionRelocationARM.h
2025-11-01 00:23:46 +08:00

308 lines
8.1 KiB
C++

#pragma once
#include "dobby/dobby_internal.h"
#include "core/arch/arm/constants-arm.h"
#include "core/assembler/assembler-arm.h"
namespace zz {
namespace arm {
// thumb1/thumb2 pseudo label type, only support Thumb1-Ldr | Thumb2-Ldr
enum ref_label_type_t { kThumb1Ldr, kThumb2LiteralLdr };
// custom thumb pseudo label for thumb/thumb2
class ThumbPseudoLabel : public AssemblerPseudoLabel {
public:
ThumbPseudoLabel(addr_t addr) : AssemblerPseudoLabel(addr) {
}
// fix the instruction which not link to the label yet.
void link_confused_instructions(CodeBuffer *buffer) {
CodeBuffer *_buffer;
if (buffer)
_buffer = buffer;
for (auto &ref_label_insn : ref_label_insns_) {
// instruction offset to label
thumb2_inst_t insn = _buffer->LoadThumb2Inst(ref_label_insn.pc_offset);
thumb1_inst_t insn1 = _buffer->LoadThumb1Inst(ref_label_insn.pc_offset);
thumb1_inst_t insn2 = _buffer->LoadThumb1Inst(ref_label_insn.pc_offset + sizeof(thumb1_inst_t));
switch (ref_label_insn.link_type) {
case kThumb1Ldr: {
UNREACHABLE();
} break;
case kThumb2LiteralLdr: {
int64_t pc = ref_label_insn.pc_offset + Thumb_PC_OFFSET;
assert(pc % 4 == 0);
int32_t imm12 = pos() - pc;
if (imm12 > 0) {
set_bit(insn1, 7, 1);
} else {
set_bit(insn1, 7, 0);
imm12 = -imm12;
}
set_bits(insn2, 0, 11, imm12);
_buffer->RewriteThumb1Inst(ref_label_insn.pc_offset, insn1);
_buffer->RewriteThumb1Inst(ref_label_insn.pc_offset + Thumb1_INST_LEN, insn2);
DEBUG_LOG("[thumb label link] insn offset %d link offset %d", ref_label_insn.pc_offset, imm12);
} break;
default:
UNREACHABLE();
break;
}
}
}
};
class ThumbRelocLabelEntry : public ThumbPseudoLabel, public RelocLabel {
public:
ThumbRelocLabelEntry(bool is_pc_register) : RelocLabel(), ThumbPseudoLabel(0), is_pc_register_(is_pc_register) {
}
template <typename T> static ThumbRelocLabelEntry *withData(T value, bool is_pc_register) {
auto label = new ThumbRelocLabelEntry(is_pc_register);
label->setData(value);
return label;
}
bool is_pc_register() {
return is_pc_register_;
}
private:
bool is_pc_register_;
};
// ---
class ThumbAssembler : public Assembler {
public:
ThumbAssembler(void *address) : Assembler(address) {
this->SetExecuteState(ThumbExecuteState);
}
ThumbAssembler(void *address, CodeBuffer *buffer) : Assembler(address, buffer) {
this->SetExecuteState(ThumbExecuteState);
}
void EmitInt16(int16_t val) {
buffer_->Emit16(val);
}
void Emit2Int16(int16_t val1, int16_t val2) {
EmitInt16(val1);
EmitInt16(val2);
}
void EmitAddress(uint32_t value) {
buffer_->Emit32(value);
}
// =====
void t1_nop() {
EmitInt16(0xbf00);
}
void t1_b(int32_t imm) {
ASSERT(CheckSignLength(imm, 12));
ASSERT(CheckAlign(imm, 2));
int32_t imm11 = bits(imm >> 1, 0, 10);
EmitInt16(0xe000 | imm11);
}
// =====
void t2_b(uint32_t imm) {
EmitThumb2Branch(AL, imm, false);
}
void t2_bl(uint32_t imm) {
EmitThumb2Branch(AL, imm, true);
}
void t2_blx(uint32_t imm) {
UNIMPLEMENTED();
}
// =====
void t2_ldr(Register dst, const MemOperand &src) {
// WARNNING: literal ldr, base = ALIGN(pc, 4)
EmitThumb2LoadStore(true, dst, src);
}
private:
void EmitThumb2LoadLiteral(Register rt, const MemOperand x) {
bool add = true;
uint32_t U, imm12;
int32_t offset = x.offset();
#if 0
// literal ldr, base = ALIGN(pc, 4)
if (rt.Is(pc)) {
// TODO: convert to `GetRealizedAddress()` ???
addr_t curr_pc = pc_offset() + (addr_t)GetRealizedAddress();
if (curr_pc % 4) {
t1_nop();
}
}
#endif
if (offset > 0) {
U = B7;
imm12 = offset;
} else {
U = 0;
imm12 = -offset;
}
EmitInt16(0xf85f | U);
EmitInt16(0x0 | (rt.code() << 12) | imm12);
}
void EmitThumb2LoadStore(bool load, Register rt, const MemOperand x) {
if (x.rn().Is(pc)) {
EmitThumb2LoadLiteral(rt, x);
return;
}
bool index, add, wback;
if (x.IsRegisterOffset() && x.offset() >= 0) {
index = true, add = true, wback = false;
uint32_t imm12 = x.offset();
EmitInt16(0xf8d0 | (x.rn().code() << 0));
EmitInt16(0x0 | (rt.code() << 12) | imm12);
} else {
// use bit accelerate
uint32_t P = 0, W = 0, U = 0;
uint32_t imm8 = x.offset() > 0 ? x.offset() : -x.offset();
U = x.offset() > 0 ? 0 : B9;
if (x.IsPostIndex()) {
P = 0, W = B8;
} else if (x.IsPreIndex()) {
P = B10, W = B8;
}
index = (P == B10);
add = (U == B9);
wback = (W == B8);
EmitInt16(0xf850 | (x.rn().code() << 0));
EmitInt16(0x0800 | (rt.code() << 12) | P | U | W | imm8);
}
}
void EmitThumb2Branch(Condition cond, int32_t imm, bool link) {
uint32_t operand = imm >> 1;
ASSERT(CheckSignLength(operand, 25));
ASSERT(CheckAlign(operand, 2));
uint32_t signbit = (imm >> 31) & 0x1;
uint32_t i1 = (operand >> 22) & 0x1;
uint32_t i2 = (operand >> 21) & 0x1;
uint32_t imm10 = (operand >> 11) & 0x03ff;
uint32_t imm11 = operand & 0x07ff;
uint32_t j1 = (!(i1 ^ signbit));
uint32_t j2 = (!(i2 ^ signbit));
if (cond != AL) {
UNIMPLEMENTED();
}
EmitInt16(0xf000 | LeftShift(signbit, 1, 10) | LeftShift(imm10, 10, 0));
if (link) {
// Not use LeftShift(1, 1, 14), and use B14 for accelerate
EmitInt16(0x9000 | LeftShift(j1, 1, 13) | (LeftShift(j2, 1, 11)) | LeftShift(imm11, 11, 0) | B14);
} else {
EmitInt16(0x9000 | LeftShift(j1, 1, 13) | (LeftShift(j2, 1, 11)) | LeftShift(imm11, 11, 0));
}
}
};
// ---
class ThumbTurboAssembler : public ThumbAssembler {
public:
ThumbTurboAssembler(void *address) : ThumbAssembler(address) {
}
ThumbTurboAssembler(void *address, CodeBuffer *buffer) : ThumbAssembler(address, buffer) {
}
~ThumbTurboAssembler() {
}
void T1_Ldr(Register rt, ThumbPseudoLabel *label) {
UNREACHABLE();
// t1_ldr: rt can't be PC register
// ===
#if 0
if (label->is_bound()) {
const int64_t dest = label->pos() - buffer_.Size();
ldr(rt, MemOperand(pc, dest));
} else {
// record this ldr, and fix later.
label->link_to(buffer_.Size(), ThumbPseudoLabel::kThumb1Ldr);
ldr(rt, MemOperand(pc, 0));
}
#endif
}
void T2_Ldr(Register rt, ThumbPseudoLabel *label) {
if (label->pos()) {
int offset = label->pos() - buffer_->GetBufferSize();
t2_ldr(rt, MemOperand(pc, offset));
} else {
// record this ldr, and fix later.
label->link_to(kThumb2LiteralLdr, buffer_->GetBufferSize());
t2_ldr(rt, MemOperand(pc, 0));
}
}
void AlignThumbNop() {
addr32_t pc = this->GetCodeBuffer()->GetBufferSize() + (uintptr_t)GetRealizedAddress();
if (pc % Thumb2_INST_LEN) {
t1_nop();
} else {
}
}
// ---
void PseudoBind(ThumbPseudoLabel *label) {
const addr_t bound_pc = buffer_->GetBufferSize();
label->bind_to(bound_pc);
// If some instructions have been wrote, before the label bound, we need link these `confused` instructions
if (label->has_confused_instructions()) {
label->link_confused_instructions(GetCodeBuffer());
}
}
void RelocBind() {
for (auto *data_label : data_labels_) {
PseudoBind(data_label);
reinterpret_cast<CodeBufferBase *>(buffer_)->EmitBuffer(data_label->data_, data_label->data_size_);
}
}
void AppendRelocLabel(ThumbRelocLabelEntry *label) {
data_labels_.push_back(label);
}
void RelocLabelFixup(tinystl::unordered_map<off_t, off_t> *relocated_offset_map) {
for (auto *data_label : data_labels_) {
auto val = data_label->data<int32_t>();
auto iter = relocated_offset_map->find(val);
if (iter != relocated_offset_map->end()) {
data_label->fixupData<int32_t>(iter->second);
}
}
}
private:
tinystl::vector<ThumbRelocLabelEntry *> data_labels_;
};
#if 0
void GenRelocateCodeAndBranch(void *buffer, CodeMemBlock *origin, CodeMemBlock *relocated);
#endif
} // namespace arm
} // namespace zz