| //===-- Disassembler.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_Disassembler_h_ |
| #define liblldb_Disassembler_h_ |
| |
| // C Includes |
| // C++ Includes |
| #include <vector> |
| |
| // Other libraries and framework includes |
| // Project includes |
| #include "lldb/lldb-private.h" |
| #include "lldb/Core/Address.h" |
| #include "lldb/Core/ArchSpec.h" |
| #include "lldb/Core/EmulateInstruction.h" |
| #include "lldb/Core/Opcode.h" |
| #include "lldb/Core/PluginInterface.h" |
| #include "lldb/Interpreter/NamedOptionValue.h" |
| |
| namespace lldb_private { |
| |
| class Instruction |
| { |
| public: |
| Instruction (const Address &address, |
| lldb::AddressClass addr_class = lldb::eAddressClassInvalid); |
| |
| virtual |
| ~Instruction(); |
| |
| const Address & |
| GetAddress () const |
| { |
| return m_address; |
| } |
| |
| const char * |
| GetMnemonic (ExecutionContextScope *exe_scope) |
| { |
| CalculateMnemonicOperandsAndCommentIfNeeded (exe_scope); |
| return m_opcode_name.c_str(); |
| } |
| const char * |
| GetOperands (ExecutionContextScope *exe_scope) |
| { |
| CalculateMnemonicOperandsAndCommentIfNeeded (exe_scope); |
| return m_mnemocics.c_str(); |
| } |
| |
| const char * |
| GetComment (ExecutionContextScope *exe_scope) |
| { |
| CalculateMnemonicOperandsAndCommentIfNeeded (exe_scope); |
| return m_comment.c_str(); |
| } |
| |
| virtual void |
| CalculateMnemonicOperandsAndComment (ExecutionContextScope *exe_scope) = 0; |
| |
| lldb::AddressClass |
| GetAddressClass (); |
| |
| void |
| SetAddress (const Address &addr) |
| { |
| // Invalidate the address class to lazily discover |
| // it if we need to. |
| m_address_class = lldb::eAddressClassInvalid; |
| m_address = addr; |
| } |
| |
| virtual void |
| Dump (Stream *s, |
| uint32_t max_opcode_byte_size, |
| bool show_address, |
| bool show_bytes, |
| const ExecutionContext *exe_ctx, |
| bool raw) = 0; |
| |
| virtual bool |
| DoesBranch () const = 0; |
| |
| virtual size_t |
| Decode (const Disassembler &disassembler, |
| const DataExtractor& data, |
| uint32_t data_offset) = 0; |
| |
| virtual void |
| SetDescription (const char *) {}; // May be overridden in sub-classes that have descriptions. |
| |
| lldb::OptionValueSP |
| ReadArray (FILE *in_file, Stream *out_stream, OptionValue::Type data_type); |
| |
| lldb::OptionValueSP |
| ReadDictionary (FILE *in_file, Stream *out_stream); |
| |
| bool |
| DumpEmulation (const ArchSpec &arch); |
| |
| virtual bool |
| TestEmulation (Stream *stream, const char *test_file_name); |
| |
| bool |
| Emulate (const ArchSpec &arch, |
| uint32_t evaluate_options, |
| void *baton, |
| EmulateInstruction::ReadMemoryCallback read_mem_callback, |
| EmulateInstruction::WriteMemoryCallback write_mem_calback, |
| EmulateInstruction::ReadRegisterCallback read_reg_callback, |
| EmulateInstruction::WriteRegisterCallback write_reg_callback); |
| |
| const Opcode & |
| GetOpcode () const |
| { |
| return m_opcode; |
| } |
| |
| protected: |
| Address m_address; // The section offset address of this instruction |
| // We include an address class in the Instruction class to |
| // allow the instruction specify the eAddressClassCodeAlternateISA |
| // (currently used for thumb), and also to specify data (eAddressClassData). |
| // The usual value will be eAddressClassCode, but often when |
| // disassembling memory, you might run into data. This can |
| // help us to disassemble appropriately. |
| lldb::AddressClass m_address_class; |
| Opcode m_opcode; // The opcode for this instruction |
| std::string m_opcode_name; |
| std::string m_mnemocics; |
| std::string m_comment; |
| bool m_calculated_strings; |
| |
| void |
| CalculateMnemonicOperandsAndCommentIfNeeded (ExecutionContextScope *exe_scope) |
| { |
| if (!m_calculated_strings) |
| { |
| m_calculated_strings = true; |
| CalculateMnemonicOperandsAndComment(exe_scope); |
| } |
| } |
| }; |
| |
| |
| class InstructionList |
| { |
| public: |
| InstructionList(); |
| ~InstructionList(); |
| |
| size_t |
| GetSize() const; |
| |
| uint32_t |
| GetMaxOpcocdeByteSize () const; |
| |
| lldb::InstructionSP |
| GetInstructionAtIndex (uint32_t idx) const; |
| |
| uint32_t |
| GetIndexOfNextBranchInstruction(uint32_t start) const; |
| |
| uint32_t |
| GetIndexOfInstructionAtLoadAddress (lldb::addr_t load_addr, Target &target); |
| |
| void |
| Clear(); |
| |
| void |
| Append (lldb::InstructionSP &inst_sp); |
| |
| void |
| Dump (Stream *s, |
| bool show_address, |
| bool show_bytes, |
| const ExecutionContext* exe_ctx); |
| |
| private: |
| typedef std::vector<lldb::InstructionSP> collection; |
| typedef collection::iterator iterator; |
| typedef collection::const_iterator const_iterator; |
| |
| collection m_instructions; |
| }; |
| |
| class PseudoInstruction : |
| public Instruction |
| { |
| public: |
| |
| PseudoInstruction (); |
| |
| virtual |
| ~PseudoInstruction (); |
| |
| virtual void |
| Dump (Stream *s, |
| uint32_t max_opcode_byte_size, |
| bool show_address, |
| bool show_bytes, |
| const ExecutionContext* exe_ctx, |
| bool raw); |
| |
| virtual bool |
| DoesBranch () const; |
| |
| virtual void |
| CalculateMnemonicOperandsAndComment (ExecutionContextScope *exe_scope) |
| { |
| // TODO: fill this in and put opcode name into Instruction::m_opcode_name, |
| // mnemonic into Instruction::m_mnemonics, and any comment into |
| // Instruction::m_comment |
| } |
| |
| virtual size_t |
| Decode (const Disassembler &disassembler, |
| const DataExtractor &data, |
| uint32_t data_offset); |
| |
| void |
| SetOpcode (size_t opcode_size, void *opcode_data); |
| |
| virtual void |
| SetDescription (const char *description); |
| |
| protected: |
| std::string m_description; |
| |
| DISALLOW_COPY_AND_ASSIGN (PseudoInstruction); |
| }; |
| |
| class Disassembler : |
| public PluginInterface |
| { |
| public: |
| |
| enum |
| { |
| eOptionNone = 0u, |
| eOptionShowBytes = (1u << 0), |
| eOptionRawOuput = (1u << 1), |
| eOptionMarkPCSourceLine = (1u << 2), // Mark the source line that contains the current PC (mixed mode only) |
| eOptionMarkPCAddress = (1u << 3) // Mark the disassembly line the contains the PC |
| }; |
| |
| static Disassembler* |
| FindPlugin (const ArchSpec &arch, const char *plugin_name); |
| |
| static lldb::DisassemblerSP |
| DisassembleRange (const ArchSpec &arch, |
| const char *plugin_name, |
| const ExecutionContext &exe_ctx, |
| const AddressRange &disasm_range); |
| |
| static lldb::DisassemblerSP |
| DisassembleBytes (const ArchSpec &arch, |
| const char *plugin_name, |
| const Address &start, |
| const void *bytes, |
| size_t length, |
| uint32_t num_instructions = UINT32_MAX); |
| |
| static bool |
| Disassemble (Debugger &debugger, |
| const ArchSpec &arch, |
| const char *plugin_name, |
| const ExecutionContext &exe_ctx, |
| const AddressRange &range, |
| uint32_t num_instructions, |
| uint32_t num_mixed_context_lines, |
| uint32_t options, |
| Stream &strm); |
| |
| static bool |
| Disassemble (Debugger &debugger, |
| const ArchSpec &arch, |
| const char *plugin_name, |
| const ExecutionContext &exe_ctx, |
| const Address &start, |
| uint32_t num_instructions, |
| uint32_t num_mixed_context_lines, |
| uint32_t options, |
| Stream &strm); |
| |
| static size_t |
| Disassemble (Debugger &debugger, |
| const ArchSpec &arch, |
| const char *plugin_name, |
| const ExecutionContext &exe_ctx, |
| SymbolContextList &sc_list, |
| uint32_t num_instructions, |
| uint32_t num_mixed_context_lines, |
| uint32_t options, |
| Stream &strm); |
| |
| static bool |
| Disassemble (Debugger &debugger, |
| const ArchSpec &arch, |
| const char *plugin_name, |
| const ExecutionContext &exe_ctx, |
| const ConstString &name, |
| Module *module, |
| uint32_t num_instructions, |
| uint32_t num_mixed_context_lines, |
| uint32_t options, |
| Stream &strm); |
| |
| static bool |
| Disassemble (Debugger &debugger, |
| const ArchSpec &arch, |
| const char *plugin_name, |
| const ExecutionContext &exe_ctx, |
| uint32_t num_instructions, |
| uint32_t num_mixed_context_lines, |
| uint32_t options, |
| Stream &strm); |
| |
| //------------------------------------------------------------------ |
| // Constructors and Destructors |
| //------------------------------------------------------------------ |
| Disassembler(const ArchSpec &arch); |
| virtual ~Disassembler(); |
| |
| typedef const char * (*SummaryCallback)(const Instruction& inst, ExecutionContext *exe_context, void *user_data); |
| |
| static bool |
| PrintInstructions (Disassembler *disasm_ptr, |
| Debugger &debugger, |
| const ArchSpec &arch, |
| const ExecutionContext &exe_ctx, |
| uint32_t num_instructions, |
| uint32_t num_mixed_context_lines, |
| uint32_t options, |
| Stream &strm); |
| |
| size_t |
| ParseInstructions (const ExecutionContext *exe_ctx, |
| const AddressRange &range); |
| |
| size_t |
| ParseInstructions (const ExecutionContext *exe_ctx, |
| const Address &range, |
| uint32_t num_instructions); |
| |
| virtual size_t |
| DecodeInstructions (const Address &base_addr, |
| const DataExtractor& data, |
| uint32_t data_offset, |
| uint32_t num_instructions, |
| bool append) = 0; |
| |
| InstructionList & |
| GetInstructionList (); |
| |
| const InstructionList & |
| GetInstructionList () const; |
| |
| const ArchSpec & |
| GetArchitecture () const |
| { |
| return m_arch; |
| } |
| |
| protected: |
| //------------------------------------------------------------------ |
| // Classes that inherit from Disassembler can see and modify these |
| //------------------------------------------------------------------ |
| const ArchSpec m_arch; |
| InstructionList m_instruction_list; |
| lldb::addr_t m_base_addr; |
| |
| private: |
| //------------------------------------------------------------------ |
| // For Disassembler only |
| //------------------------------------------------------------------ |
| DISALLOW_COPY_AND_ASSIGN (Disassembler); |
| }; |
| |
| } // namespace lldb_private |
| |
| #endif // liblldb_Disassembler_h_ |