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 1030fdc8d8SChris Lattner #include "lldb/Core/Disassembler.h" 1130fdc8d8SChris Lattner 1230fdc8d8SChris Lattner // C Includes 1330fdc8d8SChris Lattner // C++ Includes 1430fdc8d8SChris Lattner // Other libraries and framework includes 1530fdc8d8SChris Lattner // Project includes 1630fdc8d8SChris Lattner #include "lldb/lldb-private.h" 1730fdc8d8SChris Lattner #include "lldb/Core/Error.h" 1830fdc8d8SChris Lattner #include "lldb/Core/DataBufferHeap.h" 1930fdc8d8SChris Lattner #include "lldb/Core/DataExtractor.h" 2030fdc8d8SChris Lattner #include "lldb/Core/Debugger.h" 21ad379efcSCaroline Tice #include "lldb/Core/EmulateInstruction.h" 2230fdc8d8SChris Lattner #include "lldb/Core/Module.h" 2330fdc8d8SChris Lattner #include "lldb/Core/PluginManager.h" 24de2fb9cfSCaroline Tice #include "lldb/Core/RegularExpression.h" 2530fdc8d8SChris Lattner #include "lldb/Core/Timer.h" 2667cc0636SGreg Clayton #include "lldb/Interpreter/OptionValue.h" 2767cc0636SGreg Clayton #include "lldb/Interpreter/OptionValueArray.h" 2867cc0636SGreg Clayton #include "lldb/Interpreter/OptionValueDictionary.h" 2967cc0636SGreg Clayton #include "lldb/Interpreter/OptionValueString.h" 3067cc0636SGreg Clayton #include "lldb/Interpreter/OptionValueUInt64.h" 31b6d70ebcSSean Callanan #include "lldb/Symbol/ClangNamespaceDecl.h" 321f746071SGreg Clayton #include "lldb/Symbol/Function.h" 3330fdc8d8SChris Lattner #include "lldb/Symbol/ObjectFile.h" 3430fdc8d8SChris Lattner #include "lldb/Target/ExecutionContext.h" 3530fdc8d8SChris Lattner #include "lldb/Target/Process.h" 36d5944cd1SGreg Clayton #include "lldb/Target/SectionLoadList.h" 37b57e4a1bSJason Molenda #include "lldb/Target/StackFrame.h" 3830fdc8d8SChris Lattner #include "lldb/Target/Target.h" 3930fdc8d8SChris Lattner 4030fdc8d8SChris Lattner #define DEFAULT_DISASM_BYTE_SIZE 32 4130fdc8d8SChris Lattner 4230fdc8d8SChris Lattner using namespace lldb; 4330fdc8d8SChris Lattner using namespace lldb_private; 4430fdc8d8SChris Lattner 4530fdc8d8SChris Lattner 467e6d4e5aSSean Callanan DisassemblerSP 470f063ba6SJim Ingham Disassembler::FindPlugin (const ArchSpec &arch, const char *flavor, const char *plugin_name) 4830fdc8d8SChris Lattner { 4930fdc8d8SChris Lattner Timer scoped_timer (__PRETTY_FUNCTION__, 501080edbcSGreg Clayton "Disassembler::FindPlugin (arch = %s, plugin_name = %s)", 511080edbcSGreg Clayton arch.GetArchitectureName(), 521080edbcSGreg Clayton plugin_name); 5330fdc8d8SChris Lattner 541080edbcSGreg Clayton DisassemblerCreateInstance create_callback = NULL; 551080edbcSGreg Clayton 561080edbcSGreg Clayton if (plugin_name) 571080edbcSGreg Clayton { 5857abc5d6SGreg Clayton ConstString const_plugin_name (plugin_name); 5957abc5d6SGreg Clayton create_callback = PluginManager::GetDisassemblerCreateCallbackForPluginName (const_plugin_name); 601080edbcSGreg Clayton if (create_callback) 611080edbcSGreg Clayton { 620f063ba6SJim Ingham DisassemblerSP disassembler_sp(create_callback(arch, flavor)); 631080edbcSGreg Clayton 647e6d4e5aSSean Callanan if (disassembler_sp.get()) 657e6d4e5aSSean Callanan return disassembler_sp; 661080edbcSGreg Clayton } 671080edbcSGreg Clayton } 681080edbcSGreg Clayton else 691080edbcSGreg Clayton { 7030fdc8d8SChris Lattner for (uint32_t idx = 0; (create_callback = PluginManager::GetDisassemblerCreateCallbackAtIndex(idx)) != NULL; ++idx) 7130fdc8d8SChris Lattner { 720f063ba6SJim Ingham DisassemblerSP disassembler_sp(create_callback(arch, flavor)); 7330fdc8d8SChris Lattner 747e6d4e5aSSean Callanan if (disassembler_sp.get()) 757e6d4e5aSSean Callanan return disassembler_sp; 7630fdc8d8SChris Lattner } 771080edbcSGreg Clayton } 787e6d4e5aSSean Callanan return DisassemblerSP(); 7930fdc8d8SChris Lattner } 8030fdc8d8SChris Lattner 810f063ba6SJim Ingham DisassemblerSP 820f063ba6SJim Ingham Disassembler::FindPluginForTarget(const TargetSP target_sp, const ArchSpec &arch, const char *flavor, const char *plugin_name) 830f063ba6SJim Ingham { 840f063ba6SJim Ingham if (target_sp && flavor == NULL) 850f063ba6SJim Ingham { 860f063ba6SJim Ingham // FIXME - we don't have the mechanism in place to do per-architecture settings. But since we know that for now 870f063ba6SJim Ingham // we only support flavors on x86 & x86_64, 880f063ba6SJim Ingham if (arch.GetTriple().getArch() == llvm::Triple::x86 890f063ba6SJim Ingham || arch.GetTriple().getArch() == llvm::Triple::x86_64) 900f063ba6SJim Ingham flavor = target_sp->GetDisassemblyFlavor(); 910f063ba6SJim Ingham } 920f063ba6SJim Ingham return FindPlugin(arch, flavor, plugin_name); 930f063ba6SJim Ingham } 940f063ba6SJim Ingham 95dda4f7b5SGreg Clayton 96357132ebSGreg Clayton static void 97357132ebSGreg Clayton ResolveAddress (const ExecutionContext &exe_ctx, 98357132ebSGreg Clayton const Address &addr, 99357132ebSGreg Clayton Address &resolved_addr) 100357132ebSGreg Clayton { 101357132ebSGreg Clayton if (!addr.IsSectionOffset()) 102357132ebSGreg Clayton { 103357132ebSGreg Clayton // If we weren't passed in a section offset address range, 104357132ebSGreg Clayton // try and resolve it to something 105c14ee32dSGreg Clayton Target *target = exe_ctx.GetTargetPtr(); 106c14ee32dSGreg Clayton if (target) 107357132ebSGreg Clayton { 108c14ee32dSGreg Clayton if (target->GetSectionLoadList().IsEmpty()) 109357132ebSGreg Clayton { 110c14ee32dSGreg Clayton target->GetImages().ResolveFileAddress (addr.GetOffset(), resolved_addr); 111357132ebSGreg Clayton } 112357132ebSGreg Clayton else 113357132ebSGreg Clayton { 114c14ee32dSGreg Clayton target->GetSectionLoadList().ResolveLoadAddress (addr.GetOffset(), resolved_addr); 115357132ebSGreg Clayton } 116357132ebSGreg Clayton // We weren't able to resolve the address, just treat it as a 117357132ebSGreg Clayton // raw address 118357132ebSGreg Clayton if (resolved_addr.IsValid()) 119357132ebSGreg Clayton return; 120357132ebSGreg Clayton } 121357132ebSGreg Clayton } 122357132ebSGreg Clayton resolved_addr = addr; 123357132ebSGreg Clayton } 124dda4f7b5SGreg Clayton 125dda4f7b5SGreg Clayton size_t 126dda4f7b5SGreg Clayton Disassembler::Disassemble 127dda4f7b5SGreg Clayton ( 128dda4f7b5SGreg Clayton Debugger &debugger, 129dda4f7b5SGreg Clayton const ArchSpec &arch, 1301080edbcSGreg Clayton const char *plugin_name, 1310f063ba6SJim Ingham const char *flavor, 132dda4f7b5SGreg Clayton const ExecutionContext &exe_ctx, 133dda4f7b5SGreg Clayton SymbolContextList &sc_list, 13437023b06SJim Ingham uint32_t num_instructions, 135dda4f7b5SGreg Clayton uint32_t num_mixed_context_lines, 1361da6f9d7SGreg Clayton uint32_t options, 137dda4f7b5SGreg Clayton Stream &strm 138dda4f7b5SGreg Clayton ) 139dda4f7b5SGreg Clayton { 140dda4f7b5SGreg Clayton size_t success_count = 0; 141dda4f7b5SGreg Clayton const size_t count = sc_list.GetSize(); 142dda4f7b5SGreg Clayton SymbolContext sc; 143dda4f7b5SGreg Clayton AddressRange range; 1447e14f91dSGreg Clayton const uint32_t scope = eSymbolContextBlock | eSymbolContextFunction | eSymbolContextSymbol; 1457e14f91dSGreg Clayton const bool use_inline_block_range = true; 146dda4f7b5SGreg Clayton for (size_t i=0; i<count; ++i) 147dda4f7b5SGreg Clayton { 148dda4f7b5SGreg Clayton if (sc_list.GetContextAtIndex(i, sc) == false) 149dda4f7b5SGreg Clayton break; 1507e14f91dSGreg Clayton for (uint32_t range_idx = 0; sc.GetAddressRange(scope, range_idx, use_inline_block_range, range); ++range_idx) 151dda4f7b5SGreg Clayton { 1521080edbcSGreg Clayton if (Disassemble (debugger, 1531080edbcSGreg Clayton arch, 1541080edbcSGreg Clayton plugin_name, 1550f063ba6SJim Ingham flavor, 1561080edbcSGreg Clayton exe_ctx, 1571080edbcSGreg Clayton range, 1581080edbcSGreg Clayton num_instructions, 1591080edbcSGreg Clayton num_mixed_context_lines, 1601da6f9d7SGreg Clayton options, 1611080edbcSGreg Clayton strm)) 162dda4f7b5SGreg Clayton { 163dda4f7b5SGreg Clayton ++success_count; 164dda4f7b5SGreg Clayton strm.EOL(); 165dda4f7b5SGreg Clayton } 166dda4f7b5SGreg Clayton } 167dda4f7b5SGreg Clayton } 168dda4f7b5SGreg Clayton return success_count; 169dda4f7b5SGreg Clayton } 170dda4f7b5SGreg Clayton 17130fdc8d8SChris Lattner bool 17230fdc8d8SChris Lattner Disassembler::Disassemble 17330fdc8d8SChris Lattner ( 1746611103cSGreg Clayton Debugger &debugger, 17530fdc8d8SChris Lattner const ArchSpec &arch, 1761080edbcSGreg Clayton const char *plugin_name, 1770f063ba6SJim Ingham const char *flavor, 17830fdc8d8SChris Lattner const ExecutionContext &exe_ctx, 179dda4f7b5SGreg Clayton const ConstString &name, 180dda4f7b5SGreg Clayton Module *module, 18137023b06SJim Ingham uint32_t num_instructions, 182dda4f7b5SGreg Clayton uint32_t num_mixed_context_lines, 1831da6f9d7SGreg Clayton uint32_t options, 18430fdc8d8SChris Lattner Stream &strm 18530fdc8d8SChris Lattner ) 18630fdc8d8SChris Lattner { 187dda4f7b5SGreg Clayton SymbolContextList sc_list; 188931180e6SGreg Clayton if (name) 189931180e6SGreg Clayton { 190931180e6SGreg Clayton const bool include_symbols = true; 1919df05fbbSSean Callanan const bool include_inlines = true; 192dda4f7b5SGreg Clayton if (module) 193dda4f7b5SGreg Clayton { 194931180e6SGreg Clayton module->FindFunctions (name, 195b6d70ebcSSean Callanan NULL, 1966ecb232bSGreg Clayton eFunctionNameTypeAuto, 197931180e6SGreg Clayton include_symbols, 1989df05fbbSSean Callanan include_inlines, 199dda4f7b5SGreg Clayton true, 200931180e6SGreg Clayton sc_list); 201dda4f7b5SGreg Clayton } 202c14ee32dSGreg Clayton else if (exe_ctx.GetTargetPtr()) 203dda4f7b5SGreg Clayton { 204c14ee32dSGreg Clayton exe_ctx.GetTargetPtr()->GetImages().FindFunctions (name, 2056ecb232bSGreg Clayton eFunctionNameTypeAuto, 206931180e6SGreg Clayton include_symbols, 2079df05fbbSSean Callanan include_inlines, 2088ade104aSSean Callanan false, 209931180e6SGreg Clayton sc_list); 210dda4f7b5SGreg Clayton } 211dda4f7b5SGreg Clayton } 212931180e6SGreg Clayton 213931180e6SGreg Clayton if (sc_list.GetSize ()) 214931180e6SGreg Clayton { 215931180e6SGreg Clayton return Disassemble (debugger, 216931180e6SGreg Clayton arch, 2171080edbcSGreg Clayton plugin_name, 2180f063ba6SJim Ingham flavor, 219931180e6SGreg Clayton exe_ctx, 220931180e6SGreg Clayton sc_list, 22137023b06SJim Ingham num_instructions, 222931180e6SGreg Clayton num_mixed_context_lines, 2231da6f9d7SGreg Clayton options, 224931180e6SGreg Clayton strm); 225dda4f7b5SGreg Clayton } 226dda4f7b5SGreg Clayton return false; 227dda4f7b5SGreg Clayton } 228dda4f7b5SGreg Clayton 2291d273166SGreg Clayton 2301d273166SGreg Clayton lldb::DisassemblerSP 2311d273166SGreg Clayton Disassembler::DisassembleRange 2321d273166SGreg Clayton ( 2331d273166SGreg Clayton const ArchSpec &arch, 2341080edbcSGreg Clayton const char *plugin_name, 2350f063ba6SJim Ingham const char *flavor, 2361d273166SGreg Clayton const ExecutionContext &exe_ctx, 2376b3e6d54SJason Molenda const AddressRange &range, 2386b3e6d54SJason Molenda bool prefer_file_cache 2391d273166SGreg Clayton ) 2401d273166SGreg Clayton { 2411d273166SGreg Clayton lldb::DisassemblerSP disasm_sp; 2421d273166SGreg Clayton if (range.GetByteSize() > 0 && range.GetBaseAddress().IsValid()) 2431d273166SGreg Clayton { 2440f063ba6SJim Ingham disasm_sp = Disassembler::FindPluginForTarget(exe_ctx.GetTargetSP(), arch, flavor, plugin_name); 2451d273166SGreg Clayton 2461d273166SGreg Clayton if (disasm_sp) 2471d273166SGreg Clayton { 2483faf47c4SGreg Clayton size_t bytes_disassembled = disasm_sp->ParseInstructions (&exe_ctx, range, NULL, prefer_file_cache); 2491d273166SGreg Clayton if (bytes_disassembled == 0) 2501d273166SGreg Clayton disasm_sp.reset(); 2511d273166SGreg Clayton } 2521d273166SGreg Clayton } 2531d273166SGreg Clayton return disasm_sp; 2541d273166SGreg Clayton } 2551d273166SGreg Clayton 25650952e95SSean Callanan lldb::DisassemblerSP 2573faf47c4SGreg Clayton Disassembler::DisassembleBytes (const ArchSpec &arch, 25850952e95SSean Callanan const char *plugin_name, 2590f063ba6SJim Ingham const char *flavor, 26050952e95SSean Callanan const Address &start, 2613faf47c4SGreg Clayton const void *src, 2623faf47c4SGreg Clayton size_t src_len, 2633faf47c4SGreg Clayton uint32_t num_instructions, 2643faf47c4SGreg Clayton bool data_from_file) 26550952e95SSean Callanan { 26650952e95SSean Callanan lldb::DisassemblerSP disasm_sp; 26750952e95SSean Callanan 2683faf47c4SGreg Clayton if (src) 26950952e95SSean Callanan { 2700f063ba6SJim Ingham disasm_sp = Disassembler::FindPlugin(arch, flavor, plugin_name); 27150952e95SSean Callanan 27250952e95SSean Callanan if (disasm_sp) 27350952e95SSean Callanan { 2743faf47c4SGreg Clayton DataExtractor data(src, src_len, arch.GetByteOrder(), arch.GetAddressByteSize()); 27550952e95SSean Callanan 27650952e95SSean Callanan (void)disasm_sp->DecodeInstructions (start, 27750952e95SSean Callanan data, 27850952e95SSean Callanan 0, 2799c766110SGreg Clayton num_instructions, 2803faf47c4SGreg Clayton false, 2813faf47c4SGreg Clayton data_from_file); 28250952e95SSean Callanan } 28350952e95SSean Callanan } 28450952e95SSean Callanan 28550952e95SSean Callanan return disasm_sp; 28650952e95SSean Callanan } 28750952e95SSean Callanan 2881d273166SGreg Clayton 289dda4f7b5SGreg Clayton bool 290dda4f7b5SGreg Clayton Disassembler::Disassemble 291dda4f7b5SGreg Clayton ( 292dda4f7b5SGreg Clayton Debugger &debugger, 293dda4f7b5SGreg Clayton const ArchSpec &arch, 2941080edbcSGreg Clayton const char *plugin_name, 2950f063ba6SJim Ingham const char *flavor, 296dda4f7b5SGreg Clayton const ExecutionContext &exe_ctx, 297dda4f7b5SGreg Clayton const AddressRange &disasm_range, 29837023b06SJim Ingham uint32_t num_instructions, 299dda4f7b5SGreg Clayton uint32_t num_mixed_context_lines, 3001da6f9d7SGreg Clayton uint32_t options, 301dda4f7b5SGreg Clayton Stream &strm 302dda4f7b5SGreg Clayton ) 303dda4f7b5SGreg Clayton { 304dda4f7b5SGreg Clayton if (disasm_range.GetByteSize()) 305dda4f7b5SGreg Clayton { 3060f063ba6SJim Ingham lldb::DisassemblerSP disasm_sp (Disassembler::FindPluginForTarget(exe_ctx.GetTargetSP(), arch, flavor, plugin_name)); 30730fdc8d8SChris Lattner 3087e6d4e5aSSean Callanan if (disasm_sp.get()) 30930fdc8d8SChris Lattner { 310357132ebSGreg Clayton AddressRange range; 311357132ebSGreg Clayton ResolveAddress (exe_ctx, disasm_range.GetBaseAddress(), range.GetBaseAddress()); 312357132ebSGreg Clayton range.SetByteSize (disasm_range.GetByteSize()); 3133faf47c4SGreg Clayton const bool prefer_file_cache = false; 3143faf47c4SGreg Clayton size_t bytes_disassembled = disasm_sp->ParseInstructions (&exe_ctx, range, &strm, prefer_file_cache); 31530fdc8d8SChris Lattner if (bytes_disassembled == 0) 31630fdc8d8SChris Lattner return false; 3171080edbcSGreg Clayton 31856d40428SJim Ingham bool result = PrintInstructions (disasm_sp.get(), 31937023b06SJim Ingham debugger, 32037023b06SJim Ingham arch, 32137023b06SJim Ingham exe_ctx, 32237023b06SJim Ingham num_instructions, 32337023b06SJim Ingham num_mixed_context_lines, 3241da6f9d7SGreg Clayton options, 32537023b06SJim Ingham strm); 32656d40428SJim Ingham 32756d40428SJim Ingham // FIXME: The DisassemblerLLVMC has a reference cycle and won't go away if it has any active instructions. 32856d40428SJim Ingham // I'll fix that but for now, just clear the list and it will go away nicely. 32956d40428SJim Ingham disasm_sp->GetInstructionList().Clear(); 33056d40428SJim Ingham return result; 33137023b06SJim Ingham } 33237023b06SJim Ingham } 33337023b06SJim Ingham return false; 33437023b06SJim Ingham } 33537023b06SJim Ingham 33637023b06SJim Ingham bool 33737023b06SJim Ingham Disassembler::Disassemble 33837023b06SJim Ingham ( 33937023b06SJim Ingham Debugger &debugger, 34037023b06SJim Ingham const ArchSpec &arch, 3411080edbcSGreg Clayton const char *plugin_name, 3420f063ba6SJim Ingham const char *flavor, 34337023b06SJim Ingham const ExecutionContext &exe_ctx, 34437023b06SJim Ingham const Address &start_address, 34537023b06SJim Ingham uint32_t num_instructions, 34637023b06SJim Ingham uint32_t num_mixed_context_lines, 3471da6f9d7SGreg Clayton uint32_t options, 34837023b06SJim Ingham Stream &strm 34937023b06SJim Ingham ) 35037023b06SJim Ingham { 35137023b06SJim Ingham if (num_instructions > 0) 35237023b06SJim Ingham { 3533faf47c4SGreg Clayton lldb::DisassemblerSP disasm_sp (Disassembler::FindPluginForTarget(exe_ctx.GetTargetSP(), 3543faf47c4SGreg Clayton arch, 3553faf47c4SGreg Clayton flavor, 3563faf47c4SGreg Clayton plugin_name)); 3577e6d4e5aSSean Callanan if (disasm_sp.get()) 35837023b06SJim Ingham { 359357132ebSGreg Clayton Address addr; 360357132ebSGreg Clayton ResolveAddress (exe_ctx, start_address, addr); 3613faf47c4SGreg Clayton const bool prefer_file_cache = false; 3623faf47c4SGreg Clayton size_t bytes_disassembled = disasm_sp->ParseInstructions (&exe_ctx, 3633faf47c4SGreg Clayton addr, 3643faf47c4SGreg Clayton num_instructions, 3653faf47c4SGreg Clayton prefer_file_cache); 36637023b06SJim Ingham if (bytes_disassembled == 0) 36737023b06SJim Ingham return false; 36856d40428SJim Ingham bool result = PrintInstructions (disasm_sp.get(), 36937023b06SJim Ingham debugger, 37037023b06SJim Ingham arch, 37137023b06SJim Ingham exe_ctx, 37237023b06SJim Ingham num_instructions, 37337023b06SJim Ingham num_mixed_context_lines, 3741da6f9d7SGreg Clayton options, 37537023b06SJim Ingham strm); 37656d40428SJim Ingham 37756d40428SJim Ingham // FIXME: The DisassemblerLLVMC has a reference cycle and won't go away if it has any active instructions. 37856d40428SJim Ingham // I'll fix that but for now, just clear the list and it will go away nicely. 37956d40428SJim Ingham disasm_sp->GetInstructionList().Clear(); 38056d40428SJim Ingham return result; 38137023b06SJim Ingham } 38237023b06SJim Ingham } 38337023b06SJim Ingham return false; 38437023b06SJim Ingham } 38537023b06SJim Ingham 38637023b06SJim Ingham bool 38737023b06SJim Ingham Disassembler::PrintInstructions 38837023b06SJim Ingham ( 38937023b06SJim Ingham Disassembler *disasm_ptr, 39037023b06SJim Ingham Debugger &debugger, 39137023b06SJim Ingham const ArchSpec &arch, 39237023b06SJim Ingham const ExecutionContext &exe_ctx, 39337023b06SJim Ingham uint32_t num_instructions, 39437023b06SJim Ingham uint32_t num_mixed_context_lines, 3951da6f9d7SGreg Clayton uint32_t options, 39637023b06SJim Ingham Stream &strm 39737023b06SJim Ingham ) 39837023b06SJim Ingham { 39930fdc8d8SChris Lattner // We got some things disassembled... 40037023b06SJim Ingham size_t num_instructions_found = disasm_ptr->GetInstructionList().GetSize(); 40137023b06SJim Ingham 40237023b06SJim Ingham if (num_instructions > 0 && num_instructions < num_instructions_found) 40337023b06SJim Ingham num_instructions_found = num_instructions; 40437023b06SJim Ingham 405357132ebSGreg Clayton const uint32_t max_opcode_byte_size = disasm_ptr->GetInstructionList().GetMaxOpcocdeByteSize (); 40630fdc8d8SChris Lattner uint32_t offset = 0; 40730fdc8d8SChris Lattner SymbolContext sc; 40830fdc8d8SChris Lattner SymbolContext prev_sc; 40930fdc8d8SChris Lattner AddressRange sc_range; 41034132754SGreg Clayton const Address *pc_addr_ptr = NULL; 411b57e4a1bSJason Molenda StackFrame *frame = exe_ctx.GetFramePtr(); 412c14ee32dSGreg Clayton 4134b2967ffSMichael Sartain TargetSP target_sp (exe_ctx.GetTargetSP()); 4144b2967ffSMichael Sartain SourceManager &source_manager = target_sp ? target_sp->GetSourceManager() : debugger.GetSourceManager(); 4154b2967ffSMichael Sartain 416c14ee32dSGreg Clayton if (frame) 417aff1b357SJason Molenda { 418c14ee32dSGreg Clayton pc_addr_ptr = &frame->GetFrameCodeAddress(); 419aff1b357SJason Molenda } 4207e14f91dSGreg Clayton const uint32_t scope = eSymbolContextLineEntry | eSymbolContextFunction | eSymbolContextSymbol; 4217e14f91dSGreg Clayton const bool use_inline_block_range = false; 422c980fa92SJason Molenda 423c980fa92SJason Molenda const FormatEntity::Entry *disassembly_format = NULL; 424c980fa92SJason Molenda FormatEntity::Entry format; 425c980fa92SJason Molenda if (exe_ctx.HasTargetScope()) 426c980fa92SJason Molenda { 427c980fa92SJason Molenda disassembly_format = exe_ctx.GetTargetRef().GetDebugger().GetDisassemblyFormat (); 428c980fa92SJason Molenda } 429c980fa92SJason Molenda else 430c980fa92SJason Molenda { 431c980fa92SJason Molenda FormatEntity::Parse("${addr}: ", format); 432c980fa92SJason Molenda disassembly_format = &format; 433c980fa92SJason Molenda } 434c980fa92SJason Molenda 435c980fa92SJason Molenda // First pass: step through the list of instructions, 436c980fa92SJason Molenda // find how long the initial addresses strings are, insert padding 437c980fa92SJason Molenda // in the second pass so the opcodes all line up nicely. 438c980fa92SJason Molenda size_t address_text_size = 0; 439c980fa92SJason Molenda for (size_t i = 0; i < num_instructions_found; ++i) 440c980fa92SJason Molenda { 441c980fa92SJason Molenda Instruction *inst = disasm_ptr->GetInstructionList().GetInstructionAtIndex (i).get(); 442c980fa92SJason Molenda if (inst) 443c980fa92SJason Molenda { 444c980fa92SJason Molenda const Address &addr = inst->GetAddress(); 445c980fa92SJason Molenda ModuleSP module_sp (addr.GetModule()); 446c980fa92SJason Molenda if (module_sp) 447c980fa92SJason Molenda { 448c980fa92SJason Molenda const uint32_t resolve_mask = eSymbolContextFunction | eSymbolContextSymbol; 449c980fa92SJason Molenda uint32_t resolved_mask = module_sp->ResolveSymbolContextForAddress(addr, resolve_mask, sc); 450c980fa92SJason Molenda if (resolved_mask) 451c980fa92SJason Molenda { 452c980fa92SJason Molenda StreamString strmstr; 453c980fa92SJason Molenda Debugger::FormatDisassemblerAddress (disassembly_format, &sc, NULL, &exe_ctx, &addr, strmstr); 454c980fa92SJason Molenda size_t cur_line = strmstr.GetSizeOfLastLine(); 455c980fa92SJason Molenda if (cur_line > address_text_size) 456c980fa92SJason Molenda address_text_size = cur_line; 457c980fa92SJason Molenda } 458c980fa92SJason Molenda sc.Clear(false); 459c980fa92SJason Molenda } 460c980fa92SJason Molenda } 461c980fa92SJason Molenda } 462c980fa92SJason Molenda 46337023b06SJim Ingham for (size_t i = 0; i < num_instructions_found; ++i) 46430fdc8d8SChris Lattner { 46537023b06SJim Ingham Instruction *inst = disasm_ptr->GetInstructionList().GetInstructionAtIndex (i).get(); 46630fdc8d8SChris Lattner if (inst) 46730fdc8d8SChris Lattner { 46832e0a750SGreg Clayton const Address &addr = inst->GetAddress(); 46932e0a750SGreg Clayton const bool inst_is_at_pc = pc_addr_ptr && addr == *pc_addr_ptr; 470dda4f7b5SGreg Clayton 47130fdc8d8SChris Lattner prev_sc = sc; 472dda4f7b5SGreg Clayton 473e72dfb32SGreg Clayton ModuleSP module_sp (addr.GetModule()); 474e72dfb32SGreg Clayton if (module_sp) 47530fdc8d8SChris Lattner { 476e72dfb32SGreg Clayton uint32_t resolved_mask = module_sp->ResolveSymbolContextForAddress(addr, eSymbolContextEverything, sc); 47730fdc8d8SChris Lattner if (resolved_mask) 47830fdc8d8SChris Lattner { 47932e0a750SGreg Clayton if (num_mixed_context_lines) 480dda4f7b5SGreg Clayton { 48132e0a750SGreg Clayton if (!sc_range.ContainsFileAddress (addr)) 482dda4f7b5SGreg Clayton { 4837e14f91dSGreg Clayton sc.GetAddressRange (scope, 0, use_inline_block_range, sc_range); 484dda4f7b5SGreg Clayton 48530fdc8d8SChris Lattner if (sc != prev_sc) 48630fdc8d8SChris Lattner { 48730fdc8d8SChris Lattner if (offset != 0) 48830fdc8d8SChris Lattner strm.EOL(); 48930fdc8d8SChris Lattner 490c980fa92SJason Molenda sc.DumpStopContext(&strm, exe_ctx.GetProcessPtr(), addr, false, true, false, false, true); 4916dbd3983SGreg Clayton strm.EOL(); 49230fdc8d8SChris Lattner 49330fdc8d8SChris Lattner if (sc.comp_unit && sc.line_entry.IsValid()) 49430fdc8d8SChris Lattner { 4954b2967ffSMichael Sartain source_manager.DisplaySourceLinesWithLineNumbers (sc.line_entry.file, 49630fdc8d8SChris Lattner sc.line_entry.line, 497dda4f7b5SGreg Clayton num_mixed_context_lines, 498dda4f7b5SGreg Clayton num_mixed_context_lines, 499b10d72f0SGreg Clayton ((inst_is_at_pc && (options & eOptionMarkPCSourceLine)) ? "->" : ""), 50030fdc8d8SChris Lattner &strm); 50130fdc8d8SChris Lattner } 50230fdc8d8SChris Lattner } 50330fdc8d8SChris Lattner } 50430fdc8d8SChris Lattner } 50532e0a750SGreg Clayton } 506dda4f7b5SGreg Clayton else 507dda4f7b5SGreg Clayton { 50872310355SGreg Clayton sc.Clear(true); 50930fdc8d8SChris Lattner } 51030fdc8d8SChris Lattner } 51132e0a750SGreg Clayton 5121da6f9d7SGreg Clayton const bool show_bytes = (options & eOptionShowBytes) != 0; 513c980fa92SJason Molenda inst->Dump (&strm, max_opcode_byte_size, true, show_bytes, &exe_ctx, &sc, &prev_sc, NULL, address_text_size); 51430fdc8d8SChris Lattner strm.EOL(); 51530fdc8d8SChris Lattner } 51630fdc8d8SChris Lattner else 51730fdc8d8SChris Lattner { 51830fdc8d8SChris Lattner break; 51930fdc8d8SChris Lattner } 52030fdc8d8SChris Lattner } 52130fdc8d8SChris Lattner 52230fdc8d8SChris Lattner return true; 52330fdc8d8SChris Lattner } 52430fdc8d8SChris Lattner 525dda4f7b5SGreg Clayton 526dda4f7b5SGreg Clayton bool 527dda4f7b5SGreg Clayton Disassembler::Disassemble 528dda4f7b5SGreg Clayton ( 529dda4f7b5SGreg Clayton Debugger &debugger, 530dda4f7b5SGreg Clayton const ArchSpec &arch, 5311080edbcSGreg Clayton const char *plugin_name, 5320f063ba6SJim Ingham const char *flavor, 533dda4f7b5SGreg Clayton const ExecutionContext &exe_ctx, 53437023b06SJim Ingham uint32_t num_instructions, 535dda4f7b5SGreg Clayton uint32_t num_mixed_context_lines, 5361da6f9d7SGreg Clayton uint32_t options, 537dda4f7b5SGreg Clayton Stream &strm 538dda4f7b5SGreg Clayton ) 539dda4f7b5SGreg Clayton { 540dda4f7b5SGreg Clayton AddressRange range; 541b57e4a1bSJason Molenda StackFrame *frame = exe_ctx.GetFramePtr(); 542c14ee32dSGreg Clayton if (frame) 543dda4f7b5SGreg Clayton { 544c14ee32dSGreg Clayton SymbolContext sc(frame->GetSymbolContext(eSymbolContextFunction | eSymbolContextSymbol)); 545dda4f7b5SGreg Clayton if (sc.function) 546dda4f7b5SGreg Clayton { 547dda4f7b5SGreg Clayton range = sc.function->GetAddressRange(); 548dda4f7b5SGreg Clayton } 549e7612134SGreg Clayton else if (sc.symbol && sc.symbol->ValueIsAddress()) 550dda4f7b5SGreg Clayton { 551*358cf1eaSGreg Clayton range.GetBaseAddress() = sc.symbol->GetAddressRef(); 552e7612134SGreg Clayton range.SetByteSize (sc.symbol->GetByteSize()); 553dda4f7b5SGreg Clayton } 554dda4f7b5SGreg Clayton else 555dda4f7b5SGreg Clayton { 556c14ee32dSGreg Clayton range.GetBaseAddress() = frame->GetFrameCodeAddress(); 557dda4f7b5SGreg Clayton } 558dda4f7b5SGreg Clayton 559dda4f7b5SGreg Clayton if (range.GetBaseAddress().IsValid() && range.GetByteSize() == 0) 560dda4f7b5SGreg Clayton range.SetByteSize (DEFAULT_DISASM_BYTE_SIZE); 561dda4f7b5SGreg Clayton } 562dda4f7b5SGreg Clayton 5631080edbcSGreg Clayton return Disassemble (debugger, 5641080edbcSGreg Clayton arch, 5651080edbcSGreg Clayton plugin_name, 5660f063ba6SJim Ingham flavor, 5671080edbcSGreg Clayton exe_ctx, 5681080edbcSGreg Clayton range, 5691080edbcSGreg Clayton num_instructions, 5701080edbcSGreg Clayton num_mixed_context_lines, 5711da6f9d7SGreg Clayton options, 5721080edbcSGreg Clayton strm); 573dda4f7b5SGreg Clayton } 574dda4f7b5SGreg Clayton 575357132ebSGreg Clayton Instruction::Instruction(const Address &address, AddressClass addr_class) : 5761080edbcSGreg Clayton m_address (address), 577357132ebSGreg Clayton m_address_class (addr_class), 578a97aa92aSSean Callanan m_opcode(), 579a97aa92aSSean Callanan m_calculated_strings(false) 5800ae96273SGreg Clayton { 58130fdc8d8SChris Lattner } 58230fdc8d8SChris Lattner 5831d273166SGreg Clayton Instruction::~Instruction() 58430fdc8d8SChris Lattner { 58530fdc8d8SChris Lattner } 58630fdc8d8SChris Lattner 587357132ebSGreg Clayton AddressClass 588357132ebSGreg Clayton Instruction::GetAddressClass () 589357132ebSGreg Clayton { 590357132ebSGreg Clayton if (m_address_class == eAddressClassInvalid) 591357132ebSGreg Clayton m_address_class = m_address.GetAddressClass(); 592357132ebSGreg Clayton return m_address_class; 593357132ebSGreg Clayton } 59430fdc8d8SChris Lattner 595ba812f42SGreg Clayton void 596ba812f42SGreg Clayton Instruction::Dump (lldb_private::Stream *s, 597ba812f42SGreg Clayton uint32_t max_opcode_byte_size, 598ba812f42SGreg Clayton bool show_address, 599ba812f42SGreg Clayton bool show_bytes, 600aff1b357SJason Molenda const ExecutionContext* exe_ctx, 601aff1b357SJason Molenda const SymbolContext *sym_ctx, 602aff1b357SJason Molenda const SymbolContext *prev_sym_ctx, 603c980fa92SJason Molenda const FormatEntity::Entry *disassembly_addr_format, 604c980fa92SJason Molenda size_t max_address_text_size) 605ba812f42SGreg Clayton { 6067a37c1ecSJason Molenda size_t opcode_column_width = 7; 607ba812f42SGreg Clayton const size_t operand_column_width = 25; 608ba812f42SGreg Clayton 609ba812f42SGreg Clayton CalculateMnemonicOperandsAndCommentIfNeeded (exe_ctx); 610ba812f42SGreg Clayton 611ba812f42SGreg Clayton StreamString ss; 612ba812f42SGreg Clayton 613ba812f42SGreg Clayton if (show_address) 614ba812f42SGreg Clayton { 615554f68d3SGreg Clayton Debugger::FormatDisassemblerAddress (disassembly_addr_format, sym_ctx, prev_sym_ctx, exe_ctx, &m_address, ss); 616c980fa92SJason Molenda ss.FillLastLineToColumn (max_address_text_size, ' '); 617ba812f42SGreg Clayton } 618ba812f42SGreg Clayton 619ba812f42SGreg Clayton if (show_bytes) 620ba812f42SGreg Clayton { 621ba812f42SGreg Clayton if (m_opcode.GetType() == Opcode::eTypeBytes) 622ba812f42SGreg Clayton { 623ba812f42SGreg Clayton // x86_64 and i386 are the only ones that use bytes right now so 624ba812f42SGreg Clayton // pad out the byte dump to be able to always show 15 bytes (3 chars each) 625ba812f42SGreg Clayton // plus a space 626ba812f42SGreg Clayton if (max_opcode_byte_size > 0) 627ba812f42SGreg Clayton m_opcode.Dump (&ss, max_opcode_byte_size * 3 + 1); 628ba812f42SGreg Clayton else 629ba812f42SGreg Clayton m_opcode.Dump (&ss, 15 * 3 + 1); 630ba812f42SGreg Clayton } 631ba812f42SGreg Clayton else 632ba812f42SGreg Clayton { 633d616c97aSEd Maste // Else, we have ARM or MIPS which can show up to a uint32_t 634d616c97aSEd Maste // 0x00000000 (10 spaces) plus two for padding... 635ba812f42SGreg Clayton if (max_opcode_byte_size > 0) 636ba812f42SGreg Clayton m_opcode.Dump (&ss, max_opcode_byte_size * 3 + 1); 637ba812f42SGreg Clayton else 638ba812f42SGreg Clayton m_opcode.Dump (&ss, 12); 639ba812f42SGreg Clayton } 640ba812f42SGreg Clayton } 641ba812f42SGreg Clayton 642aff1b357SJason Molenda const size_t opcode_pos = ss.GetSizeOfLastLine(); 643ba812f42SGreg Clayton 6447a37c1ecSJason Molenda // The default opcode size of 7 characters is plenty for most architectures 6457a37c1ecSJason Molenda // but some like arm can pull out the occasional vqrshrun.s16. We won't get 6467a37c1ecSJason Molenda // consistent column spacing in these cases, unfortunately. 6477a37c1ecSJason Molenda if (m_opcode_name.length() >= opcode_column_width) 6487a37c1ecSJason Molenda { 6497a37c1ecSJason Molenda opcode_column_width = m_opcode_name.length() + 1; 6507a37c1ecSJason Molenda } 6517a37c1ecSJason Molenda 652ba812f42SGreg Clayton ss.PutCString (m_opcode_name.c_str()); 653ba812f42SGreg Clayton ss.FillLastLineToColumn (opcode_pos + opcode_column_width, ' '); 6540f063ba6SJim Ingham ss.PutCString (m_mnemonics.c_str()); 655ba812f42SGreg Clayton 656ba812f42SGreg Clayton if (!m_comment.empty()) 657ba812f42SGreg Clayton { 658ba812f42SGreg Clayton ss.FillLastLineToColumn (opcode_pos + opcode_column_width + operand_column_width, ' '); 659ba812f42SGreg Clayton ss.PutCString (" ; "); 660ba812f42SGreg Clayton ss.PutCString (m_comment.c_str()); 661ba812f42SGreg Clayton } 662ba812f42SGreg Clayton s->Write (ss.GetData(), ss.GetSize()); 663ba812f42SGreg Clayton } 664ba812f42SGreg Clayton 6657c9dd3ceSCaroline Tice bool 6667c9dd3ceSCaroline Tice Instruction::DumpEmulation (const ArchSpec &arch) 6677c9dd3ceSCaroline Tice { 6687b0992d9SGreg Clayton std::unique_ptr<EmulateInstruction> insn_emulator_ap (EmulateInstruction::FindPlugin (arch, eInstructionTypeAny, NULL)); 6697c9dd3ceSCaroline Tice if (insn_emulator_ap.get()) 6707c9dd3ceSCaroline Tice { 6712ed751bdSGreg Clayton insn_emulator_ap->SetInstruction (GetOpcode(), GetAddress(), NULL); 6722ed751bdSGreg Clayton return insn_emulator_ap->EvaluateInstruction (0); 6737c9dd3ceSCaroline Tice } 6747c9dd3ceSCaroline Tice 6757c9dd3ceSCaroline Tice return false; 6767c9dd3ceSCaroline Tice } 6777c9dd3ceSCaroline Tice 678de2fb9cfSCaroline Tice OptionValueSP 679de2fb9cfSCaroline Tice Instruction::ReadArray (FILE *in_file, Stream *out_stream, OptionValue::Type data_type) 680de2fb9cfSCaroline Tice { 681de2fb9cfSCaroline Tice bool done = false; 682de2fb9cfSCaroline Tice char buffer[1024]; 683de2fb9cfSCaroline Tice 684de2fb9cfSCaroline Tice OptionValueSP option_value_sp (new OptionValueArray (1u << data_type)); 685de2fb9cfSCaroline Tice 686de2fb9cfSCaroline Tice int idx = 0; 687de2fb9cfSCaroline Tice while (!done) 688de2fb9cfSCaroline Tice { 689de2fb9cfSCaroline Tice if (!fgets (buffer, 1023, in_file)) 690de2fb9cfSCaroline Tice { 691762f7135SGreg Clayton out_stream->Printf ("Instruction::ReadArray: Error reading file (fgets).\n"); 692de2fb9cfSCaroline Tice option_value_sp.reset (); 693de2fb9cfSCaroline Tice return option_value_sp; 694de2fb9cfSCaroline Tice } 695de2fb9cfSCaroline Tice 696de2fb9cfSCaroline Tice std::string line (buffer); 697de2fb9cfSCaroline Tice 698c7bece56SGreg Clayton size_t len = line.size(); 699de2fb9cfSCaroline Tice if (line[len-1] == '\n') 700de2fb9cfSCaroline Tice { 701de2fb9cfSCaroline Tice line[len-1] = '\0'; 702de2fb9cfSCaroline Tice line.resize (len-1); 703de2fb9cfSCaroline Tice } 704de2fb9cfSCaroline Tice 705de2fb9cfSCaroline Tice if ((line.size() == 1) && line[0] == ']') 706de2fb9cfSCaroline Tice { 707de2fb9cfSCaroline Tice done = true; 708de2fb9cfSCaroline Tice line.clear(); 709de2fb9cfSCaroline Tice } 710de2fb9cfSCaroline Tice 711de2fb9cfSCaroline Tice if (line.size() > 0) 712de2fb9cfSCaroline Tice { 713de2fb9cfSCaroline Tice std::string value; 714bc43cab5SGreg Clayton static RegularExpression g_reg_exp ("^[ \t]*([^ \t]+)[ \t]*$"); 715bc43cab5SGreg Clayton RegularExpression::Match regex_match(1); 716bc43cab5SGreg Clayton bool reg_exp_success = g_reg_exp.Execute (line.c_str(), ®ex_match); 717de2fb9cfSCaroline Tice if (reg_exp_success) 718bc43cab5SGreg Clayton regex_match.GetMatchAtIndex (line.c_str(), 1, value); 719de2fb9cfSCaroline Tice else 720de2fb9cfSCaroline Tice value = line; 721de2fb9cfSCaroline Tice 722de2fb9cfSCaroline Tice OptionValueSP data_value_sp; 723de2fb9cfSCaroline Tice switch (data_type) 724de2fb9cfSCaroline Tice { 725de2fb9cfSCaroline Tice case OptionValue::eTypeUInt64: 726de2fb9cfSCaroline Tice data_value_sp.reset (new OptionValueUInt64 (0, 0)); 727c95f7e2aSPavel Labath data_value_sp->SetValueFromString (value); 728de2fb9cfSCaroline Tice break; 729de2fb9cfSCaroline Tice // Other types can be added later as needed. 730de2fb9cfSCaroline Tice default: 731de2fb9cfSCaroline Tice data_value_sp.reset (new OptionValueString (value.c_str(), "")); 732de2fb9cfSCaroline Tice break; 733de2fb9cfSCaroline Tice } 734de2fb9cfSCaroline Tice 73584c39663SGreg Clayton option_value_sp->GetAsArray()->InsertValue (idx, data_value_sp); 736de2fb9cfSCaroline Tice ++idx; 737de2fb9cfSCaroline Tice } 738de2fb9cfSCaroline Tice } 739de2fb9cfSCaroline Tice 740de2fb9cfSCaroline Tice return option_value_sp; 741de2fb9cfSCaroline Tice } 742de2fb9cfSCaroline Tice 743de2fb9cfSCaroline Tice OptionValueSP 744de2fb9cfSCaroline Tice Instruction::ReadDictionary (FILE *in_file, Stream *out_stream) 745de2fb9cfSCaroline Tice { 746de2fb9cfSCaroline Tice bool done = false; 747de2fb9cfSCaroline Tice char buffer[1024]; 748de2fb9cfSCaroline Tice 749de2fb9cfSCaroline Tice OptionValueSP option_value_sp (new OptionValueDictionary()); 750de2fb9cfSCaroline Tice static ConstString encoding_key ("data_encoding"); 751de2fb9cfSCaroline Tice OptionValue::Type data_type = OptionValue::eTypeInvalid; 752de2fb9cfSCaroline Tice 753de2fb9cfSCaroline Tice 754de2fb9cfSCaroline Tice while (!done) 755de2fb9cfSCaroline Tice { 756de2fb9cfSCaroline Tice // Read the next line in the file 757de2fb9cfSCaroline Tice if (!fgets (buffer, 1023, in_file)) 758de2fb9cfSCaroline Tice { 759de2fb9cfSCaroline Tice out_stream->Printf ("Instruction::ReadDictionary: Error reading file (fgets).\n"); 760de2fb9cfSCaroline Tice option_value_sp.reset (); 761de2fb9cfSCaroline Tice return option_value_sp; 762de2fb9cfSCaroline Tice } 763de2fb9cfSCaroline Tice 764de2fb9cfSCaroline Tice // Check to see if the line contains the end-of-dictionary marker ("}") 765de2fb9cfSCaroline Tice std::string line (buffer); 766de2fb9cfSCaroline Tice 767c7bece56SGreg Clayton size_t len = line.size(); 768de2fb9cfSCaroline Tice if (line[len-1] == '\n') 769de2fb9cfSCaroline Tice { 770de2fb9cfSCaroline Tice line[len-1] = '\0'; 771de2fb9cfSCaroline Tice line.resize (len-1); 772de2fb9cfSCaroline Tice } 773de2fb9cfSCaroline Tice 774de2fb9cfSCaroline Tice if ((line.size() == 1) && (line[0] == '}')) 775de2fb9cfSCaroline Tice { 776de2fb9cfSCaroline Tice done = true; 777de2fb9cfSCaroline Tice line.clear(); 778de2fb9cfSCaroline Tice } 779de2fb9cfSCaroline Tice 780de2fb9cfSCaroline Tice // Try to find a key-value pair in the current line and add it to the dictionary. 781de2fb9cfSCaroline Tice if (line.size() > 0) 782de2fb9cfSCaroline Tice { 783bc43cab5SGreg Clayton static RegularExpression g_reg_exp ("^[ \t]*([a-zA-Z_][a-zA-Z0-9_]*)[ \t]*=[ \t]*(.*)[ \t]*$"); 784bc43cab5SGreg Clayton RegularExpression::Match regex_match(2); 785bc43cab5SGreg Clayton 786bc43cab5SGreg Clayton bool reg_exp_success = g_reg_exp.Execute (line.c_str(), ®ex_match); 787de2fb9cfSCaroline Tice std::string key; 788de2fb9cfSCaroline Tice std::string value; 789de2fb9cfSCaroline Tice if (reg_exp_success) 790de2fb9cfSCaroline Tice { 791bc43cab5SGreg Clayton regex_match.GetMatchAtIndex (line.c_str(), 1, key); 792bc43cab5SGreg Clayton regex_match.GetMatchAtIndex (line.c_str(), 2, value); 793de2fb9cfSCaroline Tice } 794de2fb9cfSCaroline Tice else 795de2fb9cfSCaroline Tice { 796de2fb9cfSCaroline Tice out_stream->Printf ("Instruction::ReadDictionary: Failure executing regular expression.\n"); 797de2fb9cfSCaroline Tice option_value_sp.reset(); 798de2fb9cfSCaroline Tice return option_value_sp; 799de2fb9cfSCaroline Tice } 800de2fb9cfSCaroline Tice 801de2fb9cfSCaroline Tice ConstString const_key (key.c_str()); 802de2fb9cfSCaroline Tice // Check value to see if it's the start of an array or dictionary. 803de2fb9cfSCaroline Tice 804de2fb9cfSCaroline Tice lldb::OptionValueSP value_sp; 805de2fb9cfSCaroline Tice assert (value.empty() == false); 806de2fb9cfSCaroline Tice assert (key.empty() == false); 807de2fb9cfSCaroline Tice 808de2fb9cfSCaroline Tice if (value[0] == '{') 809de2fb9cfSCaroline Tice { 810de2fb9cfSCaroline Tice assert (value.size() == 1); 811de2fb9cfSCaroline Tice // value is a dictionary 812de2fb9cfSCaroline Tice value_sp = ReadDictionary (in_file, out_stream); 813de2fb9cfSCaroline Tice if (value_sp.get() == NULL) 814de2fb9cfSCaroline Tice { 815de2fb9cfSCaroline Tice option_value_sp.reset (); 816de2fb9cfSCaroline Tice return option_value_sp; 817de2fb9cfSCaroline Tice } 818de2fb9cfSCaroline Tice } 819de2fb9cfSCaroline Tice else if (value[0] == '[') 820de2fb9cfSCaroline Tice { 821de2fb9cfSCaroline Tice assert (value.size() == 1); 822de2fb9cfSCaroline Tice // value is an array 823de2fb9cfSCaroline Tice value_sp = ReadArray (in_file, out_stream, data_type); 824de2fb9cfSCaroline Tice if (value_sp.get() == NULL) 825de2fb9cfSCaroline Tice { 826de2fb9cfSCaroline Tice option_value_sp.reset (); 827de2fb9cfSCaroline Tice return option_value_sp; 828de2fb9cfSCaroline Tice } 829de2fb9cfSCaroline Tice // We've used the data_type to read an array; re-set the type to Invalid 830de2fb9cfSCaroline Tice data_type = OptionValue::eTypeInvalid; 831de2fb9cfSCaroline Tice } 832de2fb9cfSCaroline Tice else if ((value[0] == '0') && (value[1] == 'x')) 833de2fb9cfSCaroline Tice { 834de2fb9cfSCaroline Tice value_sp.reset (new OptionValueUInt64 (0, 0)); 835c95f7e2aSPavel Labath value_sp->SetValueFromString (value); 836de2fb9cfSCaroline Tice } 837de2fb9cfSCaroline Tice else 838de2fb9cfSCaroline Tice { 839c7bece56SGreg Clayton size_t len = value.size(); 840de2fb9cfSCaroline Tice if ((value[0] == '"') && (value[len-1] == '"')) 841de2fb9cfSCaroline Tice value = value.substr (1, len-2); 842de2fb9cfSCaroline Tice value_sp.reset (new OptionValueString (value.c_str(), "")); 843de2fb9cfSCaroline Tice } 844de2fb9cfSCaroline Tice 845de2fb9cfSCaroline Tice 846de2fb9cfSCaroline Tice 847de2fb9cfSCaroline Tice if (const_key == encoding_key) 848de2fb9cfSCaroline Tice { 849de2fb9cfSCaroline Tice // A 'data_encoding=..." is NOT a normal key-value pair; it is meta-data indicating the 850de2fb9cfSCaroline Tice // data type of an upcoming array (usually the next bit of data to be read in). 851de2fb9cfSCaroline Tice if (strcmp (value.c_str(), "uint32_t") == 0) 852de2fb9cfSCaroline Tice data_type = OptionValue::eTypeUInt64; 853de2fb9cfSCaroline Tice } 854de2fb9cfSCaroline Tice else 85584c39663SGreg Clayton option_value_sp->GetAsDictionary()->SetValueForKey (const_key, value_sp, false); 856de2fb9cfSCaroline Tice } 857de2fb9cfSCaroline Tice } 858de2fb9cfSCaroline Tice 859de2fb9cfSCaroline Tice return option_value_sp; 860de2fb9cfSCaroline Tice } 861de2fb9cfSCaroline Tice 8627c9dd3ceSCaroline Tice bool 8633ac6711aSCaroline Tice Instruction::TestEmulation (Stream *out_stream, const char *file_name) 8643ac6711aSCaroline Tice { 8653ac6711aSCaroline Tice if (!out_stream) 8663ac6711aSCaroline Tice return false; 8673ac6711aSCaroline Tice 8683ac6711aSCaroline Tice if (!file_name) 8693ac6711aSCaroline Tice { 870ea80ba8bSJohnny Chen out_stream->Printf ("Instruction::TestEmulation: Missing file_name."); 8713ac6711aSCaroline Tice return false; 8723ac6711aSCaroline Tice } 8733ac6711aSCaroline Tice 8743ac6711aSCaroline Tice FILE *test_file = fopen (file_name, "r"); 8753ac6711aSCaroline Tice if (!test_file) 8763ac6711aSCaroline Tice { 877ea80ba8bSJohnny Chen out_stream->Printf ("Instruction::TestEmulation: Attempt to open test file failed."); 8783ac6711aSCaroline Tice return false; 8793ac6711aSCaroline Tice } 8803ac6711aSCaroline Tice 8813ac6711aSCaroline Tice char buffer[256]; 882de2fb9cfSCaroline Tice if (!fgets (buffer, 255, test_file)) 8833ac6711aSCaroline Tice { 884de2fb9cfSCaroline Tice out_stream->Printf ("Instruction::TestEmulation: Error reading first line of test file.\n"); 8853ac6711aSCaroline Tice fclose (test_file); 8863ac6711aSCaroline Tice return false; 8873ac6711aSCaroline Tice } 8883ac6711aSCaroline Tice 889de2fb9cfSCaroline Tice if (strncmp (buffer, "InstructionEmulationState={", 27) != 0) 890de2fb9cfSCaroline Tice { 891de2fb9cfSCaroline Tice out_stream->Printf ("Instructin::TestEmulation: Test file does not contain emulation state dictionary\n"); 892de2fb9cfSCaroline Tice fclose (test_file); 893de2fb9cfSCaroline Tice return false; 894de2fb9cfSCaroline Tice } 895de2fb9cfSCaroline Tice 896de2fb9cfSCaroline Tice // Read all the test information from the test file into an OptionValueDictionary. 897de2fb9cfSCaroline Tice 898de2fb9cfSCaroline Tice OptionValueSP data_dictionary_sp (ReadDictionary (test_file, out_stream)); 899de2fb9cfSCaroline Tice if (data_dictionary_sp.get() == NULL) 900de2fb9cfSCaroline Tice { 901de2fb9cfSCaroline Tice out_stream->Printf ("Instruction::TestEmulation: Error reading Dictionary Object.\n"); 902de2fb9cfSCaroline Tice fclose (test_file); 903de2fb9cfSCaroline Tice return false; 904de2fb9cfSCaroline Tice } 905de2fb9cfSCaroline Tice 906de2fb9cfSCaroline Tice fclose (test_file); 907de2fb9cfSCaroline Tice 90884c39663SGreg Clayton OptionValueDictionary *data_dictionary = data_dictionary_sp->GetAsDictionary(); 909de2fb9cfSCaroline Tice static ConstString description_key ("assembly_string"); 910de2fb9cfSCaroline Tice static ConstString triple_key ("triple"); 911de2fb9cfSCaroline Tice 912de2fb9cfSCaroline Tice OptionValueSP value_sp = data_dictionary->GetValueForKey (description_key); 913de2fb9cfSCaroline Tice 914de2fb9cfSCaroline Tice if (value_sp.get() == NULL) 915de2fb9cfSCaroline Tice { 916de2fb9cfSCaroline Tice out_stream->Printf ("Instruction::TestEmulation: Test file does not contain description string.\n"); 917de2fb9cfSCaroline Tice return false; 918de2fb9cfSCaroline Tice } 919de2fb9cfSCaroline Tice 920de2fb9cfSCaroline Tice SetDescription (value_sp->GetStringValue()); 921de2fb9cfSCaroline Tice 922de2fb9cfSCaroline Tice 923de2fb9cfSCaroline Tice value_sp = data_dictionary->GetValueForKey (triple_key); 924de2fb9cfSCaroline Tice if (value_sp.get() == NULL) 925de2fb9cfSCaroline Tice { 926de2fb9cfSCaroline Tice out_stream->Printf ("Instruction::TestEmulation: Test file does not contain triple.\n"); 927de2fb9cfSCaroline Tice return false; 928de2fb9cfSCaroline Tice } 929de2fb9cfSCaroline Tice 930de2fb9cfSCaroline Tice ArchSpec arch; 931de2fb9cfSCaroline Tice arch.SetTriple (llvm::Triple (value_sp->GetStringValue())); 9323ac6711aSCaroline Tice 9333ac6711aSCaroline Tice bool success = false; 9347b0992d9SGreg Clayton std::unique_ptr<EmulateInstruction> insn_emulator_ap (EmulateInstruction::FindPlugin (arch, eInstructionTypeAny, NULL)); 9353ac6711aSCaroline Tice if (insn_emulator_ap.get()) 936de2fb9cfSCaroline Tice success = insn_emulator_ap->TestEmulation (out_stream, arch, data_dictionary); 9373ac6711aSCaroline Tice 9383ac6711aSCaroline Tice if (success) 939ea80ba8bSJohnny Chen out_stream->Printf ("Emulation test succeeded."); 9403ac6711aSCaroline Tice else 941ea80ba8bSJohnny Chen out_stream->Printf ("Emulation test failed."); 9423ac6711aSCaroline Tice 9433ac6711aSCaroline Tice return success; 9443ac6711aSCaroline Tice } 9453ac6711aSCaroline Tice 9463ac6711aSCaroline Tice bool 9477c9dd3ceSCaroline Tice Instruction::Emulate (const ArchSpec &arch, 9482ed751bdSGreg Clayton uint32_t evaluate_options, 9497c9dd3ceSCaroline Tice void *baton, 9507349bd90SGreg Clayton EmulateInstruction::ReadMemoryCallback read_mem_callback, 9517349bd90SGreg Clayton EmulateInstruction::WriteMemoryCallback write_mem_callback, 9527349bd90SGreg Clayton EmulateInstruction::ReadRegisterCallback read_reg_callback, 9537349bd90SGreg Clayton EmulateInstruction::WriteRegisterCallback write_reg_callback) 9547c9dd3ceSCaroline Tice { 9557b0992d9SGreg Clayton std::unique_ptr<EmulateInstruction> insn_emulator_ap (EmulateInstruction::FindPlugin (arch, eInstructionTypeAny, NULL)); 9567c9dd3ceSCaroline Tice if (insn_emulator_ap.get()) 9577c9dd3ceSCaroline Tice { 9587c9dd3ceSCaroline Tice insn_emulator_ap->SetBaton (baton); 9597c9dd3ceSCaroline Tice insn_emulator_ap->SetCallbacks (read_mem_callback, write_mem_callback, read_reg_callback, write_reg_callback); 9602ed751bdSGreg Clayton insn_emulator_ap->SetInstruction (GetOpcode(), GetAddress(), NULL); 9612ed751bdSGreg Clayton return insn_emulator_ap->EvaluateInstruction (evaluate_options); 9627c9dd3ceSCaroline Tice } 9637c9dd3ceSCaroline Tice 9647c9dd3ceSCaroline Tice return false; 9657c9dd3ceSCaroline Tice } 9667c9dd3ceSCaroline Tice 967ba812f42SGreg Clayton 968ba812f42SGreg Clayton uint32_t 969ba812f42SGreg Clayton Instruction::GetData (DataExtractor &data) 970ba812f42SGreg Clayton { 971cd4ae1abSSean Callanan return m_opcode.GetData(data); 972ba812f42SGreg Clayton } 973ba812f42SGreg Clayton 9741d273166SGreg Clayton InstructionList::InstructionList() : 97530fdc8d8SChris Lattner m_instructions() 97630fdc8d8SChris Lattner { 97730fdc8d8SChris Lattner } 97830fdc8d8SChris Lattner 9791d273166SGreg Clayton InstructionList::~InstructionList() 98030fdc8d8SChris Lattner { 98130fdc8d8SChris Lattner } 98230fdc8d8SChris Lattner 98330fdc8d8SChris Lattner size_t 9841d273166SGreg Clayton InstructionList::GetSize() const 98530fdc8d8SChris Lattner { 98630fdc8d8SChris Lattner return m_instructions.size(); 98730fdc8d8SChris Lattner } 98830fdc8d8SChris Lattner 989357132ebSGreg Clayton uint32_t 990357132ebSGreg Clayton InstructionList::GetMaxOpcocdeByteSize () const 991357132ebSGreg Clayton { 992357132ebSGreg Clayton uint32_t max_inst_size = 0; 993357132ebSGreg Clayton collection::const_iterator pos, end; 994357132ebSGreg Clayton for (pos = m_instructions.begin(), end = m_instructions.end(); 995357132ebSGreg Clayton pos != end; 996357132ebSGreg Clayton ++pos) 997357132ebSGreg Clayton { 998357132ebSGreg Clayton uint32_t inst_size = (*pos)->GetOpcode().GetByteSize(); 999357132ebSGreg Clayton if (max_inst_size < inst_size) 1000357132ebSGreg Clayton max_inst_size = inst_size; 1001357132ebSGreg Clayton } 1002357132ebSGreg Clayton return max_inst_size; 1003357132ebSGreg Clayton } 1004357132ebSGreg Clayton 1005357132ebSGreg Clayton 100630fdc8d8SChris Lattner 10071d273166SGreg Clayton InstructionSP 1008c7bece56SGreg Clayton InstructionList::GetInstructionAtIndex (size_t idx) const 100930fdc8d8SChris Lattner { 10101d273166SGreg Clayton InstructionSP inst_sp; 101130fdc8d8SChris Lattner if (idx < m_instructions.size()) 10121d273166SGreg Clayton inst_sp = m_instructions[idx]; 10131d273166SGreg Clayton return inst_sp; 101430fdc8d8SChris Lattner } 101530fdc8d8SChris Lattner 101630fdc8d8SChris Lattner void 10175009f9d5SGreg Clayton InstructionList::Dump (Stream *s, 10185009f9d5SGreg Clayton bool show_address, 10195009f9d5SGreg Clayton bool show_bytes, 10205009f9d5SGreg Clayton const ExecutionContext* exe_ctx) 10215009f9d5SGreg Clayton { 10225009f9d5SGreg Clayton const uint32_t max_opcode_byte_size = GetMaxOpcocdeByteSize(); 10235009f9d5SGreg Clayton collection::const_iterator pos, begin, end; 1024554f68d3SGreg Clayton 1025554f68d3SGreg Clayton const FormatEntity::Entry *disassembly_format = NULL; 1026554f68d3SGreg Clayton FormatEntity::Entry format; 1027554f68d3SGreg Clayton if (exe_ctx && exe_ctx->HasTargetScope()) 1028aff1b357SJason Molenda { 1029554f68d3SGreg Clayton disassembly_format = exe_ctx->GetTargetRef().GetDebugger().GetDisassemblyFormat (); 1030aff1b357SJason Molenda } 1031554f68d3SGreg Clayton else 1032554f68d3SGreg Clayton { 1033554f68d3SGreg Clayton FormatEntity::Parse("${addr}: ", format); 1034554f68d3SGreg Clayton disassembly_format = &format; 1035554f68d3SGreg Clayton } 1036554f68d3SGreg Clayton 10375009f9d5SGreg Clayton for (begin = m_instructions.begin(), end = m_instructions.end(), pos = begin; 10385009f9d5SGreg Clayton pos != end; 10395009f9d5SGreg Clayton ++pos) 10405009f9d5SGreg Clayton { 10415009f9d5SGreg Clayton if (pos != begin) 10425009f9d5SGreg Clayton s->EOL(); 1043c980fa92SJason Molenda (*pos)->Dump(s, max_opcode_byte_size, show_address, show_bytes, exe_ctx, NULL, NULL, disassembly_format, 0); 10445009f9d5SGreg Clayton } 10455009f9d5SGreg Clayton } 10465009f9d5SGreg Clayton 10475009f9d5SGreg Clayton 10485009f9d5SGreg Clayton void 10491d273166SGreg Clayton InstructionList::Clear() 105030fdc8d8SChris Lattner { 105130fdc8d8SChris Lattner m_instructions.clear(); 105230fdc8d8SChris Lattner } 105330fdc8d8SChris Lattner 105430fdc8d8SChris Lattner void 10551d273166SGreg Clayton InstructionList::Append (lldb::InstructionSP &inst_sp) 105630fdc8d8SChris Lattner { 105730fdc8d8SChris Lattner if (inst_sp) 105830fdc8d8SChris Lattner m_instructions.push_back(inst_sp); 105930fdc8d8SChris Lattner } 106030fdc8d8SChris Lattner 1061564d8bc2SJim Ingham uint32_t 1062e76e7e93STed Woodward InstructionList::GetIndexOfNextBranchInstruction(uint32_t start, Target &target) const 1063564d8bc2SJim Ingham { 1064564d8bc2SJim Ingham size_t num_instructions = m_instructions.size(); 1065564d8bc2SJim Ingham 1066564d8bc2SJim Ingham uint32_t next_branch = UINT32_MAX; 1067e76e7e93STed Woodward size_t i; 1068e76e7e93STed Woodward for (i = start; i < num_instructions; i++) 1069564d8bc2SJim Ingham { 1070564d8bc2SJim Ingham if (m_instructions[i]->DoesBranch()) 1071564d8bc2SJim Ingham { 1072564d8bc2SJim Ingham next_branch = i; 1073564d8bc2SJim Ingham break; 1074564d8bc2SJim Ingham } 1075564d8bc2SJim Ingham } 1076e76e7e93STed Woodward 1077e76e7e93STed Woodward // Hexagon needs the first instruction of the packet with the branch. 1078e76e7e93STed Woodward // Go backwards until we find an instruction marked end-of-packet, or 1079e76e7e93STed Woodward // until we hit start. 1080e76e7e93STed Woodward if (target.GetArchitecture().GetTriple().getArch() == llvm::Triple::hexagon) 1081e76e7e93STed Woodward { 1082e76e7e93STed Woodward // If we didn't find a branch, find the last packet start. 1083e76e7e93STed Woodward if (next_branch == UINT32_MAX) 1084e76e7e93STed Woodward { 1085e76e7e93STed Woodward i = num_instructions - 1; 1086e76e7e93STed Woodward } 1087e76e7e93STed Woodward 1088e76e7e93STed Woodward while (i > start) 1089e76e7e93STed Woodward { 1090e76e7e93STed Woodward --i; 1091e76e7e93STed Woodward 1092e76e7e93STed Woodward Error error; 1093e76e7e93STed Woodward uint32_t inst_bytes; 1094e76e7e93STed Woodward bool prefer_file_cache = false; // Read from process if process is running 1095e76e7e93STed Woodward lldb::addr_t load_addr = LLDB_INVALID_ADDRESS; 1096e76e7e93STed Woodward target.ReadMemory(m_instructions[i]->GetAddress(), 1097e76e7e93STed Woodward prefer_file_cache, 1098e76e7e93STed Woodward &inst_bytes, 1099e76e7e93STed Woodward sizeof(inst_bytes), 1100e76e7e93STed Woodward error, 1101e76e7e93STed Woodward &load_addr); 1102e76e7e93STed Woodward // If we have an error reading memory, return start 1103e76e7e93STed Woodward if (!error.Success()) 1104e76e7e93STed Woodward return start; 1105e76e7e93STed Woodward // check if this is the last instruction in a packet 1106e76e7e93STed Woodward // bits 15:14 will be 11b or 00b for a duplex 1107e76e7e93STed Woodward if (((inst_bytes & 0xC000) == 0xC000) || 1108e76e7e93STed Woodward ((inst_bytes & 0xC000) == 0x0000)) 1109e76e7e93STed Woodward { 1110e76e7e93STed Woodward // instruction after this should be the start of next packet 1111e76e7e93STed Woodward next_branch = i + 1; 1112e76e7e93STed Woodward break; 1113e76e7e93STed Woodward } 1114e76e7e93STed Woodward } 1115e76e7e93STed Woodward 1116e76e7e93STed Woodward if (next_branch == UINT32_MAX) 1117e76e7e93STed Woodward { 1118e76e7e93STed Woodward // We couldn't find the previous packet, so return start 1119e76e7e93STed Woodward next_branch = start; 1120e76e7e93STed Woodward } 1121e76e7e93STed Woodward } 1122564d8bc2SJim Ingham return next_branch; 1123564d8bc2SJim Ingham } 1124564d8bc2SJim Ingham 1125564d8bc2SJim Ingham uint32_t 112644d93782SGreg Clayton InstructionList::GetIndexOfInstructionAtAddress (const Address &address) 1127564d8bc2SJim Ingham { 1128c7bece56SGreg Clayton size_t num_instructions = m_instructions.size(); 1129564d8bc2SJim Ingham uint32_t index = UINT32_MAX; 1130c7bece56SGreg Clayton for (size_t i = 0; i < num_instructions; i++) 1131564d8bc2SJim Ingham { 1132564d8bc2SJim Ingham if (m_instructions[i]->GetAddress() == address) 1133564d8bc2SJim Ingham { 1134564d8bc2SJim Ingham index = i; 1135564d8bc2SJim Ingham break; 1136564d8bc2SJim Ingham } 1137564d8bc2SJim Ingham } 1138564d8bc2SJim Ingham return index; 1139564d8bc2SJim Ingham } 114030fdc8d8SChris Lattner 114144d93782SGreg Clayton 114244d93782SGreg Clayton uint32_t 114344d93782SGreg Clayton InstructionList::GetIndexOfInstructionAtLoadAddress (lldb::addr_t load_addr, Target &target) 114444d93782SGreg Clayton { 114544d93782SGreg Clayton Address address; 114644d93782SGreg Clayton address.SetLoadAddress(load_addr, &target); 114744d93782SGreg Clayton return GetIndexOfInstructionAtAddress(address); 114844d93782SGreg Clayton } 114944d93782SGreg Clayton 115030fdc8d8SChris Lattner size_t 11513faf47c4SGreg Clayton Disassembler::ParseInstructions (const ExecutionContext *exe_ctx, 115257f0630cSGreg Clayton const AddressRange &range, 11533faf47c4SGreg Clayton Stream *error_strm_ptr, 11543faf47c4SGreg Clayton bool prefer_file_cache) 115530fdc8d8SChris Lattner { 1156c14ee32dSGreg Clayton if (exe_ctx) 1157c14ee32dSGreg Clayton { 1158c14ee32dSGreg Clayton Target *target = exe_ctx->GetTargetPtr(); 1159dda4f7b5SGreg Clayton const addr_t byte_size = range.GetByteSize(); 1160dda4f7b5SGreg Clayton if (target == NULL || byte_size == 0 || !range.GetBaseAddress().IsValid()) 116130fdc8d8SChris Lattner return 0; 116230fdc8d8SChris Lattner 1163dda4f7b5SGreg Clayton DataBufferHeap *heap_buffer = new DataBufferHeap (byte_size, '\0'); 1164dda4f7b5SGreg Clayton DataBufferSP data_sp(heap_buffer); 116530fdc8d8SChris Lattner 116630fdc8d8SChris Lattner Error error; 11673faf47c4SGreg Clayton lldb::addr_t load_addr = LLDB_INVALID_ADDRESS; 1168357132ebSGreg Clayton const size_t bytes_read = target->ReadMemory (range.GetBaseAddress(), 1169357132ebSGreg Clayton prefer_file_cache, 1170357132ebSGreg Clayton heap_buffer->GetBytes(), 1171357132ebSGreg Clayton heap_buffer->GetByteSize(), 11723faf47c4SGreg Clayton error, 11733faf47c4SGreg Clayton &load_addr); 1174dda4f7b5SGreg Clayton 1175dda4f7b5SGreg Clayton if (bytes_read > 0) 117630fdc8d8SChris Lattner { 1177dda4f7b5SGreg Clayton if (bytes_read != heap_buffer->GetByteSize()) 1178dda4f7b5SGreg Clayton heap_buffer->SetByteSize (bytes_read); 1179357132ebSGreg Clayton DataExtractor data (data_sp, 1180357132ebSGreg Clayton m_arch.GetByteOrder(), 1181357132ebSGreg Clayton m_arch.GetAddressByteSize()); 11823faf47c4SGreg Clayton const bool data_from_file = load_addr == LLDB_INVALID_ADDRESS; 11833faf47c4SGreg Clayton return DecodeInstructions (range.GetBaseAddress(), data, 0, UINT32_MAX, false, data_from_file); 118430fdc8d8SChris Lattner } 118557f0630cSGreg Clayton else if (error_strm_ptr) 118657f0630cSGreg Clayton { 118757f0630cSGreg Clayton const char *error_cstr = error.AsCString(); 118857f0630cSGreg Clayton if (error_cstr) 118957f0630cSGreg Clayton { 119057f0630cSGreg Clayton error_strm_ptr->Printf("error: %s\n", error_cstr); 119157f0630cSGreg Clayton } 119257f0630cSGreg Clayton } 119357f0630cSGreg Clayton } 119457f0630cSGreg Clayton else if (error_strm_ptr) 119557f0630cSGreg Clayton { 119657f0630cSGreg Clayton error_strm_ptr->PutCString("error: invalid execution context\n"); 1197c14ee32dSGreg Clayton } 119830fdc8d8SChris Lattner return 0; 119930fdc8d8SChris Lattner } 120030fdc8d8SChris Lattner 120137023b06SJim Ingham size_t 12023faf47c4SGreg Clayton Disassembler::ParseInstructions (const ExecutionContext *exe_ctx, 120337023b06SJim Ingham const Address &start, 12043faf47c4SGreg Clayton uint32_t num_instructions, 12053faf47c4SGreg Clayton bool prefer_file_cache) 120637023b06SJim Ingham { 1207357132ebSGreg Clayton m_instruction_list.Clear(); 120837023b06SJim Ingham 1209c14ee32dSGreg Clayton if (exe_ctx == NULL || num_instructions == 0 || !start.IsValid()) 121037023b06SJim Ingham return 0; 121137023b06SJim Ingham 1212c14ee32dSGreg Clayton Target *target = exe_ctx->GetTargetPtr(); 1213357132ebSGreg Clayton // Calculate the max buffer size we will need in order to disassemble 1214357132ebSGreg Clayton const addr_t byte_size = num_instructions * m_arch.GetMaximumOpcodeByteSize(); 121537023b06SJim Ingham 1216357132ebSGreg Clayton if (target == NULL || byte_size == 0) 121737023b06SJim Ingham return 0; 121837023b06SJim Ingham 121937023b06SJim Ingham DataBufferHeap *heap_buffer = new DataBufferHeap (byte_size, '\0'); 122037023b06SJim Ingham DataBufferSP data_sp (heap_buffer); 122137023b06SJim Ingham 122237023b06SJim Ingham Error error; 12233faf47c4SGreg Clayton lldb::addr_t load_addr = LLDB_INVALID_ADDRESS; 1224357132ebSGreg Clayton const size_t bytes_read = target->ReadMemory (start, 1225357132ebSGreg Clayton prefer_file_cache, 1226357132ebSGreg Clayton heap_buffer->GetBytes(), 1227357132ebSGreg Clayton byte_size, 12283faf47c4SGreg Clayton error, 12293faf47c4SGreg Clayton &load_addr); 12303faf47c4SGreg Clayton 12313faf47c4SGreg Clayton const bool data_from_file = load_addr == LLDB_INVALID_ADDRESS; 123237023b06SJim Ingham 123337023b06SJim Ingham if (bytes_read == 0) 1234357132ebSGreg Clayton return 0; 1235357132ebSGreg Clayton DataExtractor data (data_sp, 1236357132ebSGreg Clayton m_arch.GetByteOrder(), 1237357132ebSGreg Clayton m_arch.GetAddressByteSize()); 123837023b06SJim Ingham 1239357132ebSGreg Clayton const bool append_instructions = true; 1240357132ebSGreg Clayton DecodeInstructions (start, 1241357132ebSGreg Clayton data, 1242357132ebSGreg Clayton 0, 1243357132ebSGreg Clayton num_instructions, 12443faf47c4SGreg Clayton append_instructions, 12453faf47c4SGreg Clayton data_from_file); 124637023b06SJim Ingham 124737023b06SJim Ingham return m_instruction_list.GetSize(); 124837023b06SJim Ingham } 124937023b06SJim Ingham 125030fdc8d8SChris Lattner //---------------------------------------------------------------------- 125130fdc8d8SChris Lattner // Disassembler copy constructor 125230fdc8d8SChris Lattner //---------------------------------------------------------------------- 12530f063ba6SJim Ingham Disassembler::Disassembler(const ArchSpec& arch, const char *flavor) : 125430fdc8d8SChris Lattner m_arch (arch), 125530fdc8d8SChris Lattner m_instruction_list(), 12560f063ba6SJim Ingham m_base_addr(LLDB_INVALID_ADDRESS), 12570f063ba6SJim Ingham m_flavor () 125830fdc8d8SChris Lattner { 12590f063ba6SJim Ingham if (flavor == NULL) 12600f063ba6SJim Ingham m_flavor.assign("default"); 12610f063ba6SJim Ingham else 12620f063ba6SJim Ingham m_flavor.assign(flavor); 126375452e8cSJason Molenda 126475452e8cSJason Molenda // If this is an arm variant that can only include thumb (T16, T32) 126575452e8cSJason Molenda // instructions, force the arch triple to be "thumbv.." instead of 126675452e8cSJason Molenda // "armv..." 126775452e8cSJason Molenda if (arch.GetTriple().getArch() == llvm::Triple::arm 126875452e8cSJason Molenda && (arch.GetCore() == ArchSpec::Core::eCore_arm_armv7m 126975452e8cSJason Molenda || arch.GetCore() == ArchSpec::Core::eCore_arm_armv7em 127075452e8cSJason Molenda || arch.GetCore() == ArchSpec::Core::eCore_arm_armv6m)) 127175452e8cSJason Molenda { 127275452e8cSJason Molenda std::string thumb_arch_name (arch.GetTriple().getArchName().str()); 127375452e8cSJason Molenda // Replace "arm" with "thumb" so we get all thumb variants correct 127475452e8cSJason Molenda if (thumb_arch_name.size() > 3) 127575452e8cSJason Molenda { 127675452e8cSJason Molenda thumb_arch_name.erase(0, 3); 127775452e8cSJason Molenda thumb_arch_name.insert(0, "thumb"); 127875452e8cSJason Molenda } 127975452e8cSJason Molenda m_arch.SetTriple (thumb_arch_name.c_str()); 128075452e8cSJason Molenda } 128130fdc8d8SChris Lattner } 128230fdc8d8SChris Lattner 128330fdc8d8SChris Lattner //---------------------------------------------------------------------- 128430fdc8d8SChris Lattner // Destructor 128530fdc8d8SChris Lattner //---------------------------------------------------------------------- 128630fdc8d8SChris Lattner Disassembler::~Disassembler() 128730fdc8d8SChris Lattner { 128830fdc8d8SChris Lattner } 128930fdc8d8SChris Lattner 12901d273166SGreg Clayton InstructionList & 129130fdc8d8SChris Lattner Disassembler::GetInstructionList () 129230fdc8d8SChris Lattner { 129330fdc8d8SChris Lattner return m_instruction_list; 129430fdc8d8SChris Lattner } 129530fdc8d8SChris Lattner 12961d273166SGreg Clayton const InstructionList & 129730fdc8d8SChris Lattner Disassembler::GetInstructionList () const 129830fdc8d8SChris Lattner { 129930fdc8d8SChris Lattner return m_instruction_list; 130030fdc8d8SChris Lattner } 13013ac6711aSCaroline Tice 13023ac6711aSCaroline Tice //---------------------------------------------------------------------- 13033ac6711aSCaroline Tice // Class PseudoInstruction 13043ac6711aSCaroline Tice //---------------------------------------------------------------------- 13053ac6711aSCaroline Tice PseudoInstruction::PseudoInstruction () : 13063ac6711aSCaroline Tice Instruction (Address(), eAddressClassUnknown), 13073ac6711aSCaroline Tice m_description () 13083ac6711aSCaroline Tice { 13093ac6711aSCaroline Tice } 13103ac6711aSCaroline Tice 13113ac6711aSCaroline Tice PseudoInstruction::~PseudoInstruction () 13123ac6711aSCaroline Tice { 13133ac6711aSCaroline Tice } 13143ac6711aSCaroline Tice 13153ac6711aSCaroline Tice bool 131632ce20c5SJim Ingham PseudoInstruction::DoesBranch () 13173ac6711aSCaroline Tice { 13183ac6711aSCaroline Tice // This is NOT a valid question for a pseudo instruction. 13193ac6711aSCaroline Tice return false; 13203ac6711aSCaroline Tice } 13213ac6711aSCaroline Tice 13223ac6711aSCaroline Tice size_t 13233ac6711aSCaroline Tice PseudoInstruction::Decode (const lldb_private::Disassembler &disassembler, 13243ac6711aSCaroline Tice const lldb_private::DataExtractor &data, 1325c7bece56SGreg Clayton lldb::offset_t data_offset) 13263ac6711aSCaroline Tice { 13273ac6711aSCaroline Tice return m_opcode.GetByteSize(); 13283ac6711aSCaroline Tice } 13293ac6711aSCaroline Tice 13303ac6711aSCaroline Tice 13313ac6711aSCaroline Tice void 13323ac6711aSCaroline Tice PseudoInstruction::SetOpcode (size_t opcode_size, void *opcode_data) 13333ac6711aSCaroline Tice { 13343ac6711aSCaroline Tice if (!opcode_data) 13353ac6711aSCaroline Tice return; 13363ac6711aSCaroline Tice 13373ac6711aSCaroline Tice switch (opcode_size) 13383ac6711aSCaroline Tice { 13393ac6711aSCaroline Tice case 8: 13403ac6711aSCaroline Tice { 13413ac6711aSCaroline Tice uint8_t value8 = *((uint8_t *) opcode_data); 134290359963SEd Maste m_opcode.SetOpcode8 (value8, eByteOrderInvalid); 13433ac6711aSCaroline Tice break; 13443ac6711aSCaroline Tice } 13453ac6711aSCaroline Tice case 16: 13463ac6711aSCaroline Tice { 13473ac6711aSCaroline Tice uint16_t value16 = *((uint16_t *) opcode_data); 134890359963SEd Maste m_opcode.SetOpcode16 (value16, eByteOrderInvalid); 13493ac6711aSCaroline Tice break; 13503ac6711aSCaroline Tice } 13513ac6711aSCaroline Tice case 32: 13523ac6711aSCaroline Tice { 13533ac6711aSCaroline Tice uint32_t value32 = *((uint32_t *) opcode_data); 135490359963SEd Maste m_opcode.SetOpcode32 (value32, eByteOrderInvalid); 13553ac6711aSCaroline Tice break; 13563ac6711aSCaroline Tice } 13573ac6711aSCaroline Tice case 64: 13583ac6711aSCaroline Tice { 13593ac6711aSCaroline Tice uint64_t value64 = *((uint64_t *) opcode_data); 136090359963SEd Maste m_opcode.SetOpcode64 (value64, eByteOrderInvalid); 13613ac6711aSCaroline Tice break; 13623ac6711aSCaroline Tice } 13633ac6711aSCaroline Tice default: 13643ac6711aSCaroline Tice break; 13653ac6711aSCaroline Tice } 13663ac6711aSCaroline Tice } 13673ac6711aSCaroline Tice 13683ac6711aSCaroline Tice void 13693ac6711aSCaroline Tice PseudoInstruction::SetDescription (const char *description) 13703ac6711aSCaroline Tice { 13713ac6711aSCaroline Tice if (description && strlen (description) > 0) 13723ac6711aSCaroline Tice m_description = description; 13733ac6711aSCaroline Tice } 1374