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(), &regex_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(), &regex_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