//===-- DisassemblerLLVMC.h -------------------------------------*- C++ -*-===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #ifndef liblldb_DisassemblerLLVMC_h_ #define liblldb_DisassemblerLLVMC_h_ #include #include "llvm-c/Disassembler.h" // Opaque references to C++ Objects in LLVM's MC. namespace llvm { class MCContext; class MCInst; class MCInstrInfo; class MCRegisterInfo; class MCDisassembler; class MCInstPrinter; class MCAsmInfo; class MCSubtargetInfo; } #include "lldb/Core/Address.h" #include "lldb/Core/Disassembler.h" #include "lldb/Core/PluginManager.h" #include "lldb/Host/Mutex.h" class InstructionLLVMC; class DisassemblerLLVMC : public lldb_private::Disassembler { // Since we need to make two actual MC Disassemblers for ARM (ARM & THUMB), and there's a bit of goo to set up and own // in the MC disassembler world, I added this class to manage the actual disassemblers. class LLVMCDisassembler { public: LLVMCDisassembler (const char *triple, unsigned flavor, DisassemblerLLVMC &owner); ~LLVMCDisassembler(); uint64_t GetMCInst (const uint8_t *opcode_data, size_t opcode_data_len, lldb::addr_t pc, llvm::MCInst &mc_inst); uint64_t PrintMCInst (llvm::MCInst &mc_inst, char *output_buffer, size_t out_buffer_len); void SetStyle (bool use_hex_immed, HexImmediateStyle hex_style); bool CanBranch (llvm::MCInst &mc_inst); bool IsValid() { return m_is_valid; } private: bool m_is_valid; std::unique_ptr m_context_ap; std::unique_ptr m_asm_info_ap; std::unique_ptr m_subtarget_info_ap; std::unique_ptr m_instr_info_ap; std::unique_ptr m_reg_info_ap; std::unique_ptr m_instr_printer_ap; std::unique_ptr m_disasm_ap; }; public: //------------------------------------------------------------------ // Static Functions //------------------------------------------------------------------ static void Initialize(); static void Terminate(); static lldb_private::ConstString GetPluginNameStatic(); static lldb_private::Disassembler * CreateInstance(const lldb_private::ArchSpec &arch, const char *flavor); DisassemblerLLVMC(const lldb_private::ArchSpec &arch, const char *flavor /* = NULL */); virtual ~DisassemblerLLVMC(); virtual size_t DecodeInstructions (const lldb_private::Address &base_addr, const lldb_private::DataExtractor& data, lldb::offset_t data_offset, size_t num_instructions, bool append, bool data_from_file); //------------------------------------------------------------------ // PluginInterface protocol //------------------------------------------------------------------ virtual lldb_private::ConstString GetPluginName(); virtual uint32_t GetPluginVersion(); protected: friend class InstructionLLVMC; virtual bool FlavorValidForArchSpec (const lldb_private::ArchSpec &arch, const char *flavor); bool IsValid() { return (m_disasm_ap.get() != NULL && m_disasm_ap->IsValid()); } int OpInfo(uint64_t PC, uint64_t Offset, uint64_t Size, int TagType, void *TagBug); const char *SymbolLookup (uint64_t ReferenceValue, uint64_t *ReferenceType, uint64_t ReferencePC, const char **ReferenceName); static int OpInfoCallback (void *DisInfo, uint64_t PC, uint64_t Offset, uint64_t Size, int TagType, void *TagBug); static const char *SymbolLookupCallback(void *DisInfo, uint64_t ReferenceValue, uint64_t *ReferenceType, uint64_t ReferencePC, const char **ReferenceName); void Lock(InstructionLLVMC *inst, const lldb_private::ExecutionContext *exe_ctx) { m_mutex.Lock(); m_inst = inst; m_exe_ctx = exe_ctx; } void Unlock() { m_inst = NULL; m_exe_ctx = NULL; m_mutex.Unlock(); } const lldb_private::ExecutionContext *m_exe_ctx; InstructionLLVMC *m_inst; lldb_private::Mutex m_mutex; bool m_data_from_file; std::unique_ptr m_disasm_ap; std::unique_ptr m_alternate_disasm_ap; }; #endif // liblldb_DisassemblerLLVM_h_