/* This file is part of the dynarmic project. * Copyright (c) 2016 MerryMage * This software may be used and distributed according to the terms of the GNU * General Public License version 2 or any later version. */ #pragma once #include #include #include #include #include #include #include "backend_x64/reg_alloc.h" #include "common/address_range.h" #include "dynarmic/callbacks.h" #include "frontend/ir/location_descriptor.h" #include "frontend/ir/terminal.h" namespace Dynarmic { class Jit; namespace IR { class Block; class Inst; } // namespace IR namespace BackendX64 { class BlockOfCode; struct EmitContext { EmitContext(RegAlloc& reg_alloc, IR::Block& block); void EraseInstruction(IR::Inst* inst); virtual bool FPSCR_RoundTowardsZero() const = 0; virtual bool FPSCR_FTZ() const = 0; virtual bool FPSCR_DN() const = 0; RegAlloc& reg_alloc; IR::Block& block; }; template class EmitX64 { public: using ProgramCounterType = typename JitStateType::ProgramCounterType; struct BlockDescriptor { CodePtr entrypoint; // Entrypoint of emitted code size_t size; // Length in bytes of emitted code IR::LocationDescriptor start_location; boost::icl::discrete_interval range; }; EmitX64(BlockOfCode* code, UserCallbacks cb, Jit* jit_interface); virtual ~EmitX64(); /// Looks up an emitted host block in the cache. boost::optional GetBasicBlock(IR::LocationDescriptor descriptor) const; /// Empties the entire cache. void ClearCache(); void InvalidateCacheRanges(const boost::icl::interval_set& ranges); protected: // Microinstruction emitters #define OPCODE(name, type, ...) void Emit##name(EmitContext& ctx, IR::Inst* inst); #define A32OPC(...) #include "frontend/ir/opcodes.inc" #undef OPCODE #undef A32OPC // Helpers void EmitAddCycles(size_t cycles); Xbyak::Label EmitCond(IR::Cond cond); void EmitCondPrelude(const IR::Block& block); void PushRSBHelper(Xbyak::Reg64 loc_desc_reg, Xbyak::Reg64 index_reg, IR::LocationDescriptor target); // Terminal instruction emitters void EmitTerminal(IR::Terminal terminal, IR::LocationDescriptor initial_location); virtual void EmitTerminalImpl(IR::Term::Interpret terminal, IR::LocationDescriptor initial_location) = 0; virtual void EmitTerminalImpl(IR::Term::ReturnToDispatch terminal, IR::LocationDescriptor initial_location) = 0; virtual void EmitTerminalImpl(IR::Term::LinkBlock terminal, IR::LocationDescriptor initial_location) = 0; virtual void EmitTerminalImpl(IR::Term::LinkBlockFast terminal, IR::LocationDescriptor initial_location) = 0; virtual void EmitTerminalImpl(IR::Term::PopRSBHint terminal, IR::LocationDescriptor initial_location) = 0; virtual void EmitTerminalImpl(IR::Term::If terminal, IR::LocationDescriptor initial_location) = 0; virtual void EmitTerminalImpl(IR::Term::CheckHalt terminal, IR::LocationDescriptor initial_location) = 0; // Patching struct PatchInformation { std::vector jg; std::vector jmp; std::vector mov_rcx; }; void Patch(const IR::LocationDescriptor& target_desc, CodePtr target_code_ptr); void Unpatch(const IR::LocationDescriptor& target_desc); virtual void EmitPatchJg(const IR::LocationDescriptor& target_desc, CodePtr target_code_ptr = nullptr) = 0; virtual void EmitPatchJmp(const IR::LocationDescriptor& target_desc, CodePtr target_code_ptr = nullptr) = 0; virtual void EmitPatchMovRcx(CodePtr target_code_ptr = nullptr) = 0; // State BlockOfCode* code; UserCallbacks cb; Jit* jit_interface; std::unordered_map block_descriptors; std::unordered_map patch_information; boost::icl::interval_map> block_ranges; }; } // namespace BackendX64 } // namespace Dynarmic