130fdc8d8SChris Lattner //===-- Disassembler.cpp ----------------------------------------*- C++ -*-===// 230fdc8d8SChris Lattner // 330fdc8d8SChris Lattner // The LLVM Compiler Infrastructure 430fdc8d8SChris Lattner // 530fdc8d8SChris Lattner // This file is distributed under the University of Illinois Open Source 630fdc8d8SChris Lattner // License. See LICENSE.TXT for details. 730fdc8d8SChris Lattner // 830fdc8d8SChris Lattner //===----------------------------------------------------------------------===// 930fdc8d8SChris Lattner 1093a64300SDaniel Malea #include "lldb/lldb-python.h" 1193a64300SDaniel Malea 1230fdc8d8SChris Lattner #include "lldb/Core/Disassembler.h" 1330fdc8d8SChris Lattner 1430fdc8d8SChris Lattner // C Includes 1530fdc8d8SChris Lattner // C++ Includes 1630fdc8d8SChris Lattner // Other libraries and framework includes 1730fdc8d8SChris Lattner // Project includes 1830fdc8d8SChris Lattner #include "lldb/lldb-private.h" 1930fdc8d8SChris Lattner #include "lldb/Core/Error.h" 2030fdc8d8SChris Lattner #include "lldb/Core/DataBufferHeap.h" 2130fdc8d8SChris Lattner #include "lldb/Core/DataExtractor.h" 2230fdc8d8SChris Lattner #include "lldb/Core/Debugger.h" 23ad379efcSCaroline Tice #include "lldb/Core/EmulateInstruction.h" 2430fdc8d8SChris Lattner #include "lldb/Core/Module.h" 2530fdc8d8SChris Lattner #include "lldb/Core/PluginManager.h" 26de2fb9cfSCaroline Tice #include "lldb/Core/RegularExpression.h" 2730fdc8d8SChris Lattner #include "lldb/Core/Timer.h" 2867cc0636SGreg Clayton #include "lldb/Interpreter/OptionValue.h" 2967cc0636SGreg Clayton #include "lldb/Interpreter/OptionValueArray.h" 3067cc0636SGreg Clayton #include "lldb/Interpreter/OptionValueDictionary.h" 3167cc0636SGreg Clayton #include "lldb/Interpreter/OptionValueString.h" 3267cc0636SGreg Clayton #include "lldb/Interpreter/OptionValueUInt64.h" 33b6d70ebcSSean Callanan #include "lldb/Symbol/ClangNamespaceDecl.h" 341f746071SGreg Clayton #include "lldb/Symbol/Function.h" 3530fdc8d8SChris Lattner #include "lldb/Symbol/ObjectFile.h" 3630fdc8d8SChris Lattner #include "lldb/Target/ExecutionContext.h" 3730fdc8d8SChris Lattner #include "lldb/Target/Process.h" 38*d5944cd1SGreg Clayton #include "lldb/Target/SectionLoadList.h" 39b57e4a1bSJason Molenda #include "lldb/Target/StackFrame.h" 4030fdc8d8SChris Lattner #include "lldb/Target/Target.h" 4130fdc8d8SChris Lattner 4230fdc8d8SChris Lattner #define DEFAULT_DISASM_BYTE_SIZE 32 4330fdc8d8SChris Lattner 4430fdc8d8SChris Lattner using namespace lldb; 4530fdc8d8SChris Lattner using namespace lldb_private; 4630fdc8d8SChris Lattner 4730fdc8d8SChris Lattner 487e6d4e5aSSean Callanan DisassemblerSP 490f063ba6SJim Ingham Disassembler::FindPlugin (const ArchSpec &arch, const char *flavor, const char *plugin_name) 5030fdc8d8SChris Lattner { 5130fdc8d8SChris Lattner Timer scoped_timer (__PRETTY_FUNCTION__, 521080edbcSGreg Clayton "Disassembler::FindPlugin (arch = %s, plugin_name = %s)", 531080edbcSGreg Clayton arch.GetArchitectureName(), 541080edbcSGreg Clayton plugin_name); 5530fdc8d8SChris Lattner 561080edbcSGreg Clayton DisassemblerCreateInstance create_callback = NULL; 571080edbcSGreg Clayton 581080edbcSGreg Clayton if (plugin_name) 591080edbcSGreg Clayton { 6057abc5d6SGreg Clayton ConstString const_plugin_name (plugin_name); 6157abc5d6SGreg Clayton create_callback = PluginManager::GetDisassemblerCreateCallbackForPluginName (const_plugin_name); 621080edbcSGreg Clayton if (create_callback) 631080edbcSGreg Clayton { 640f063ba6SJim Ingham DisassemblerSP disassembler_sp(create_callback(arch, flavor)); 651080edbcSGreg Clayton 667e6d4e5aSSean Callanan if (disassembler_sp.get()) 677e6d4e5aSSean Callanan return disassembler_sp; 681080edbcSGreg Clayton } 691080edbcSGreg Clayton } 701080edbcSGreg Clayton else 711080edbcSGreg Clayton { 7230fdc8d8SChris Lattner for (uint32_t idx = 0; (create_callback = PluginManager::GetDisassemblerCreateCallbackAtIndex(idx)) != NULL; ++idx) 7330fdc8d8SChris Lattner { 740f063ba6SJim Ingham DisassemblerSP disassembler_sp(create_callback(arch, flavor)); 7530fdc8d8SChris Lattner 767e6d4e5aSSean Callanan if (disassembler_sp.get()) 777e6d4e5aSSean Callanan return disassembler_sp; 7830fdc8d8SChris Lattner } 791080edbcSGreg Clayton } 807e6d4e5aSSean Callanan return DisassemblerSP(); 8130fdc8d8SChris Lattner } 8230fdc8d8SChris Lattner 830f063ba6SJim Ingham DisassemblerSP 840f063ba6SJim Ingham Disassembler::FindPluginForTarget(const TargetSP target_sp, const ArchSpec &arch, const char *flavor, const char *plugin_name) 850f063ba6SJim Ingham { 860f063ba6SJim Ingham if (target_sp && flavor == NULL) 870f063ba6SJim Ingham { 880f063ba6SJim Ingham // FIXME - we don't have the mechanism in place to do per-architecture settings. But since we know that for now 890f063ba6SJim Ingham // we only support flavors on x86 & x86_64, 900f063ba6SJim Ingham if (arch.GetTriple().getArch() == llvm::Triple::x86 910f063ba6SJim Ingham || arch.GetTriple().getArch() == llvm::Triple::x86_64) 920f063ba6SJim Ingham flavor = target_sp->GetDisassemblyFlavor(); 930f063ba6SJim Ingham } 940f063ba6SJim Ingham return FindPlugin(arch, flavor, plugin_name); 950f063ba6SJim Ingham } 960f063ba6SJim Ingham 97dda4f7b5SGreg Clayton 98357132ebSGreg Clayton static void 99357132ebSGreg Clayton ResolveAddress (const ExecutionContext &exe_ctx, 100357132ebSGreg Clayton const Address &addr, 101357132ebSGreg Clayton Address &resolved_addr) 102357132ebSGreg Clayton { 103357132ebSGreg Clayton if (!addr.IsSectionOffset()) 104357132ebSGreg Clayton { 105357132ebSGreg Clayton // If we weren't passed in a section offset address range, 106357132ebSGreg Clayton // try and resolve it to something 107c14ee32dSGreg Clayton Target *target = exe_ctx.GetTargetPtr(); 108c14ee32dSGreg Clayton if (target) 109357132ebSGreg Clayton { 110c14ee32dSGreg Clayton if (target->GetSectionLoadList().IsEmpty()) 111357132ebSGreg Clayton { 112c14ee32dSGreg Clayton target->GetImages().ResolveFileAddress (addr.GetOffset(), resolved_addr); 113357132ebSGreg Clayton } 114357132ebSGreg Clayton else 115357132ebSGreg Clayton { 116c14ee32dSGreg Clayton target->GetSectionLoadList().ResolveLoadAddress (addr.GetOffset(), resolved_addr); 117357132ebSGreg Clayton } 118357132ebSGreg Clayton // We weren't able to resolve the address, just treat it as a 119357132ebSGreg Clayton // raw address 120357132ebSGreg Clayton if (resolved_addr.IsValid()) 121357132ebSGreg Clayton return; 122357132ebSGreg Clayton } 123357132ebSGreg Clayton } 124357132ebSGreg Clayton resolved_addr = addr; 125357132ebSGreg Clayton } 126dda4f7b5SGreg Clayton 127dda4f7b5SGreg Clayton size_t 128dda4f7b5SGreg Clayton Disassembler::Disassemble 129dda4f7b5SGreg Clayton ( 130dda4f7b5SGreg Clayton Debugger &debugger, 131dda4f7b5SGreg Clayton const ArchSpec &arch, 1321080edbcSGreg Clayton const char *plugin_name, 1330f063ba6SJim Ingham const char *flavor, 134dda4f7b5SGreg Clayton const ExecutionContext &exe_ctx, 135dda4f7b5SGreg Clayton SymbolContextList &sc_list, 13637023b06SJim Ingham uint32_t num_instructions, 137dda4f7b5SGreg Clayton uint32_t num_mixed_context_lines, 1381da6f9d7SGreg Clayton uint32_t options, 139dda4f7b5SGreg Clayton Stream &strm 140dda4f7b5SGreg Clayton ) 141dda4f7b5SGreg Clayton { 142dda4f7b5SGreg Clayton size_t success_count = 0; 143dda4f7b5SGreg Clayton const size_t count = sc_list.GetSize(); 144dda4f7b5SGreg Clayton SymbolContext sc; 145dda4f7b5SGreg Clayton AddressRange range; 1467e14f91dSGreg Clayton const uint32_t scope = eSymbolContextBlock | eSymbolContextFunction | eSymbolContextSymbol; 1477e14f91dSGreg Clayton const bool use_inline_block_range = true; 148dda4f7b5SGreg Clayton for (size_t i=0; i<count; ++i) 149dda4f7b5SGreg Clayton { 150dda4f7b5SGreg Clayton if (sc_list.GetContextAtIndex(i, sc) == false) 151dda4f7b5SGreg Clayton break; 1527e14f91dSGreg Clayton for (uint32_t range_idx = 0; sc.GetAddressRange(scope, range_idx, use_inline_block_range, range); ++range_idx) 153dda4f7b5SGreg Clayton { 1541080edbcSGreg Clayton if (Disassemble (debugger, 1551080edbcSGreg Clayton arch, 1561080edbcSGreg Clayton plugin_name, 1570f063ba6SJim Ingham flavor, 1581080edbcSGreg Clayton exe_ctx, 1591080edbcSGreg Clayton range, 1601080edbcSGreg Clayton num_instructions, 1611080edbcSGreg Clayton num_mixed_context_lines, 1621da6f9d7SGreg Clayton options, 1631080edbcSGreg Clayton strm)) 164dda4f7b5SGreg Clayton { 165dda4f7b5SGreg Clayton ++success_count; 166dda4f7b5SGreg Clayton strm.EOL(); 167dda4f7b5SGreg Clayton } 168dda4f7b5SGreg Clayton } 169dda4f7b5SGreg Clayton } 170dda4f7b5SGreg Clayton return success_count; 171dda4f7b5SGreg Clayton } 172dda4f7b5SGreg Clayton 17330fdc8d8SChris Lattner bool 17430fdc8d8SChris Lattner Disassembler::Disassemble 17530fdc8d8SChris Lattner ( 1766611103cSGreg Clayton Debugger &debugger, 17730fdc8d8SChris Lattner const ArchSpec &arch, 1781080edbcSGreg Clayton const char *plugin_name, 1790f063ba6SJim Ingham const char *flavor, 18030fdc8d8SChris Lattner const ExecutionContext &exe_ctx, 181dda4f7b5SGreg Clayton const ConstString &name, 182dda4f7b5SGreg Clayton Module *module, 18337023b06SJim Ingham uint32_t num_instructions, 184dda4f7b5SGreg Clayton uint32_t num_mixed_context_lines, 1851da6f9d7SGreg Clayton uint32_t options, 18630fdc8d8SChris Lattner Stream &strm 18730fdc8d8SChris Lattner ) 18830fdc8d8SChris Lattner { 189dda4f7b5SGreg Clayton SymbolContextList sc_list; 190931180e6SGreg Clayton if (name) 191931180e6SGreg Clayton { 192931180e6SGreg Clayton const bool include_symbols = true; 1939df05fbbSSean Callanan const bool include_inlines = true; 194dda4f7b5SGreg Clayton if (module) 195dda4f7b5SGreg Clayton { 196931180e6SGreg Clayton module->FindFunctions (name, 197b6d70ebcSSean Callanan NULL, 1986ecb232bSGreg Clayton eFunctionNameTypeAuto, 199931180e6SGreg Clayton include_symbols, 2009df05fbbSSean Callanan include_inlines, 201dda4f7b5SGreg Clayton true, 202931180e6SGreg Clayton sc_list); 203dda4f7b5SGreg Clayton } 204c14ee32dSGreg Clayton else if (exe_ctx.GetTargetPtr()) 205dda4f7b5SGreg Clayton { 206c14ee32dSGreg Clayton exe_ctx.GetTargetPtr()->GetImages().FindFunctions (name, 2076ecb232bSGreg Clayton eFunctionNameTypeAuto, 208931180e6SGreg Clayton include_symbols, 2099df05fbbSSean Callanan include_inlines, 2108ade104aSSean Callanan false, 211931180e6SGreg Clayton sc_list); 212dda4f7b5SGreg Clayton } 213dda4f7b5SGreg Clayton } 214931180e6SGreg Clayton 215931180e6SGreg Clayton if (sc_list.GetSize ()) 216931180e6SGreg Clayton { 217931180e6SGreg Clayton return Disassemble (debugger, 218931180e6SGreg Clayton arch, 2191080edbcSGreg Clayton plugin_name, 2200f063ba6SJim Ingham flavor, 221931180e6SGreg Clayton exe_ctx, 222931180e6SGreg Clayton sc_list, 22337023b06SJim Ingham num_instructions, 224931180e6SGreg Clayton num_mixed_context_lines, 2251da6f9d7SGreg Clayton options, 226931180e6SGreg Clayton strm); 227dda4f7b5SGreg Clayton } 228dda4f7b5SGreg Clayton return false; 229dda4f7b5SGreg Clayton } 230dda4f7b5SGreg Clayton 2311d273166SGreg Clayton 2321d273166SGreg Clayton lldb::DisassemblerSP 2331d273166SGreg Clayton Disassembler::DisassembleRange 2341d273166SGreg Clayton ( 2351d273166SGreg Clayton const ArchSpec &arch, 2361080edbcSGreg Clayton const char *plugin_name, 2370f063ba6SJim Ingham const char *flavor, 2381d273166SGreg Clayton const ExecutionContext &exe_ctx, 2396b3e6d54SJason Molenda const AddressRange &range, 2406b3e6d54SJason Molenda bool prefer_file_cache 2411d273166SGreg Clayton ) 2421d273166SGreg Clayton { 2431d273166SGreg Clayton lldb::DisassemblerSP disasm_sp; 2441d273166SGreg Clayton if (range.GetByteSize() > 0 && range.GetBaseAddress().IsValid()) 2451d273166SGreg Clayton { 2460f063ba6SJim Ingham disasm_sp = Disassembler::FindPluginForTarget(exe_ctx.GetTargetSP(), arch, flavor, plugin_name); 2471d273166SGreg Clayton 2481d273166SGreg Clayton if (disasm_sp) 2491d273166SGreg Clayton { 2503faf47c4SGreg Clayton size_t bytes_disassembled = disasm_sp->ParseInstructions (&exe_ctx, range, NULL, prefer_file_cache); 2511d273166SGreg Clayton if (bytes_disassembled == 0) 2521d273166SGreg Clayton disasm_sp.reset(); 2531d273166SGreg Clayton } 2541d273166SGreg Clayton } 2551d273166SGreg Clayton return disasm_sp; 2561d273166SGreg Clayton } 2571d273166SGreg Clayton 25850952e95SSean Callanan lldb::DisassemblerSP 2593faf47c4SGreg Clayton Disassembler::DisassembleBytes (const ArchSpec &arch, 26050952e95SSean Callanan const char *plugin_name, 2610f063ba6SJim Ingham const char *flavor, 26250952e95SSean Callanan const Address &start, 2633faf47c4SGreg Clayton const void *src, 2643faf47c4SGreg Clayton size_t src_len, 2653faf47c4SGreg Clayton uint32_t num_instructions, 2663faf47c4SGreg Clayton bool data_from_file) 26750952e95SSean Callanan { 26850952e95SSean Callanan lldb::DisassemblerSP disasm_sp; 26950952e95SSean Callanan 2703faf47c4SGreg Clayton if (src) 27150952e95SSean Callanan { 2720f063ba6SJim Ingham disasm_sp = Disassembler::FindPlugin(arch, flavor, plugin_name); 27350952e95SSean Callanan 27450952e95SSean Callanan if (disasm_sp) 27550952e95SSean Callanan { 2763faf47c4SGreg Clayton DataExtractor data(src, src_len, arch.GetByteOrder(), arch.GetAddressByteSize()); 27750952e95SSean Callanan 27850952e95SSean Callanan (void)disasm_sp->DecodeInstructions (start, 27950952e95SSean Callanan data, 28050952e95SSean Callanan 0, 2819c766110SGreg Clayton num_instructions, 2823faf47c4SGreg Clayton false, 2833faf47c4SGreg Clayton data_from_file); 28450952e95SSean Callanan } 28550952e95SSean Callanan } 28650952e95SSean Callanan 28750952e95SSean Callanan return disasm_sp; 28850952e95SSean Callanan } 28950952e95SSean Callanan 2901d273166SGreg Clayton 291dda4f7b5SGreg Clayton bool 292dda4f7b5SGreg Clayton Disassembler::Disassemble 293dda4f7b5SGreg Clayton ( 294dda4f7b5SGreg Clayton Debugger &debugger, 295dda4f7b5SGreg Clayton const ArchSpec &arch, 2961080edbcSGreg Clayton const char *plugin_name, 2970f063ba6SJim Ingham const char *flavor, 298dda4f7b5SGreg Clayton const ExecutionContext &exe_ctx, 299dda4f7b5SGreg Clayton const AddressRange &disasm_range, 30037023b06SJim Ingham uint32_t num_instructions, 301dda4f7b5SGreg Clayton uint32_t num_mixed_context_lines, 3021da6f9d7SGreg Clayton uint32_t options, 303dda4f7b5SGreg Clayton Stream &strm 304dda4f7b5SGreg Clayton ) 305dda4f7b5SGreg Clayton { 306dda4f7b5SGreg Clayton if (disasm_range.GetByteSize()) 307dda4f7b5SGreg Clayton { 3080f063ba6SJim Ingham lldb::DisassemblerSP disasm_sp (Disassembler::FindPluginForTarget(exe_ctx.GetTargetSP(), arch, flavor, plugin_name)); 30930fdc8d8SChris Lattner 3107e6d4e5aSSean Callanan if (disasm_sp.get()) 31130fdc8d8SChris Lattner { 312357132ebSGreg Clayton AddressRange range; 313357132ebSGreg Clayton ResolveAddress (exe_ctx, disasm_range.GetBaseAddress(), range.GetBaseAddress()); 314357132ebSGreg Clayton range.SetByteSize (disasm_range.GetByteSize()); 3153faf47c4SGreg Clayton const bool prefer_file_cache = false; 3163faf47c4SGreg Clayton size_t bytes_disassembled = disasm_sp->ParseInstructions (&exe_ctx, range, &strm, prefer_file_cache); 31730fdc8d8SChris Lattner if (bytes_disassembled == 0) 31830fdc8d8SChris Lattner return false; 3191080edbcSGreg Clayton 32056d40428SJim Ingham bool result = PrintInstructions (disasm_sp.get(), 32137023b06SJim Ingham debugger, 32237023b06SJim Ingham arch, 32337023b06SJim Ingham exe_ctx, 32437023b06SJim Ingham num_instructions, 32537023b06SJim Ingham num_mixed_context_lines, 3261da6f9d7SGreg Clayton options, 32737023b06SJim Ingham strm); 32856d40428SJim Ingham 32956d40428SJim Ingham // FIXME: The DisassemblerLLVMC has a reference cycle and won't go away if it has any active instructions. 33056d40428SJim Ingham // I'll fix that but for now, just clear the list and it will go away nicely. 33156d40428SJim Ingham disasm_sp->GetInstructionList().Clear(); 33256d40428SJim Ingham return result; 33337023b06SJim Ingham } 33437023b06SJim Ingham } 33537023b06SJim Ingham return false; 33637023b06SJim Ingham } 33737023b06SJim Ingham 33837023b06SJim Ingham bool 33937023b06SJim Ingham Disassembler::Disassemble 34037023b06SJim Ingham ( 34137023b06SJim Ingham Debugger &debugger, 34237023b06SJim Ingham const ArchSpec &arch, 3431080edbcSGreg Clayton const char *plugin_name, 3440f063ba6SJim Ingham const char *flavor, 34537023b06SJim Ingham const ExecutionContext &exe_ctx, 34637023b06SJim Ingham const Address &start_address, 34737023b06SJim Ingham uint32_t num_instructions, 34837023b06SJim Ingham uint32_t num_mixed_context_lines, 3491da6f9d7SGreg Clayton uint32_t options, 35037023b06SJim Ingham Stream &strm 35137023b06SJim Ingham ) 35237023b06SJim Ingham { 35337023b06SJim Ingham if (num_instructions > 0) 35437023b06SJim Ingham { 3553faf47c4SGreg Clayton lldb::DisassemblerSP disasm_sp (Disassembler::FindPluginForTarget(exe_ctx.GetTargetSP(), 3563faf47c4SGreg Clayton arch, 3573faf47c4SGreg Clayton flavor, 3583faf47c4SGreg Clayton plugin_name)); 3597e6d4e5aSSean Callanan if (disasm_sp.get()) 36037023b06SJim Ingham { 361357132ebSGreg Clayton Address addr; 362357132ebSGreg Clayton ResolveAddress (exe_ctx, start_address, addr); 3633faf47c4SGreg Clayton const bool prefer_file_cache = false; 3643faf47c4SGreg Clayton size_t bytes_disassembled = disasm_sp->ParseInstructions (&exe_ctx, 3653faf47c4SGreg Clayton addr, 3663faf47c4SGreg Clayton num_instructions, 3673faf47c4SGreg Clayton prefer_file_cache); 36837023b06SJim Ingham if (bytes_disassembled == 0) 36937023b06SJim Ingham return false; 37056d40428SJim Ingham bool result = PrintInstructions (disasm_sp.get(), 37137023b06SJim Ingham debugger, 37237023b06SJim Ingham arch, 37337023b06SJim Ingham exe_ctx, 37437023b06SJim Ingham num_instructions, 37537023b06SJim Ingham num_mixed_context_lines, 3761da6f9d7SGreg Clayton options, 37737023b06SJim Ingham strm); 37856d40428SJim Ingham 37956d40428SJim Ingham // FIXME: The DisassemblerLLVMC has a reference cycle and won't go away if it has any active instructions. 38056d40428SJim Ingham // I'll fix that but for now, just clear the list and it will go away nicely. 38156d40428SJim Ingham disasm_sp->GetInstructionList().Clear(); 38256d40428SJim Ingham return result; 38337023b06SJim Ingham } 38437023b06SJim Ingham } 38537023b06SJim Ingham return false; 38637023b06SJim Ingham } 38737023b06SJim Ingham 38837023b06SJim Ingham bool 38937023b06SJim Ingham Disassembler::PrintInstructions 39037023b06SJim Ingham ( 39137023b06SJim Ingham Disassembler *disasm_ptr, 39237023b06SJim Ingham Debugger &debugger, 39337023b06SJim Ingham const ArchSpec &arch, 39437023b06SJim Ingham const ExecutionContext &exe_ctx, 39537023b06SJim Ingham uint32_t num_instructions, 39637023b06SJim Ingham uint32_t num_mixed_context_lines, 3971da6f9d7SGreg Clayton uint32_t options, 39837023b06SJim Ingham Stream &strm 39937023b06SJim Ingham ) 40037023b06SJim Ingham { 40130fdc8d8SChris Lattner // We got some things disassembled... 40237023b06SJim Ingham size_t num_instructions_found = disasm_ptr->GetInstructionList().GetSize(); 40337023b06SJim Ingham 40437023b06SJim Ingham if (num_instructions > 0 && num_instructions < num_instructions_found) 40537023b06SJim Ingham num_instructions_found = num_instructions; 40637023b06SJim Ingham 407357132ebSGreg Clayton const uint32_t max_opcode_byte_size = disasm_ptr->GetInstructionList().GetMaxOpcocdeByteSize (); 40830fdc8d8SChris Lattner uint32_t offset = 0; 40930fdc8d8SChris Lattner SymbolContext sc; 41030fdc8d8SChris Lattner SymbolContext prev_sc; 41130fdc8d8SChris Lattner AddressRange sc_range; 41234132754SGreg Clayton const Address *pc_addr_ptr = NULL; 4137e14f91dSGreg Clayton ExecutionContextScope *exe_scope = exe_ctx.GetBestExecutionContextScope(); 414b57e4a1bSJason Molenda StackFrame *frame = exe_ctx.GetFramePtr(); 415c14ee32dSGreg Clayton 4164b2967ffSMichael Sartain TargetSP target_sp (exe_ctx.GetTargetSP()); 4174b2967ffSMichael Sartain SourceManager &source_manager = target_sp ? target_sp->GetSourceManager() : debugger.GetSourceManager(); 4184b2967ffSMichael Sartain 419c14ee32dSGreg Clayton if (frame) 420c14ee32dSGreg Clayton pc_addr_ptr = &frame->GetFrameCodeAddress(); 4217e14f91dSGreg Clayton const uint32_t scope = eSymbolContextLineEntry | eSymbolContextFunction | eSymbolContextSymbol; 4227e14f91dSGreg Clayton const bool use_inline_block_range = false; 42337023b06SJim Ingham for (size_t i=0; i<num_instructions_found; ++i) 42430fdc8d8SChris Lattner { 42537023b06SJim Ingham Instruction *inst = disasm_ptr->GetInstructionList().GetInstructionAtIndex (i).get(); 42630fdc8d8SChris Lattner if (inst) 42730fdc8d8SChris Lattner { 42832e0a750SGreg Clayton const Address &addr = inst->GetAddress(); 42932e0a750SGreg Clayton const bool inst_is_at_pc = pc_addr_ptr && addr == *pc_addr_ptr; 430dda4f7b5SGreg Clayton 43130fdc8d8SChris Lattner prev_sc = sc; 432dda4f7b5SGreg Clayton 433e72dfb32SGreg Clayton ModuleSP module_sp (addr.GetModule()); 434e72dfb32SGreg Clayton if (module_sp) 43530fdc8d8SChris Lattner { 436e72dfb32SGreg Clayton uint32_t resolved_mask = module_sp->ResolveSymbolContextForAddress(addr, eSymbolContextEverything, sc); 43730fdc8d8SChris Lattner if (resolved_mask) 43830fdc8d8SChris Lattner { 43932e0a750SGreg Clayton if (num_mixed_context_lines) 440dda4f7b5SGreg Clayton { 44132e0a750SGreg Clayton if (!sc_range.ContainsFileAddress (addr)) 442dda4f7b5SGreg Clayton { 4437e14f91dSGreg Clayton sc.GetAddressRange (scope, 0, use_inline_block_range, sc_range); 444dda4f7b5SGreg Clayton 44530fdc8d8SChris Lattner if (sc != prev_sc) 44630fdc8d8SChris Lattner { 44730fdc8d8SChris Lattner if (offset != 0) 44830fdc8d8SChris Lattner strm.EOL(); 44930fdc8d8SChris Lattner 450c14ee32dSGreg Clayton sc.DumpStopContext(&strm, exe_ctx.GetProcessPtr(), addr, false, true, false); 4516dbd3983SGreg Clayton strm.EOL(); 45230fdc8d8SChris Lattner 45330fdc8d8SChris Lattner if (sc.comp_unit && sc.line_entry.IsValid()) 45430fdc8d8SChris Lattner { 4554b2967ffSMichael Sartain source_manager.DisplaySourceLinesWithLineNumbers (sc.line_entry.file, 45630fdc8d8SChris Lattner sc.line_entry.line, 457dda4f7b5SGreg Clayton num_mixed_context_lines, 458dda4f7b5SGreg Clayton num_mixed_context_lines, 459b10d72f0SGreg Clayton ((inst_is_at_pc && (options & eOptionMarkPCSourceLine)) ? "->" : ""), 46030fdc8d8SChris Lattner &strm); 46130fdc8d8SChris Lattner } 46230fdc8d8SChris Lattner } 46330fdc8d8SChris Lattner } 46430fdc8d8SChris Lattner } 4651bba2bedSGreg Clayton else if ((sc.function || sc.symbol) && (sc.function != prev_sc.function || sc.symbol != prev_sc.symbol)) 46632e0a750SGreg Clayton { 46732e0a750SGreg Clayton if (prev_sc.function || prev_sc.symbol) 46832e0a750SGreg Clayton strm.EOL(); 46932e0a750SGreg Clayton 4707e14f91dSGreg Clayton bool show_fullpaths = false; 4717e14f91dSGreg Clayton bool show_module = true; 4727e14f91dSGreg Clayton bool show_inlined_frames = true; 4737e14f91dSGreg Clayton sc.DumpStopContext (&strm, 4747e14f91dSGreg Clayton exe_scope, 4757e14f91dSGreg Clayton addr, 4767e14f91dSGreg Clayton show_fullpaths, 4777e14f91dSGreg Clayton show_module, 4787e14f91dSGreg Clayton show_inlined_frames); 47932e0a750SGreg Clayton 48032e0a750SGreg Clayton strm << ":\n"; 48132e0a750SGreg Clayton } 48232e0a750SGreg Clayton } 483dda4f7b5SGreg Clayton else 484dda4f7b5SGreg Clayton { 48572310355SGreg Clayton sc.Clear(true); 48630fdc8d8SChris Lattner } 48730fdc8d8SChris Lattner } 48832e0a750SGreg Clayton 489b10d72f0SGreg Clayton if ((options & eOptionMarkPCAddress) && pc_addr_ptr) 49032e0a750SGreg Clayton { 491b10d72f0SGreg Clayton strm.PutCString(inst_is_at_pc ? "-> " : " "); 49232e0a750SGreg Clayton } 4931da6f9d7SGreg Clayton const bool show_bytes = (options & eOptionShowBytes) != 0; 494ba812f42SGreg Clayton inst->Dump(&strm, max_opcode_byte_size, true, show_bytes, &exe_ctx); 49530fdc8d8SChris Lattner strm.EOL(); 49630fdc8d8SChris Lattner } 49730fdc8d8SChris Lattner else 49830fdc8d8SChris Lattner { 49930fdc8d8SChris Lattner break; 50030fdc8d8SChris Lattner } 50130fdc8d8SChris Lattner } 50230fdc8d8SChris Lattner 50330fdc8d8SChris Lattner return true; 50430fdc8d8SChris Lattner } 50530fdc8d8SChris Lattner 506dda4f7b5SGreg Clayton 507dda4f7b5SGreg Clayton bool 508dda4f7b5SGreg Clayton Disassembler::Disassemble 509dda4f7b5SGreg Clayton ( 510dda4f7b5SGreg Clayton Debugger &debugger, 511dda4f7b5SGreg Clayton const ArchSpec &arch, 5121080edbcSGreg Clayton const char *plugin_name, 5130f063ba6SJim Ingham const char *flavor, 514dda4f7b5SGreg Clayton const ExecutionContext &exe_ctx, 51537023b06SJim Ingham uint32_t num_instructions, 516dda4f7b5SGreg Clayton uint32_t num_mixed_context_lines, 5171da6f9d7SGreg Clayton uint32_t options, 518dda4f7b5SGreg Clayton Stream &strm 519dda4f7b5SGreg Clayton ) 520dda4f7b5SGreg Clayton { 521dda4f7b5SGreg Clayton AddressRange range; 522b57e4a1bSJason Molenda StackFrame *frame = exe_ctx.GetFramePtr(); 523c14ee32dSGreg Clayton if (frame) 524dda4f7b5SGreg Clayton { 525c14ee32dSGreg Clayton SymbolContext sc(frame->GetSymbolContext(eSymbolContextFunction | eSymbolContextSymbol)); 526dda4f7b5SGreg Clayton if (sc.function) 527dda4f7b5SGreg Clayton { 528dda4f7b5SGreg Clayton range = sc.function->GetAddressRange(); 529dda4f7b5SGreg Clayton } 530e7612134SGreg Clayton else if (sc.symbol && sc.symbol->ValueIsAddress()) 531dda4f7b5SGreg Clayton { 532e7612134SGreg Clayton range.GetBaseAddress() = sc.symbol->GetAddress(); 533e7612134SGreg Clayton range.SetByteSize (sc.symbol->GetByteSize()); 534dda4f7b5SGreg Clayton } 535dda4f7b5SGreg Clayton else 536dda4f7b5SGreg Clayton { 537c14ee32dSGreg Clayton range.GetBaseAddress() = frame->GetFrameCodeAddress(); 538dda4f7b5SGreg Clayton } 539dda4f7b5SGreg Clayton 540dda4f7b5SGreg Clayton if (range.GetBaseAddress().IsValid() && range.GetByteSize() == 0) 541dda4f7b5SGreg Clayton range.SetByteSize (DEFAULT_DISASM_BYTE_SIZE); 542dda4f7b5SGreg Clayton } 543dda4f7b5SGreg Clayton 5441080edbcSGreg Clayton return Disassemble (debugger, 5451080edbcSGreg Clayton arch, 5461080edbcSGreg Clayton plugin_name, 5470f063ba6SJim Ingham flavor, 5481080edbcSGreg Clayton exe_ctx, 5491080edbcSGreg Clayton range, 5501080edbcSGreg Clayton num_instructions, 5511080edbcSGreg Clayton num_mixed_context_lines, 5521da6f9d7SGreg Clayton options, 5531080edbcSGreg Clayton strm); 554dda4f7b5SGreg Clayton } 555dda4f7b5SGreg Clayton 556357132ebSGreg Clayton Instruction::Instruction(const Address &address, AddressClass addr_class) : 5571080edbcSGreg Clayton m_address (address), 558357132ebSGreg Clayton m_address_class (addr_class), 559a97aa92aSSean Callanan m_opcode(), 560a97aa92aSSean Callanan m_calculated_strings(false) 5610ae96273SGreg Clayton { 56230fdc8d8SChris Lattner } 56330fdc8d8SChris Lattner 5641d273166SGreg Clayton Instruction::~Instruction() 56530fdc8d8SChris Lattner { 56630fdc8d8SChris Lattner } 56730fdc8d8SChris Lattner 568357132ebSGreg Clayton AddressClass 569357132ebSGreg Clayton Instruction::GetAddressClass () 570357132ebSGreg Clayton { 571357132ebSGreg Clayton if (m_address_class == eAddressClassInvalid) 572357132ebSGreg Clayton m_address_class = m_address.GetAddressClass(); 573357132ebSGreg Clayton return m_address_class; 574357132ebSGreg Clayton } 57530fdc8d8SChris Lattner 576ba812f42SGreg Clayton void 577ba812f42SGreg Clayton Instruction::Dump (lldb_private::Stream *s, 578ba812f42SGreg Clayton uint32_t max_opcode_byte_size, 579ba812f42SGreg Clayton bool show_address, 580ba812f42SGreg Clayton bool show_bytes, 581ba812f42SGreg Clayton const ExecutionContext* exe_ctx) 582ba812f42SGreg Clayton { 5837a37c1ecSJason Molenda size_t opcode_column_width = 7; 584ba812f42SGreg Clayton const size_t operand_column_width = 25; 585ba812f42SGreg Clayton 586ba812f42SGreg Clayton CalculateMnemonicOperandsAndCommentIfNeeded (exe_ctx); 587ba812f42SGreg Clayton 588ba812f42SGreg Clayton StreamString ss; 589ba812f42SGreg Clayton 590ba812f42SGreg Clayton if (show_address) 591ba812f42SGreg Clayton { 592ba812f42SGreg Clayton m_address.Dump(&ss, 593ba812f42SGreg Clayton exe_ctx ? exe_ctx->GetBestExecutionContextScope() : NULL, 594ba812f42SGreg Clayton Address::DumpStyleLoadAddress, 595ba812f42SGreg Clayton Address::DumpStyleModuleWithFileAddress, 596ba812f42SGreg Clayton 0); 597ba812f42SGreg Clayton 598ba812f42SGreg Clayton ss.PutCString(": "); 599ba812f42SGreg Clayton } 600ba812f42SGreg Clayton 601ba812f42SGreg Clayton if (show_bytes) 602ba812f42SGreg Clayton { 603ba812f42SGreg Clayton if (m_opcode.GetType() == Opcode::eTypeBytes) 604ba812f42SGreg Clayton { 605ba812f42SGreg Clayton // x86_64 and i386 are the only ones that use bytes right now so 606ba812f42SGreg Clayton // pad out the byte dump to be able to always show 15 bytes (3 chars each) 607ba812f42SGreg Clayton // plus a space 608ba812f42SGreg Clayton if (max_opcode_byte_size > 0) 609ba812f42SGreg Clayton m_opcode.Dump (&ss, max_opcode_byte_size * 3 + 1); 610ba812f42SGreg Clayton else 611ba812f42SGreg Clayton m_opcode.Dump (&ss, 15 * 3 + 1); 612ba812f42SGreg Clayton } 613ba812f42SGreg Clayton else 614ba812f42SGreg Clayton { 615d616c97aSEd Maste // Else, we have ARM or MIPS which can show up to a uint32_t 616d616c97aSEd Maste // 0x00000000 (10 spaces) plus two for padding... 617ba812f42SGreg Clayton if (max_opcode_byte_size > 0) 618ba812f42SGreg Clayton m_opcode.Dump (&ss, max_opcode_byte_size * 3 + 1); 619ba812f42SGreg Clayton else 620ba812f42SGreg Clayton m_opcode.Dump (&ss, 12); 621ba812f42SGreg Clayton } 622ba812f42SGreg Clayton } 623ba812f42SGreg Clayton 624ba812f42SGreg Clayton const size_t opcode_pos = ss.GetSize(); 625ba812f42SGreg Clayton 6267a37c1ecSJason Molenda // The default opcode size of 7 characters is plenty for most architectures 6277a37c1ecSJason Molenda // but some like arm can pull out the occasional vqrshrun.s16. We won't get 6287a37c1ecSJason Molenda // consistent column spacing in these cases, unfortunately. 6297a37c1ecSJason Molenda if (m_opcode_name.length() >= opcode_column_width) 6307a37c1ecSJason Molenda { 6317a37c1ecSJason Molenda opcode_column_width = m_opcode_name.length() + 1; 6327a37c1ecSJason Molenda } 6337a37c1ecSJason Molenda 634ba812f42SGreg Clayton ss.PutCString (m_opcode_name.c_str()); 635ba812f42SGreg Clayton ss.FillLastLineToColumn (opcode_pos + opcode_column_width, ' '); 6360f063ba6SJim Ingham ss.PutCString (m_mnemonics.c_str()); 637ba812f42SGreg Clayton 638ba812f42SGreg Clayton if (!m_comment.empty()) 639ba812f42SGreg Clayton { 640ba812f42SGreg Clayton ss.FillLastLineToColumn (opcode_pos + opcode_column_width + operand_column_width, ' '); 641ba812f42SGreg Clayton ss.PutCString (" ; "); 642ba812f42SGreg Clayton ss.PutCString (m_comment.c_str()); 643ba812f42SGreg Clayton } 644ba812f42SGreg Clayton s->Write (ss.GetData(), ss.GetSize()); 645ba812f42SGreg Clayton } 646ba812f42SGreg Clayton 6477c9dd3ceSCaroline Tice bool 6487c9dd3ceSCaroline Tice Instruction::DumpEmulation (const ArchSpec &arch) 6497c9dd3ceSCaroline Tice { 6507b0992d9SGreg Clayton std::unique_ptr<EmulateInstruction> insn_emulator_ap (EmulateInstruction::FindPlugin (arch, eInstructionTypeAny, NULL)); 6517c9dd3ceSCaroline Tice if (insn_emulator_ap.get()) 6527c9dd3ceSCaroline Tice { 6532ed751bdSGreg Clayton insn_emulator_ap->SetInstruction (GetOpcode(), GetAddress(), NULL); 6542ed751bdSGreg Clayton return insn_emulator_ap->EvaluateInstruction (0); 6557c9dd3ceSCaroline Tice } 6567c9dd3ceSCaroline Tice 6577c9dd3ceSCaroline Tice return false; 6587c9dd3ceSCaroline Tice } 6597c9dd3ceSCaroline Tice 660de2fb9cfSCaroline Tice OptionValueSP 661de2fb9cfSCaroline Tice Instruction::ReadArray (FILE *in_file, Stream *out_stream, OptionValue::Type data_type) 662de2fb9cfSCaroline Tice { 663de2fb9cfSCaroline Tice bool done = false; 664de2fb9cfSCaroline Tice char buffer[1024]; 665de2fb9cfSCaroline Tice 666de2fb9cfSCaroline Tice OptionValueSP option_value_sp (new OptionValueArray (1u << data_type)); 667de2fb9cfSCaroline Tice 668de2fb9cfSCaroline Tice int idx = 0; 669de2fb9cfSCaroline Tice while (!done) 670de2fb9cfSCaroline Tice { 671de2fb9cfSCaroline Tice if (!fgets (buffer, 1023, in_file)) 672de2fb9cfSCaroline Tice { 673762f7135SGreg Clayton out_stream->Printf ("Instruction::ReadArray: Error reading file (fgets).\n"); 674de2fb9cfSCaroline Tice option_value_sp.reset (); 675de2fb9cfSCaroline Tice return option_value_sp; 676de2fb9cfSCaroline Tice } 677de2fb9cfSCaroline Tice 678de2fb9cfSCaroline Tice std::string line (buffer); 679de2fb9cfSCaroline Tice 680c7bece56SGreg Clayton size_t len = line.size(); 681de2fb9cfSCaroline Tice if (line[len-1] == '\n') 682de2fb9cfSCaroline Tice { 683de2fb9cfSCaroline Tice line[len-1] = '\0'; 684de2fb9cfSCaroline Tice line.resize (len-1); 685de2fb9cfSCaroline Tice } 686de2fb9cfSCaroline Tice 687de2fb9cfSCaroline Tice if ((line.size() == 1) && line[0] == ']') 688de2fb9cfSCaroline Tice { 689de2fb9cfSCaroline Tice done = true; 690de2fb9cfSCaroline Tice line.clear(); 691de2fb9cfSCaroline Tice } 692de2fb9cfSCaroline Tice 693de2fb9cfSCaroline Tice if (line.size() > 0) 694de2fb9cfSCaroline Tice { 695de2fb9cfSCaroline Tice std::string value; 696bc43cab5SGreg Clayton static RegularExpression g_reg_exp ("^[ \t]*([^ \t]+)[ \t]*$"); 697bc43cab5SGreg Clayton RegularExpression::Match regex_match(1); 698bc43cab5SGreg Clayton bool reg_exp_success = g_reg_exp.Execute (line.c_str(), ®ex_match); 699de2fb9cfSCaroline Tice if (reg_exp_success) 700bc43cab5SGreg Clayton regex_match.GetMatchAtIndex (line.c_str(), 1, value); 701de2fb9cfSCaroline Tice else 702de2fb9cfSCaroline Tice value = line; 703de2fb9cfSCaroline Tice 704de2fb9cfSCaroline Tice OptionValueSP data_value_sp; 705de2fb9cfSCaroline Tice switch (data_type) 706de2fb9cfSCaroline Tice { 707de2fb9cfSCaroline Tice case OptionValue::eTypeUInt64: 708de2fb9cfSCaroline Tice data_value_sp.reset (new OptionValueUInt64 (0, 0)); 709de2fb9cfSCaroline Tice data_value_sp->SetValueFromCString (value.c_str()); 710de2fb9cfSCaroline Tice break; 711de2fb9cfSCaroline Tice // Other types can be added later as needed. 712de2fb9cfSCaroline Tice default: 713de2fb9cfSCaroline Tice data_value_sp.reset (new OptionValueString (value.c_str(), "")); 714de2fb9cfSCaroline Tice break; 715de2fb9cfSCaroline Tice } 716de2fb9cfSCaroline Tice 71784c39663SGreg Clayton option_value_sp->GetAsArray()->InsertValue (idx, data_value_sp); 718de2fb9cfSCaroline Tice ++idx; 719de2fb9cfSCaroline Tice } 720de2fb9cfSCaroline Tice } 721de2fb9cfSCaroline Tice 722de2fb9cfSCaroline Tice return option_value_sp; 723de2fb9cfSCaroline Tice } 724de2fb9cfSCaroline Tice 725de2fb9cfSCaroline Tice OptionValueSP 726de2fb9cfSCaroline Tice Instruction::ReadDictionary (FILE *in_file, Stream *out_stream) 727de2fb9cfSCaroline Tice { 728de2fb9cfSCaroline Tice bool done = false; 729de2fb9cfSCaroline Tice char buffer[1024]; 730de2fb9cfSCaroline Tice 731de2fb9cfSCaroline Tice OptionValueSP option_value_sp (new OptionValueDictionary()); 732de2fb9cfSCaroline Tice static ConstString encoding_key ("data_encoding"); 733de2fb9cfSCaroline Tice OptionValue::Type data_type = OptionValue::eTypeInvalid; 734de2fb9cfSCaroline Tice 735de2fb9cfSCaroline Tice 736de2fb9cfSCaroline Tice while (!done) 737de2fb9cfSCaroline Tice { 738de2fb9cfSCaroline Tice // Read the next line in the file 739de2fb9cfSCaroline Tice if (!fgets (buffer, 1023, in_file)) 740de2fb9cfSCaroline Tice { 741de2fb9cfSCaroline Tice out_stream->Printf ("Instruction::ReadDictionary: Error reading file (fgets).\n"); 742de2fb9cfSCaroline Tice option_value_sp.reset (); 743de2fb9cfSCaroline Tice return option_value_sp; 744de2fb9cfSCaroline Tice } 745de2fb9cfSCaroline Tice 746de2fb9cfSCaroline Tice // Check to see if the line contains the end-of-dictionary marker ("}") 747de2fb9cfSCaroline Tice std::string line (buffer); 748de2fb9cfSCaroline Tice 749c7bece56SGreg Clayton size_t len = line.size(); 750de2fb9cfSCaroline Tice if (line[len-1] == '\n') 751de2fb9cfSCaroline Tice { 752de2fb9cfSCaroline Tice line[len-1] = '\0'; 753de2fb9cfSCaroline Tice line.resize (len-1); 754de2fb9cfSCaroline Tice } 755de2fb9cfSCaroline Tice 756de2fb9cfSCaroline Tice if ((line.size() == 1) && (line[0] == '}')) 757de2fb9cfSCaroline Tice { 758de2fb9cfSCaroline Tice done = true; 759de2fb9cfSCaroline Tice line.clear(); 760de2fb9cfSCaroline Tice } 761de2fb9cfSCaroline Tice 762de2fb9cfSCaroline Tice // Try to find a key-value pair in the current line and add it to the dictionary. 763de2fb9cfSCaroline Tice if (line.size() > 0) 764de2fb9cfSCaroline Tice { 765bc43cab5SGreg Clayton static RegularExpression g_reg_exp ("^[ \t]*([a-zA-Z_][a-zA-Z0-9_]*)[ \t]*=[ \t]*(.*)[ \t]*$"); 766bc43cab5SGreg Clayton RegularExpression::Match regex_match(2); 767bc43cab5SGreg Clayton 768bc43cab5SGreg Clayton bool reg_exp_success = g_reg_exp.Execute (line.c_str(), ®ex_match); 769de2fb9cfSCaroline Tice std::string key; 770de2fb9cfSCaroline Tice std::string value; 771de2fb9cfSCaroline Tice if (reg_exp_success) 772de2fb9cfSCaroline Tice { 773bc43cab5SGreg Clayton regex_match.GetMatchAtIndex (line.c_str(), 1, key); 774bc43cab5SGreg Clayton regex_match.GetMatchAtIndex (line.c_str(), 2, value); 775de2fb9cfSCaroline Tice } 776de2fb9cfSCaroline Tice else 777de2fb9cfSCaroline Tice { 778de2fb9cfSCaroline Tice out_stream->Printf ("Instruction::ReadDictionary: Failure executing regular expression.\n"); 779de2fb9cfSCaroline Tice option_value_sp.reset(); 780de2fb9cfSCaroline Tice return option_value_sp; 781de2fb9cfSCaroline Tice } 782de2fb9cfSCaroline Tice 783de2fb9cfSCaroline Tice ConstString const_key (key.c_str()); 784de2fb9cfSCaroline Tice // Check value to see if it's the start of an array or dictionary. 785de2fb9cfSCaroline Tice 786de2fb9cfSCaroline Tice lldb::OptionValueSP value_sp; 787de2fb9cfSCaroline Tice assert (value.empty() == false); 788de2fb9cfSCaroline Tice assert (key.empty() == false); 789de2fb9cfSCaroline Tice 790de2fb9cfSCaroline Tice if (value[0] == '{') 791de2fb9cfSCaroline Tice { 792de2fb9cfSCaroline Tice assert (value.size() == 1); 793de2fb9cfSCaroline Tice // value is a dictionary 794de2fb9cfSCaroline Tice value_sp = ReadDictionary (in_file, out_stream); 795de2fb9cfSCaroline Tice if (value_sp.get() == NULL) 796de2fb9cfSCaroline Tice { 797de2fb9cfSCaroline Tice option_value_sp.reset (); 798de2fb9cfSCaroline Tice return option_value_sp; 799de2fb9cfSCaroline Tice } 800de2fb9cfSCaroline Tice } 801de2fb9cfSCaroline Tice else if (value[0] == '[') 802de2fb9cfSCaroline Tice { 803de2fb9cfSCaroline Tice assert (value.size() == 1); 804de2fb9cfSCaroline Tice // value is an array 805de2fb9cfSCaroline Tice value_sp = ReadArray (in_file, out_stream, data_type); 806de2fb9cfSCaroline Tice if (value_sp.get() == NULL) 807de2fb9cfSCaroline Tice { 808de2fb9cfSCaroline Tice option_value_sp.reset (); 809de2fb9cfSCaroline Tice return option_value_sp; 810de2fb9cfSCaroline Tice } 811de2fb9cfSCaroline Tice // We've used the data_type to read an array; re-set the type to Invalid 812de2fb9cfSCaroline Tice data_type = OptionValue::eTypeInvalid; 813de2fb9cfSCaroline Tice } 814de2fb9cfSCaroline Tice else if ((value[0] == '0') && (value[1] == 'x')) 815de2fb9cfSCaroline Tice { 816de2fb9cfSCaroline Tice value_sp.reset (new OptionValueUInt64 (0, 0)); 817de2fb9cfSCaroline Tice value_sp->SetValueFromCString (value.c_str()); 818de2fb9cfSCaroline Tice } 819de2fb9cfSCaroline Tice else 820de2fb9cfSCaroline Tice { 821c7bece56SGreg Clayton size_t len = value.size(); 822de2fb9cfSCaroline Tice if ((value[0] == '"') && (value[len-1] == '"')) 823de2fb9cfSCaroline Tice value = value.substr (1, len-2); 824de2fb9cfSCaroline Tice value_sp.reset (new OptionValueString (value.c_str(), "")); 825de2fb9cfSCaroline Tice } 826de2fb9cfSCaroline Tice 827de2fb9cfSCaroline Tice 828de2fb9cfSCaroline Tice 829de2fb9cfSCaroline Tice if (const_key == encoding_key) 830de2fb9cfSCaroline Tice { 831de2fb9cfSCaroline Tice // A 'data_encoding=..." is NOT a normal key-value pair; it is meta-data indicating the 832de2fb9cfSCaroline Tice // data type of an upcoming array (usually the next bit of data to be read in). 833de2fb9cfSCaroline Tice if (strcmp (value.c_str(), "uint32_t") == 0) 834de2fb9cfSCaroline Tice data_type = OptionValue::eTypeUInt64; 835de2fb9cfSCaroline Tice } 836de2fb9cfSCaroline Tice else 83784c39663SGreg Clayton option_value_sp->GetAsDictionary()->SetValueForKey (const_key, value_sp, false); 838de2fb9cfSCaroline Tice } 839de2fb9cfSCaroline Tice } 840de2fb9cfSCaroline Tice 841de2fb9cfSCaroline Tice return option_value_sp; 842de2fb9cfSCaroline Tice } 843de2fb9cfSCaroline Tice 8447c9dd3ceSCaroline Tice bool 8453ac6711aSCaroline Tice Instruction::TestEmulation (Stream *out_stream, const char *file_name) 8463ac6711aSCaroline Tice { 8473ac6711aSCaroline Tice if (!out_stream) 8483ac6711aSCaroline Tice return false; 8493ac6711aSCaroline Tice 8503ac6711aSCaroline Tice if (!file_name) 8513ac6711aSCaroline Tice { 852ea80ba8bSJohnny Chen out_stream->Printf ("Instruction::TestEmulation: Missing file_name."); 8533ac6711aSCaroline Tice return false; 8543ac6711aSCaroline Tice } 8553ac6711aSCaroline Tice 8563ac6711aSCaroline Tice FILE *test_file = fopen (file_name, "r"); 8573ac6711aSCaroline Tice if (!test_file) 8583ac6711aSCaroline Tice { 859ea80ba8bSJohnny Chen out_stream->Printf ("Instruction::TestEmulation: Attempt to open test file failed."); 8603ac6711aSCaroline Tice return false; 8613ac6711aSCaroline Tice } 8623ac6711aSCaroline Tice 8633ac6711aSCaroline Tice char buffer[256]; 864de2fb9cfSCaroline Tice if (!fgets (buffer, 255, test_file)) 8653ac6711aSCaroline Tice { 866de2fb9cfSCaroline Tice out_stream->Printf ("Instruction::TestEmulation: Error reading first line of test file.\n"); 8673ac6711aSCaroline Tice fclose (test_file); 8683ac6711aSCaroline Tice return false; 8693ac6711aSCaroline Tice } 8703ac6711aSCaroline Tice 871de2fb9cfSCaroline Tice if (strncmp (buffer, "InstructionEmulationState={", 27) != 0) 872de2fb9cfSCaroline Tice { 873de2fb9cfSCaroline Tice out_stream->Printf ("Instructin::TestEmulation: Test file does not contain emulation state dictionary\n"); 874de2fb9cfSCaroline Tice fclose (test_file); 875de2fb9cfSCaroline Tice return false; 876de2fb9cfSCaroline Tice } 877de2fb9cfSCaroline Tice 878de2fb9cfSCaroline Tice // Read all the test information from the test file into an OptionValueDictionary. 879de2fb9cfSCaroline Tice 880de2fb9cfSCaroline Tice OptionValueSP data_dictionary_sp (ReadDictionary (test_file, out_stream)); 881de2fb9cfSCaroline Tice if (data_dictionary_sp.get() == NULL) 882de2fb9cfSCaroline Tice { 883de2fb9cfSCaroline Tice out_stream->Printf ("Instruction::TestEmulation: Error reading Dictionary Object.\n"); 884de2fb9cfSCaroline Tice fclose (test_file); 885de2fb9cfSCaroline Tice return false; 886de2fb9cfSCaroline Tice } 887de2fb9cfSCaroline Tice 888de2fb9cfSCaroline Tice fclose (test_file); 889de2fb9cfSCaroline Tice 89084c39663SGreg Clayton OptionValueDictionary *data_dictionary = data_dictionary_sp->GetAsDictionary(); 891de2fb9cfSCaroline Tice static ConstString description_key ("assembly_string"); 892de2fb9cfSCaroline Tice static ConstString triple_key ("triple"); 893de2fb9cfSCaroline Tice 894de2fb9cfSCaroline Tice OptionValueSP value_sp = data_dictionary->GetValueForKey (description_key); 895de2fb9cfSCaroline Tice 896de2fb9cfSCaroline Tice if (value_sp.get() == NULL) 897de2fb9cfSCaroline Tice { 898de2fb9cfSCaroline Tice out_stream->Printf ("Instruction::TestEmulation: Test file does not contain description string.\n"); 899de2fb9cfSCaroline Tice return false; 900de2fb9cfSCaroline Tice } 901de2fb9cfSCaroline Tice 902de2fb9cfSCaroline Tice SetDescription (value_sp->GetStringValue()); 903de2fb9cfSCaroline Tice 904de2fb9cfSCaroline Tice 905de2fb9cfSCaroline Tice value_sp = data_dictionary->GetValueForKey (triple_key); 906de2fb9cfSCaroline Tice if (value_sp.get() == NULL) 907de2fb9cfSCaroline Tice { 908de2fb9cfSCaroline Tice out_stream->Printf ("Instruction::TestEmulation: Test file does not contain triple.\n"); 909de2fb9cfSCaroline Tice return false; 910de2fb9cfSCaroline Tice } 911de2fb9cfSCaroline Tice 912de2fb9cfSCaroline Tice ArchSpec arch; 913de2fb9cfSCaroline Tice arch.SetTriple (llvm::Triple (value_sp->GetStringValue())); 9143ac6711aSCaroline Tice 9153ac6711aSCaroline Tice bool success = false; 9167b0992d9SGreg Clayton std::unique_ptr<EmulateInstruction> insn_emulator_ap (EmulateInstruction::FindPlugin (arch, eInstructionTypeAny, NULL)); 9173ac6711aSCaroline Tice if (insn_emulator_ap.get()) 918de2fb9cfSCaroline Tice success = insn_emulator_ap->TestEmulation (out_stream, arch, data_dictionary); 9193ac6711aSCaroline Tice 9203ac6711aSCaroline Tice if (success) 921ea80ba8bSJohnny Chen out_stream->Printf ("Emulation test succeeded."); 9223ac6711aSCaroline Tice else 923ea80ba8bSJohnny Chen out_stream->Printf ("Emulation test failed."); 9243ac6711aSCaroline Tice 9253ac6711aSCaroline Tice return success; 9263ac6711aSCaroline Tice } 9273ac6711aSCaroline Tice 9283ac6711aSCaroline Tice bool 9297c9dd3ceSCaroline Tice Instruction::Emulate (const ArchSpec &arch, 9302ed751bdSGreg Clayton uint32_t evaluate_options, 9317c9dd3ceSCaroline Tice void *baton, 9327349bd90SGreg Clayton EmulateInstruction::ReadMemoryCallback read_mem_callback, 9337349bd90SGreg Clayton EmulateInstruction::WriteMemoryCallback write_mem_callback, 9347349bd90SGreg Clayton EmulateInstruction::ReadRegisterCallback read_reg_callback, 9357349bd90SGreg Clayton EmulateInstruction::WriteRegisterCallback write_reg_callback) 9367c9dd3ceSCaroline Tice { 9377b0992d9SGreg Clayton std::unique_ptr<EmulateInstruction> insn_emulator_ap (EmulateInstruction::FindPlugin (arch, eInstructionTypeAny, NULL)); 9387c9dd3ceSCaroline Tice if (insn_emulator_ap.get()) 9397c9dd3ceSCaroline Tice { 9407c9dd3ceSCaroline Tice insn_emulator_ap->SetBaton (baton); 9417c9dd3ceSCaroline Tice insn_emulator_ap->SetCallbacks (read_mem_callback, write_mem_callback, read_reg_callback, write_reg_callback); 9422ed751bdSGreg Clayton insn_emulator_ap->SetInstruction (GetOpcode(), GetAddress(), NULL); 9432ed751bdSGreg Clayton return insn_emulator_ap->EvaluateInstruction (evaluate_options); 9447c9dd3ceSCaroline Tice } 9457c9dd3ceSCaroline Tice 9467c9dd3ceSCaroline Tice return false; 9477c9dd3ceSCaroline Tice } 9487c9dd3ceSCaroline Tice 949ba812f42SGreg Clayton 950ba812f42SGreg Clayton uint32_t 951ba812f42SGreg Clayton Instruction::GetData (DataExtractor &data) 952ba812f42SGreg Clayton { 953cd4ae1abSSean Callanan return m_opcode.GetData(data); 954ba812f42SGreg Clayton } 955ba812f42SGreg Clayton 9561d273166SGreg Clayton InstructionList::InstructionList() : 95730fdc8d8SChris Lattner m_instructions() 95830fdc8d8SChris Lattner { 95930fdc8d8SChris Lattner } 96030fdc8d8SChris Lattner 9611d273166SGreg Clayton InstructionList::~InstructionList() 96230fdc8d8SChris Lattner { 96330fdc8d8SChris Lattner } 96430fdc8d8SChris Lattner 96530fdc8d8SChris Lattner size_t 9661d273166SGreg Clayton InstructionList::GetSize() const 96730fdc8d8SChris Lattner { 96830fdc8d8SChris Lattner return m_instructions.size(); 96930fdc8d8SChris Lattner } 97030fdc8d8SChris Lattner 971357132ebSGreg Clayton uint32_t 972357132ebSGreg Clayton InstructionList::GetMaxOpcocdeByteSize () const 973357132ebSGreg Clayton { 974357132ebSGreg Clayton uint32_t max_inst_size = 0; 975357132ebSGreg Clayton collection::const_iterator pos, end; 976357132ebSGreg Clayton for (pos = m_instructions.begin(), end = m_instructions.end(); 977357132ebSGreg Clayton pos != end; 978357132ebSGreg Clayton ++pos) 979357132ebSGreg Clayton { 980357132ebSGreg Clayton uint32_t inst_size = (*pos)->GetOpcode().GetByteSize(); 981357132ebSGreg Clayton if (max_inst_size < inst_size) 982357132ebSGreg Clayton max_inst_size = inst_size; 983357132ebSGreg Clayton } 984357132ebSGreg Clayton return max_inst_size; 985357132ebSGreg Clayton } 986357132ebSGreg Clayton 987357132ebSGreg Clayton 98830fdc8d8SChris Lattner 9891d273166SGreg Clayton InstructionSP 990c7bece56SGreg Clayton InstructionList::GetInstructionAtIndex (size_t idx) const 99130fdc8d8SChris Lattner { 9921d273166SGreg Clayton InstructionSP inst_sp; 99330fdc8d8SChris Lattner if (idx < m_instructions.size()) 9941d273166SGreg Clayton inst_sp = m_instructions[idx]; 9951d273166SGreg Clayton return inst_sp; 99630fdc8d8SChris Lattner } 99730fdc8d8SChris Lattner 99830fdc8d8SChris Lattner void 9995009f9d5SGreg Clayton InstructionList::Dump (Stream *s, 10005009f9d5SGreg Clayton bool show_address, 10015009f9d5SGreg Clayton bool show_bytes, 10025009f9d5SGreg Clayton const ExecutionContext* exe_ctx) 10035009f9d5SGreg Clayton { 10045009f9d5SGreg Clayton const uint32_t max_opcode_byte_size = GetMaxOpcocdeByteSize(); 10055009f9d5SGreg Clayton collection::const_iterator pos, begin, end; 10065009f9d5SGreg Clayton for (begin = m_instructions.begin(), end = m_instructions.end(), pos = begin; 10075009f9d5SGreg Clayton pos != end; 10085009f9d5SGreg Clayton ++pos) 10095009f9d5SGreg Clayton { 10105009f9d5SGreg Clayton if (pos != begin) 10115009f9d5SGreg Clayton s->EOL(); 1012ba812f42SGreg Clayton (*pos)->Dump(s, max_opcode_byte_size, show_address, show_bytes, exe_ctx); 10135009f9d5SGreg Clayton } 10145009f9d5SGreg Clayton } 10155009f9d5SGreg Clayton 10165009f9d5SGreg Clayton 10175009f9d5SGreg Clayton void 10181d273166SGreg Clayton InstructionList::Clear() 101930fdc8d8SChris Lattner { 102030fdc8d8SChris Lattner m_instructions.clear(); 102130fdc8d8SChris Lattner } 102230fdc8d8SChris Lattner 102330fdc8d8SChris Lattner void 10241d273166SGreg Clayton InstructionList::Append (lldb::InstructionSP &inst_sp) 102530fdc8d8SChris Lattner { 102630fdc8d8SChris Lattner if (inst_sp) 102730fdc8d8SChris Lattner m_instructions.push_back(inst_sp); 102830fdc8d8SChris Lattner } 102930fdc8d8SChris Lattner 1030564d8bc2SJim Ingham uint32_t 1031564d8bc2SJim Ingham InstructionList::GetIndexOfNextBranchInstruction(uint32_t start) const 1032564d8bc2SJim Ingham { 1033564d8bc2SJim Ingham size_t num_instructions = m_instructions.size(); 1034564d8bc2SJim Ingham 1035564d8bc2SJim Ingham uint32_t next_branch = UINT32_MAX; 1036564d8bc2SJim Ingham for (size_t i = start; i < num_instructions; i++) 1037564d8bc2SJim Ingham { 1038564d8bc2SJim Ingham if (m_instructions[i]->DoesBranch()) 1039564d8bc2SJim Ingham { 1040564d8bc2SJim Ingham next_branch = i; 1041564d8bc2SJim Ingham break; 1042564d8bc2SJim Ingham } 1043564d8bc2SJim Ingham } 1044564d8bc2SJim Ingham return next_branch; 1045564d8bc2SJim Ingham } 1046564d8bc2SJim Ingham 1047564d8bc2SJim Ingham uint32_t 1048564d8bc2SJim Ingham InstructionList::GetIndexOfInstructionAtLoadAddress (lldb::addr_t load_addr, Target &target) 1049564d8bc2SJim Ingham { 1050564d8bc2SJim Ingham Address address; 1051564d8bc2SJim Ingham address.SetLoadAddress(load_addr, &target); 1052c7bece56SGreg Clayton size_t num_instructions = m_instructions.size(); 1053564d8bc2SJim Ingham uint32_t index = UINT32_MAX; 1054c7bece56SGreg Clayton for (size_t i = 0; i < num_instructions; i++) 1055564d8bc2SJim Ingham { 1056564d8bc2SJim Ingham if (m_instructions[i]->GetAddress() == address) 1057564d8bc2SJim Ingham { 1058564d8bc2SJim Ingham index = i; 1059564d8bc2SJim Ingham break; 1060564d8bc2SJim Ingham } 1061564d8bc2SJim Ingham } 1062564d8bc2SJim Ingham return index; 1063564d8bc2SJim Ingham } 106430fdc8d8SChris Lattner 106530fdc8d8SChris Lattner size_t 10663faf47c4SGreg Clayton Disassembler::ParseInstructions (const ExecutionContext *exe_ctx, 106757f0630cSGreg Clayton const AddressRange &range, 10683faf47c4SGreg Clayton Stream *error_strm_ptr, 10693faf47c4SGreg Clayton bool prefer_file_cache) 107030fdc8d8SChris Lattner { 1071c14ee32dSGreg Clayton if (exe_ctx) 1072c14ee32dSGreg Clayton { 1073c14ee32dSGreg Clayton Target *target = exe_ctx->GetTargetPtr(); 1074dda4f7b5SGreg Clayton const addr_t byte_size = range.GetByteSize(); 1075dda4f7b5SGreg Clayton if (target == NULL || byte_size == 0 || !range.GetBaseAddress().IsValid()) 107630fdc8d8SChris Lattner return 0; 107730fdc8d8SChris Lattner 1078dda4f7b5SGreg Clayton DataBufferHeap *heap_buffer = new DataBufferHeap (byte_size, '\0'); 1079dda4f7b5SGreg Clayton DataBufferSP data_sp(heap_buffer); 108030fdc8d8SChris Lattner 108130fdc8d8SChris Lattner Error error; 10823faf47c4SGreg Clayton lldb::addr_t load_addr = LLDB_INVALID_ADDRESS; 1083357132ebSGreg Clayton const size_t bytes_read = target->ReadMemory (range.GetBaseAddress(), 1084357132ebSGreg Clayton prefer_file_cache, 1085357132ebSGreg Clayton heap_buffer->GetBytes(), 1086357132ebSGreg Clayton heap_buffer->GetByteSize(), 10873faf47c4SGreg Clayton error, 10883faf47c4SGreg Clayton &load_addr); 1089dda4f7b5SGreg Clayton 1090dda4f7b5SGreg Clayton if (bytes_read > 0) 109130fdc8d8SChris Lattner { 1092dda4f7b5SGreg Clayton if (bytes_read != heap_buffer->GetByteSize()) 1093dda4f7b5SGreg Clayton heap_buffer->SetByteSize (bytes_read); 1094357132ebSGreg Clayton DataExtractor data (data_sp, 1095357132ebSGreg Clayton m_arch.GetByteOrder(), 1096357132ebSGreg Clayton m_arch.GetAddressByteSize()); 10973faf47c4SGreg Clayton const bool data_from_file = load_addr == LLDB_INVALID_ADDRESS; 10983faf47c4SGreg Clayton return DecodeInstructions (range.GetBaseAddress(), data, 0, UINT32_MAX, false, data_from_file); 109930fdc8d8SChris Lattner } 110057f0630cSGreg Clayton else if (error_strm_ptr) 110157f0630cSGreg Clayton { 110257f0630cSGreg Clayton const char *error_cstr = error.AsCString(); 110357f0630cSGreg Clayton if (error_cstr) 110457f0630cSGreg Clayton { 110557f0630cSGreg Clayton error_strm_ptr->Printf("error: %s\n", error_cstr); 110657f0630cSGreg Clayton } 110757f0630cSGreg Clayton } 110857f0630cSGreg Clayton } 110957f0630cSGreg Clayton else if (error_strm_ptr) 111057f0630cSGreg Clayton { 111157f0630cSGreg Clayton error_strm_ptr->PutCString("error: invalid execution context\n"); 1112c14ee32dSGreg Clayton } 111330fdc8d8SChris Lattner return 0; 111430fdc8d8SChris Lattner } 111530fdc8d8SChris Lattner 111637023b06SJim Ingham size_t 11173faf47c4SGreg Clayton Disassembler::ParseInstructions (const ExecutionContext *exe_ctx, 111837023b06SJim Ingham const Address &start, 11193faf47c4SGreg Clayton uint32_t num_instructions, 11203faf47c4SGreg Clayton bool prefer_file_cache) 112137023b06SJim Ingham { 1122357132ebSGreg Clayton m_instruction_list.Clear(); 112337023b06SJim Ingham 1124c14ee32dSGreg Clayton if (exe_ctx == NULL || num_instructions == 0 || !start.IsValid()) 112537023b06SJim Ingham return 0; 112637023b06SJim Ingham 1127c14ee32dSGreg Clayton Target *target = exe_ctx->GetTargetPtr(); 1128357132ebSGreg Clayton // Calculate the max buffer size we will need in order to disassemble 1129357132ebSGreg Clayton const addr_t byte_size = num_instructions * m_arch.GetMaximumOpcodeByteSize(); 113037023b06SJim Ingham 1131357132ebSGreg Clayton if (target == NULL || byte_size == 0) 113237023b06SJim Ingham return 0; 113337023b06SJim Ingham 113437023b06SJim Ingham DataBufferHeap *heap_buffer = new DataBufferHeap (byte_size, '\0'); 113537023b06SJim Ingham DataBufferSP data_sp (heap_buffer); 113637023b06SJim Ingham 113737023b06SJim Ingham Error error; 11383faf47c4SGreg Clayton lldb::addr_t load_addr = LLDB_INVALID_ADDRESS; 1139357132ebSGreg Clayton const size_t bytes_read = target->ReadMemory (start, 1140357132ebSGreg Clayton prefer_file_cache, 1141357132ebSGreg Clayton heap_buffer->GetBytes(), 1142357132ebSGreg Clayton byte_size, 11433faf47c4SGreg Clayton error, 11443faf47c4SGreg Clayton &load_addr); 11453faf47c4SGreg Clayton 11463faf47c4SGreg Clayton const bool data_from_file = load_addr == LLDB_INVALID_ADDRESS; 114737023b06SJim Ingham 114837023b06SJim Ingham if (bytes_read == 0) 1149357132ebSGreg Clayton return 0; 1150357132ebSGreg Clayton DataExtractor data (data_sp, 1151357132ebSGreg Clayton m_arch.GetByteOrder(), 1152357132ebSGreg Clayton m_arch.GetAddressByteSize()); 115337023b06SJim Ingham 1154357132ebSGreg Clayton const bool append_instructions = true; 1155357132ebSGreg Clayton DecodeInstructions (start, 1156357132ebSGreg Clayton data, 1157357132ebSGreg Clayton 0, 1158357132ebSGreg Clayton num_instructions, 11593faf47c4SGreg Clayton append_instructions, 11603faf47c4SGreg Clayton data_from_file); 116137023b06SJim Ingham 116237023b06SJim Ingham return m_instruction_list.GetSize(); 116337023b06SJim Ingham } 116437023b06SJim Ingham 116530fdc8d8SChris Lattner //---------------------------------------------------------------------- 116630fdc8d8SChris Lattner // Disassembler copy constructor 116730fdc8d8SChris Lattner //---------------------------------------------------------------------- 11680f063ba6SJim Ingham Disassembler::Disassembler(const ArchSpec& arch, const char *flavor) : 116930fdc8d8SChris Lattner m_arch (arch), 117030fdc8d8SChris Lattner m_instruction_list(), 11710f063ba6SJim Ingham m_base_addr(LLDB_INVALID_ADDRESS), 11720f063ba6SJim Ingham m_flavor () 117330fdc8d8SChris Lattner { 11740f063ba6SJim Ingham if (flavor == NULL) 11750f063ba6SJim Ingham m_flavor.assign("default"); 11760f063ba6SJim Ingham else 11770f063ba6SJim Ingham m_flavor.assign(flavor); 117830fdc8d8SChris Lattner } 117930fdc8d8SChris Lattner 118030fdc8d8SChris Lattner //---------------------------------------------------------------------- 118130fdc8d8SChris Lattner // Destructor 118230fdc8d8SChris Lattner //---------------------------------------------------------------------- 118330fdc8d8SChris Lattner Disassembler::~Disassembler() 118430fdc8d8SChris Lattner { 118530fdc8d8SChris Lattner } 118630fdc8d8SChris Lattner 11871d273166SGreg Clayton InstructionList & 118830fdc8d8SChris Lattner Disassembler::GetInstructionList () 118930fdc8d8SChris Lattner { 119030fdc8d8SChris Lattner return m_instruction_list; 119130fdc8d8SChris Lattner } 119230fdc8d8SChris Lattner 11931d273166SGreg Clayton const InstructionList & 119430fdc8d8SChris Lattner Disassembler::GetInstructionList () const 119530fdc8d8SChris Lattner { 119630fdc8d8SChris Lattner return m_instruction_list; 119730fdc8d8SChris Lattner } 11983ac6711aSCaroline Tice 11993ac6711aSCaroline Tice //---------------------------------------------------------------------- 12003ac6711aSCaroline Tice // Class PseudoInstruction 12013ac6711aSCaroline Tice //---------------------------------------------------------------------- 12023ac6711aSCaroline Tice PseudoInstruction::PseudoInstruction () : 12033ac6711aSCaroline Tice Instruction (Address(), eAddressClassUnknown), 12043ac6711aSCaroline Tice m_description () 12053ac6711aSCaroline Tice { 12063ac6711aSCaroline Tice } 12073ac6711aSCaroline Tice 12083ac6711aSCaroline Tice PseudoInstruction::~PseudoInstruction () 12093ac6711aSCaroline Tice { 12103ac6711aSCaroline Tice } 12113ac6711aSCaroline Tice 12123ac6711aSCaroline Tice bool 121332ce20c5SJim Ingham PseudoInstruction::DoesBranch () 12143ac6711aSCaroline Tice { 12153ac6711aSCaroline Tice // This is NOT a valid question for a pseudo instruction. 12163ac6711aSCaroline Tice return false; 12173ac6711aSCaroline Tice } 12183ac6711aSCaroline Tice 12193ac6711aSCaroline Tice size_t 12203ac6711aSCaroline Tice PseudoInstruction::Decode (const lldb_private::Disassembler &disassembler, 12213ac6711aSCaroline Tice const lldb_private::DataExtractor &data, 1222c7bece56SGreg Clayton lldb::offset_t data_offset) 12233ac6711aSCaroline Tice { 12243ac6711aSCaroline Tice return m_opcode.GetByteSize(); 12253ac6711aSCaroline Tice } 12263ac6711aSCaroline Tice 12273ac6711aSCaroline Tice 12283ac6711aSCaroline Tice void 12293ac6711aSCaroline Tice PseudoInstruction::SetOpcode (size_t opcode_size, void *opcode_data) 12303ac6711aSCaroline Tice { 12313ac6711aSCaroline Tice if (!opcode_data) 12323ac6711aSCaroline Tice return; 12333ac6711aSCaroline Tice 12343ac6711aSCaroline Tice switch (opcode_size) 12353ac6711aSCaroline Tice { 12363ac6711aSCaroline Tice case 8: 12373ac6711aSCaroline Tice { 12383ac6711aSCaroline Tice uint8_t value8 = *((uint8_t *) opcode_data); 12393ac6711aSCaroline Tice m_opcode.SetOpcode8 (value8); 12403ac6711aSCaroline Tice break; 12413ac6711aSCaroline Tice } 12423ac6711aSCaroline Tice case 16: 12433ac6711aSCaroline Tice { 12443ac6711aSCaroline Tice uint16_t value16 = *((uint16_t *) opcode_data); 12453ac6711aSCaroline Tice m_opcode.SetOpcode16 (value16); 12463ac6711aSCaroline Tice break; 12473ac6711aSCaroline Tice } 12483ac6711aSCaroline Tice case 32: 12493ac6711aSCaroline Tice { 12503ac6711aSCaroline Tice uint32_t value32 = *((uint32_t *) opcode_data); 12513ac6711aSCaroline Tice m_opcode.SetOpcode32 (value32); 12523ac6711aSCaroline Tice break; 12533ac6711aSCaroline Tice } 12543ac6711aSCaroline Tice case 64: 12553ac6711aSCaroline Tice { 12563ac6711aSCaroline Tice uint64_t value64 = *((uint64_t *) opcode_data); 12573ac6711aSCaroline Tice m_opcode.SetOpcode64 (value64); 12583ac6711aSCaroline Tice break; 12593ac6711aSCaroline Tice } 12603ac6711aSCaroline Tice default: 12613ac6711aSCaroline Tice break; 12623ac6711aSCaroline Tice } 12633ac6711aSCaroline Tice } 12643ac6711aSCaroline Tice 12653ac6711aSCaroline Tice void 12663ac6711aSCaroline Tice PseudoInstruction::SetDescription (const char *description) 12673ac6711aSCaroline Tice { 12683ac6711aSCaroline Tice if (description && strlen (description) > 0) 12693ac6711aSCaroline Tice m_description = description; 12703ac6711aSCaroline Tice } 1271