130fdc8d8SChris Lattner //===-- CommandObjectDisassemble.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 "CommandObjectDisassemble.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/Core/AddressRange.h"
191f746071SGreg Clayton #include "lldb/Core/Disassembler.h"
20801237a2SJason Molenda #include "lldb/Core/Module.h"
211f746071SGreg Clayton #include "lldb/Core/SourceManager.h"
2240af72e1SJim Ingham #include "lldb/Interpreter/Args.h"
2330fdc8d8SChris Lattner #include "lldb/Interpreter/CommandCompletions.h"
2430fdc8d8SChris Lattner #include "lldb/Interpreter/CommandInterpreter.h"
2530fdc8d8SChris Lattner #include "lldb/Interpreter/CommandReturnObject.h"
2640af72e1SJim Ingham #include "lldb/Interpreter/Options.h"
271f746071SGreg Clayton #include "lldb/Symbol/Function.h"
2830fdc8d8SChris Lattner #include "lldb/Symbol/Symbol.h"
2930fdc8d8SChris Lattner #include "lldb/Target/Process.h"
30d5944cd1SGreg Clayton #include "lldb/Target/SectionLoadList.h"
31b57e4a1bSJason Molenda #include "lldb/Target/StackFrame.h"
3230fdc8d8SChris Lattner #include "lldb/Target/Target.h"
3330fdc8d8SChris Lattner 
3430fdc8d8SChris Lattner #define DEFAULT_DISASM_BYTE_SIZE 32
3537023b06SJim Ingham #define DEFAULT_DISASM_NUM_INS  4
3630fdc8d8SChris Lattner 
3730fdc8d8SChris Lattner using namespace lldb;
3830fdc8d8SChris Lattner using namespace lldb_private;
3930fdc8d8SChris Lattner 
40eb0103f2SGreg Clayton CommandObjectDisassemble::CommandOptions::CommandOptions (CommandInterpreter &interpreter) :
411f1b269bSJohnny Chen     Options(interpreter),
4237023b06SJim Ingham     num_lines_context(0),
4337023b06SJim Ingham     num_instructions (0),
4432e0a750SGreg Clayton     func_name(),
451fb2e7dfSGreg Clayton     current_function (false),
4632e0a750SGreg Clayton     start_addr(),
4732e0a750SGreg Clayton     end_addr (),
4832e0a750SGreg Clayton     at_pc (false),
4932e0a750SGreg Clayton     frame_line (false),
5032e0a750SGreg Clayton     plugin_name (),
510f063ba6SJim Ingham     flavor_string(),
523555b5d7SJim Ingham     arch(),
53801237a2SJason Molenda     some_location_specified (false),
54801237a2SJason Molenda     symbol_containing_addr ()
5530fdc8d8SChris Lattner {
56f6b8b581SGreg Clayton     OptionParsingStarting();
5730fdc8d8SChris Lattner }
5830fdc8d8SChris Lattner 
5930fdc8d8SChris Lattner CommandObjectDisassemble::CommandOptions::~CommandOptions ()
6030fdc8d8SChris Lattner {
6130fdc8d8SChris Lattner }
6230fdc8d8SChris Lattner 
6330fdc8d8SChris Lattner Error
64f6b8b581SGreg Clayton CommandObjectDisassemble::CommandOptions::SetOptionValue (uint32_t option_idx, const char *option_arg)
6530fdc8d8SChris Lattner {
6630fdc8d8SChris Lattner     Error error;
6730fdc8d8SChris Lattner 
683bcdfc0eSGreg Clayton     const int short_option = m_getopt_table[option_idx].val;
6930fdc8d8SChris Lattner 
7037023b06SJim Ingham     bool success;
7137023b06SJim Ingham 
7230fdc8d8SChris Lattner     switch (short_option)
7330fdc8d8SChris Lattner     {
7430fdc8d8SChris Lattner     case 'm':
7530fdc8d8SChris Lattner         show_mixed = true;
7630fdc8d8SChris Lattner         break;
7730fdc8d8SChris Lattner 
78357132ebSGreg Clayton     case 'C':
7937023b06SJim Ingham         num_lines_context = Args::StringToUInt32(option_arg, 0, 0, &success);
8037023b06SJim Ingham         if (!success)
8186edbf41SGreg Clayton             error.SetErrorStringWithFormat ("invalid num context lines string: \"%s\"", option_arg);
8237023b06SJim Ingham         break;
8337023b06SJim Ingham 
8430fdc8d8SChris Lattner     case 'c':
8537023b06SJim Ingham         num_instructions = Args::StringToUInt32(option_arg, 0, 0, &success);
8637023b06SJim Ingham         if (!success)
8786edbf41SGreg Clayton             error.SetErrorStringWithFormat ("invalid num of instructions string: \"%s\"", option_arg);
8830fdc8d8SChris Lattner         break;
8930fdc8d8SChris Lattner 
9030fdc8d8SChris Lattner     case 'b':
9130fdc8d8SChris Lattner         show_bytes = true;
9230fdc8d8SChris Lattner         break;
9330fdc8d8SChris Lattner 
948651121cSJim Ingham     case 's':
95b9d5df58SGreg Clayton         {
96b9d5df58SGreg Clayton             ExecutionContext exe_ctx (m_interpreter.GetExecutionContext());
97b9d5df58SGreg Clayton             start_addr = Args::StringToAddress(&exe_ctx, option_arg, LLDB_INVALID_ADDRESS, &error);
98b9d5df58SGreg Clayton             if (start_addr != LLDB_INVALID_ADDRESS)
993555b5d7SJim Ingham                 some_location_specified = true;
100b9d5df58SGreg Clayton         }
1018651121cSJim Ingham         break;
1028651121cSJim Ingham     case 'e':
103b9d5df58SGreg Clayton         {
104b9d5df58SGreg Clayton             ExecutionContext exe_ctx (m_interpreter.GetExecutionContext());
105b9d5df58SGreg Clayton             end_addr = Args::StringToAddress(&exe_ctx, option_arg, LLDB_INVALID_ADDRESS, &error);
106b9d5df58SGreg Clayton             if (end_addr != LLDB_INVALID_ADDRESS)
1073555b5d7SJim Ingham                 some_location_specified = true;
108b9d5df58SGreg Clayton         }
109b9d5df58SGreg Clayton         break;
11030fdc8d8SChris Lattner     case 'n':
11132e0a750SGreg Clayton         func_name.assign (option_arg);
1123555b5d7SJim Ingham         some_location_specified = true;
11330fdc8d8SChris Lattner         break;
11430fdc8d8SChris Lattner 
11537023b06SJim Ingham     case 'p':
11632e0a750SGreg Clayton         at_pc = true;
1173555b5d7SJim Ingham         some_location_specified = true;
11832e0a750SGreg Clayton         break;
11932e0a750SGreg Clayton 
12032e0a750SGreg Clayton     case 'l':
12132e0a750SGreg Clayton         frame_line = true;
12232e0a750SGreg Clayton         // Disassemble the current source line kind of implies showing mixed
12332e0a750SGreg Clayton         // source code context.
12432e0a750SGreg Clayton         show_mixed = true;
1253555b5d7SJim Ingham         some_location_specified = true;
12637023b06SJim Ingham         break;
12737023b06SJim Ingham 
1281080edbcSGreg Clayton     case 'P':
12932e0a750SGreg Clayton         plugin_name.assign (option_arg);
1301080edbcSGreg Clayton         break;
1311080edbcSGreg Clayton 
1320f063ba6SJim Ingham     case 'F':
1330f063ba6SJim Ingham         {
1340f063ba6SJim Ingham             Target *target = m_interpreter.GetExecutionContext().GetTargetPtr();
1350f063ba6SJim Ingham             if (target->GetArchitecture().GetTriple().getArch() == llvm::Triple::x86
1360f063ba6SJim Ingham                 || target->GetArchitecture().GetTriple().getArch() == llvm::Triple::x86_64)
1370f063ba6SJim Ingham             {
1380f063ba6SJim Ingham                 flavor_string.assign (option_arg);
1390f063ba6SJim Ingham             }
1400f063ba6SJim Ingham             else
1410f063ba6SJim Ingham                 error.SetErrorStringWithFormat("Disassembler flavors are currently only supported for x86 and x86_64 targets.");
1420f063ba6SJim Ingham             break;
1430f063ba6SJim Ingham         }
14430fdc8d8SChris Lattner     case 'r':
14530fdc8d8SChris Lattner         raw = true;
14630fdc8d8SChris Lattner         break;
14730fdc8d8SChris Lattner 
1488ceb8ba2SJohnny Chen     case 'f':
1491fb2e7dfSGreg Clayton         current_function = true;
1503555b5d7SJim Ingham         some_location_specified = true;
1518ceb8ba2SJohnny Chen         break;
1528ceb8ba2SJohnny Chen 
153801237a2SJason Molenda     case 'A':
15470512317SGreg Clayton         if (!arch.SetTriple (option_arg, m_interpreter.GetPlatform (true).get()))
15570512317SGreg Clayton             arch.SetTriple (option_arg);
156357132ebSGreg Clayton         break;
157357132ebSGreg Clayton 
158801237a2SJason Molenda     case 'a':
159801237a2SJason Molenda         {
160801237a2SJason Molenda             ExecutionContext exe_ctx (m_interpreter.GetExecutionContext());
161801237a2SJason Molenda             symbol_containing_addr = Args::StringToAddress(&exe_ctx, option_arg, LLDB_INVALID_ADDRESS, &error);
162801237a2SJason Molenda             if (symbol_containing_addr != LLDB_INVALID_ADDRESS)
163801237a2SJason Molenda             {
164801237a2SJason Molenda                 some_location_specified = true;
165801237a2SJason Molenda             }
166801237a2SJason Molenda         }
167801237a2SJason Molenda         break;
168801237a2SJason Molenda 
16930fdc8d8SChris Lattner     default:
17086edbf41SGreg Clayton         error.SetErrorStringWithFormat("unrecognized short option '%c'", short_option);
17130fdc8d8SChris Lattner         break;
17230fdc8d8SChris Lattner     }
17330fdc8d8SChris Lattner 
17430fdc8d8SChris Lattner     return error;
17530fdc8d8SChris Lattner }
17630fdc8d8SChris Lattner 
17730fdc8d8SChris Lattner void
178f6b8b581SGreg Clayton CommandObjectDisassemble::CommandOptions::OptionParsingStarting ()
17930fdc8d8SChris Lattner {
18030fdc8d8SChris Lattner     show_mixed = false;
18130fdc8d8SChris Lattner     show_bytes = false;
18230fdc8d8SChris Lattner     num_lines_context = 0;
18337023b06SJim Ingham     num_instructions = 0;
18432e0a750SGreg Clayton     func_name.clear();
1851fb2e7dfSGreg Clayton     current_function = false;
18632e0a750SGreg Clayton     at_pc = false;
18732e0a750SGreg Clayton     frame_line = false;
18832e0a750SGreg Clayton     start_addr = LLDB_INVALID_ADDRESS;
18932e0a750SGreg Clayton     end_addr = LLDB_INVALID_ADDRESS;
190801237a2SJason Molenda     symbol_containing_addr = LLDB_INVALID_ADDRESS;
191a68c1a21SSean Callanan     raw = false;
19232e0a750SGreg Clayton     plugin_name.clear();
1930f063ba6SJim Ingham 
1940f063ba6SJim Ingham     Target *target = m_interpreter.GetExecutionContext().GetTargetPtr();
1950f063ba6SJim Ingham 
1960f063ba6SJim Ingham     // This is a hack till we get the ability to specify features based on architecture.  For now GetDisassemblyFlavor
1970f063ba6SJim Ingham     // is really only valid for x86 (and for the llvm assembler plugin, but I'm papering over that since that is the
1980f063ba6SJim Ingham     // only disassembler plugin we have...
1990f063ba6SJim Ingham     if (target)
2000f063ba6SJim Ingham     {
2010f063ba6SJim Ingham         if (target->GetArchitecture().GetTriple().getArch() == llvm::Triple::x86
2020f063ba6SJim Ingham             || target->GetArchitecture().GetTriple().getArch() == llvm::Triple::x86_64)
2030f063ba6SJim Ingham         {
2040f063ba6SJim Ingham             flavor_string.assign(target->GetDisassemblyFlavor());
2050f063ba6SJim Ingham         }
2060f063ba6SJim Ingham         else
2070f063ba6SJim Ingham             flavor_string.assign ("default");
2080f063ba6SJim Ingham 
2090f063ba6SJim Ingham     }
2100f063ba6SJim Ingham     else
2110f063ba6SJim Ingham         flavor_string.assign("default");
2120f063ba6SJim Ingham 
21332e0a750SGreg Clayton     arch.Clear();
2143555b5d7SJim Ingham     some_location_specified = false;
2153555b5d7SJim Ingham }
2163555b5d7SJim Ingham 
2173555b5d7SJim Ingham Error
2183555b5d7SJim Ingham CommandObjectDisassemble::CommandOptions::OptionParsingFinished ()
2193555b5d7SJim Ingham {
2203555b5d7SJim Ingham     if (!some_location_specified)
2211fb2e7dfSGreg Clayton         current_function = true;
2223555b5d7SJim Ingham     return Error();
2233555b5d7SJim Ingham 
22430fdc8d8SChris Lattner }
22530fdc8d8SChris Lattner 
226e0d378b3SGreg Clayton const OptionDefinition*
22730fdc8d8SChris Lattner CommandObjectDisassemble::CommandOptions::GetDefinitions ()
22830fdc8d8SChris Lattner {
22930fdc8d8SChris Lattner     return g_option_table;
23030fdc8d8SChris Lattner }
23130fdc8d8SChris Lattner 
232e0d378b3SGreg Clayton OptionDefinition
23330fdc8d8SChris Lattner CommandObjectDisassemble::CommandOptions::g_option_table[] =
23430fdc8d8SChris Lattner {
235e2607b50SVirgile Bello { LLDB_OPT_SET_ALL, false, "bytes"        , 'b', OptionParser::eNoArgument        , NULL, 0, eArgTypeNone,        "Show opcode bytes when disassembling."},
236e2607b50SVirgile Bello { LLDB_OPT_SET_ALL, false, "context"      , 'C', OptionParser::eRequiredArgument  , NULL, 0, eArgTypeNumLines,    "Number of context lines of source to show."},
237e2607b50SVirgile Bello { LLDB_OPT_SET_ALL, false, "mixed"        , 'm', OptionParser::eNoArgument        , NULL, 0, eArgTypeNone,        "Enable mixed source and assembly display."},
238e2607b50SVirgile Bello { LLDB_OPT_SET_ALL, false, "raw"          , 'r', OptionParser::eNoArgument        , NULL, 0, eArgTypeNone,        "Print raw disassembly with no symbol information."},
239e2607b50SVirgile Bello { LLDB_OPT_SET_ALL, false, "plugin"       , 'P', OptionParser::eRequiredArgument  , NULL, 0, eArgTypePlugin,      "Name of the disassembler plugin you want to use."},
240e2607b50SVirgile Bello { LLDB_OPT_SET_ALL, false, "flavor"       , 'F', OptionParser::eRequiredArgument  , NULL, 0, eArgTypeDisassemblyFlavor,        "Name of the disassembly flavor you want to use.  "
2410f063ba6SJim Ingham                                                                                                     "Currently the only valid options are default, and for Intel"
2420f063ba6SJim Ingham                                                                                                     " architectures, att and intel."},
243e2607b50SVirgile Bello { LLDB_OPT_SET_ALL, false, "arch"         , 'A', OptionParser::eRequiredArgument  , NULL, 0, eArgTypeArchitecture,"Specify the architecture to use from cross disassembly."},
244357132ebSGreg Clayton { LLDB_OPT_SET_1  |
245e2607b50SVirgile Bello   LLDB_OPT_SET_2  , true , "start-address", 's', OptionParser::eRequiredArgument  , NULL, 0, eArgTypeAddressOrExpression,"Address at which to start disassembling."},
246e2607b50SVirgile Bello { LLDB_OPT_SET_1  , false, "end-address"  , 'e', OptionParser::eRequiredArgument  , NULL, 0, eArgTypeAddressOrExpression,  "Address at which to end disassembling."},
247357132ebSGreg Clayton { LLDB_OPT_SET_2  |
248357132ebSGreg Clayton   LLDB_OPT_SET_3  |
249357132ebSGreg Clayton   LLDB_OPT_SET_4  |
250e2607b50SVirgile Bello   LLDB_OPT_SET_5  , false, "count"        , 'c', OptionParser::eRequiredArgument  , NULL, 0, eArgTypeNumLines,    "Number of instructions to display."},
251e2607b50SVirgile Bello { LLDB_OPT_SET_3  , false, "name"         , 'n', OptionParser::eRequiredArgument  , NULL, CommandCompletions::eSymbolCompletion, eArgTypeFunctionName,
2521fb2e7dfSGreg Clayton                                                                                                     "Disassemble entire contents of the given function name."},
253e2607b50SVirgile Bello { LLDB_OPT_SET_4  , false, "frame"        , 'f', OptionParser::eNoArgument        , NULL, 0, eArgTypeNone,        "Disassemble from the start of the current frame's function."},
254e2607b50SVirgile Bello { LLDB_OPT_SET_5  , false, "pc"           , 'p', OptionParser::eNoArgument        , NULL, 0, eArgTypeNone,        "Disassemble around the current pc."},
255bcf81ea4SBen Langmuir { LLDB_OPT_SET_6  , false, "line"         , 'l', OptionParser::eNoArgument        , NULL, 0, eArgTypeNone,        "Disassemble the current frame's current source line instructions if there is debug line table information, else disassemble around the pc."},
256e2607b50SVirgile Bello { LLDB_OPT_SET_7  , false, "address"      , 'a', OptionParser::eRequiredArgument  , NULL, 0, eArgTypeAddressOrExpression, "Disassemble function containing this address."},
257deaab222SCaroline Tice { 0               , false, NULL           ,   0, 0                  , NULL, 0, eArgTypeNone,        NULL }
25830fdc8d8SChris Lattner };
25930fdc8d8SChris Lattner 
26030fdc8d8SChris Lattner 
26130fdc8d8SChris Lattner 
26230fdc8d8SChris Lattner //-------------------------------------------------------------------------
26330fdc8d8SChris Lattner // CommandObjectDisassemble
26430fdc8d8SChris Lattner //-------------------------------------------------------------------------
26530fdc8d8SChris Lattner 
266a7015092SGreg Clayton CommandObjectDisassemble::CommandObjectDisassemble (CommandInterpreter &interpreter) :
2675a988416SJim Ingham     CommandObjectParsed (interpreter,
268a7015092SGreg Clayton                          "disassemble",
2693f4c09c1SCaroline Tice                          "Disassemble bytes in the current function, or elsewhere in the executable program as specified by the user.",
270eb0103f2SGreg Clayton                          "disassemble [<cmd-options>]"),
271eb0103f2SGreg Clayton     m_options (interpreter)
27230fdc8d8SChris Lattner {
27330fdc8d8SChris Lattner }
27430fdc8d8SChris Lattner 
27530fdc8d8SChris Lattner CommandObjectDisassemble::~CommandObjectDisassemble()
27630fdc8d8SChris Lattner {
27730fdc8d8SChris Lattner }
27830fdc8d8SChris Lattner 
27930fdc8d8SChris Lattner bool
2805a988416SJim Ingham CommandObjectDisassemble::DoExecute (Args& command, CommandReturnObject &result)
28130fdc8d8SChris Lattner {
282a7015092SGreg Clayton     Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get();
28330fdc8d8SChris Lattner     if (target == NULL)
28430fdc8d8SChris Lattner     {
285effe5c95SGreg Clayton         result.AppendError ("invalid target, create a debug target using the 'target create' command");
28630fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
28730fdc8d8SChris Lattner         return false;
28830fdc8d8SChris Lattner     }
28932e0a750SGreg Clayton     if (!m_options.arch.IsValid())
29032e0a750SGreg Clayton         m_options.arch = target->GetArchitecture();
29130fdc8d8SChris Lattner 
29232e0a750SGreg Clayton     if (!m_options.arch.IsValid())
29330fdc8d8SChris Lattner     {
294357132ebSGreg Clayton         result.AppendError ("use the --arch option or set the target architecure to disassemble");
29530fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
29630fdc8d8SChris Lattner         return false;
29730fdc8d8SChris Lattner     }
29830fdc8d8SChris Lattner 
2991080edbcSGreg Clayton     const char *plugin_name = m_options.GetPluginName ();
3000f063ba6SJim Ingham     const char *flavor_string = m_options.GetFlavorString();
3010f063ba6SJim Ingham 
3020f063ba6SJim Ingham     DisassemblerSP disassembler = Disassembler::FindPlugin(m_options.arch, flavor_string, plugin_name);
30330fdc8d8SChris Lattner 
3049a028519SSean Callanan     if (!disassembler)
30530fdc8d8SChris Lattner     {
3061080edbcSGreg Clayton         if (plugin_name)
3070f063ba6SJim Ingham         {
308357132ebSGreg Clayton             result.AppendErrorWithFormat ("Unable to find Disassembler plug-in named '%s' that supports the '%s' architecture.\n",
309357132ebSGreg Clayton                                           plugin_name,
31032e0a750SGreg Clayton                                           m_options.arch.GetArchitectureName());
3110f063ba6SJim Ingham         }
3121080edbcSGreg Clayton         else
313357132ebSGreg Clayton             result.AppendErrorWithFormat ("Unable to find Disassembler plug-in for the '%s' architecture.\n",
31432e0a750SGreg Clayton                                           m_options.arch.GetArchitectureName());
31530fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
31630fdc8d8SChris Lattner         return false;
31730fdc8d8SChris Lattner     }
3180f063ba6SJim Ingham     else if (flavor_string != NULL && !disassembler->FlavorValidForArchSpec(m_options.arch, flavor_string))
3190f063ba6SJim Ingham         result.AppendWarningWithFormat("invalid disassembler flavor \"%s\", using default.\n", flavor_string);
32030fdc8d8SChris Lattner 
32130fdc8d8SChris Lattner     result.SetStatus (eReturnStatusSuccessFinishResult);
32230fdc8d8SChris Lattner 
323dda4f7b5SGreg Clayton     if (command.GetArgumentCount() != 0)
32430fdc8d8SChris Lattner     {
325a7015092SGreg Clayton         result.AppendErrorWithFormat ("\"disassemble\" arguments are specified as options.\n");
326eb0103f2SGreg Clayton         GetOptions()->GenerateOptionUsage (result.GetErrorStream(), this);
3278651121cSJim Ingham         result.SetStatus (eReturnStatusFailed);
3288651121cSJim Ingham         return false;
3298651121cSJim Ingham     }
3308651121cSJim Ingham 
331dda4f7b5SGreg Clayton     if (m_options.show_mixed && m_options.num_lines_context == 0)
3326dbd3983SGreg Clayton         m_options.num_lines_context = 1;
333dda4f7b5SGreg Clayton 
334b10d72f0SGreg Clayton     // Always show the PC in the disassembly
335b10d72f0SGreg Clayton     uint32_t options = Disassembler::eOptionMarkPCAddress;
3361da6f9d7SGreg Clayton 
337b10d72f0SGreg Clayton     // Mark the source line for the current PC only if we are doing mixed source and assembly
338b10d72f0SGreg Clayton     if (m_options.show_mixed)
339b10d72f0SGreg Clayton         options |= Disassembler::eOptionMarkPCSourceLine;
3401da6f9d7SGreg Clayton 
3411da6f9d7SGreg Clayton     if (m_options.show_bytes)
3421da6f9d7SGreg Clayton         options |= Disassembler::eOptionShowBytes;
3431da6f9d7SGreg Clayton 
3441da6f9d7SGreg Clayton     if (m_options.raw)
3451da6f9d7SGreg Clayton         options |= Disassembler::eOptionRawOuput;
34637023b06SJim Ingham 
34732e0a750SGreg Clayton     if (!m_options.func_name.empty())
348dda4f7b5SGreg Clayton     {
34932e0a750SGreg Clayton         ConstString name(m_options.func_name.c_str());
350dda4f7b5SGreg Clayton 
351a7015092SGreg Clayton         if (Disassembler::Disassemble (m_interpreter.GetDebugger(),
35232e0a750SGreg Clayton                                        m_options.arch,
3531080edbcSGreg Clayton                                        plugin_name,
3540f063ba6SJim Ingham                                        flavor_string,
355f9fc609fSGreg Clayton                                        m_exe_ctx,
356dda4f7b5SGreg Clayton                                        name,
357dda4f7b5SGreg Clayton                                        NULL,    // Module *
35837023b06SJim Ingham                                        m_options.num_instructions,
359dda4f7b5SGreg Clayton                                        m_options.show_mixed ? m_options.num_lines_context : 0,
3601da6f9d7SGreg Clayton                                        options,
361dda4f7b5SGreg Clayton                                        result.GetOutputStream()))
362dda4f7b5SGreg Clayton         {
363dda4f7b5SGreg Clayton             result.SetStatus (eReturnStatusSuccessFinishResult);
364dda4f7b5SGreg Clayton         }
365dda4f7b5SGreg Clayton         else
366dda4f7b5SGreg Clayton         {
367dda4f7b5SGreg Clayton             result.AppendErrorWithFormat ("Unable to find symbol with name '%s'.\n", name.GetCString());
368dda4f7b5SGreg Clayton             result.SetStatus (eReturnStatusFailed);
369dda4f7b5SGreg Clayton         }
370dda4f7b5SGreg Clayton     }
371dda4f7b5SGreg Clayton     else
372dda4f7b5SGreg Clayton     {
373*2f2c876eSJim Ingham         std::vector<AddressRange> ranges;
37432e0a750SGreg Clayton         AddressRange range;
375b57e4a1bSJason Molenda         StackFrame *frame = m_exe_ctx.GetFramePtr();
37632e0a750SGreg Clayton         if (m_options.frame_line)
37732e0a750SGreg Clayton         {
378c14ee32dSGreg Clayton             if (frame == NULL)
3793555b5d7SJim Ingham             {
3803555b5d7SJim Ingham                 result.AppendError ("Cannot disassemble around the current line without a selected frame.\n");
3813555b5d7SJim Ingham                 result.SetStatus (eReturnStatusFailed);
3823555b5d7SJim Ingham                 return false;
3833555b5d7SJim Ingham             }
384c14ee32dSGreg Clayton             LineEntry pc_line_entry (frame->GetSymbolContext(eSymbolContextLineEntry).line_entry);
38532e0a750SGreg Clayton             if (pc_line_entry.IsValid())
38632e0a750SGreg Clayton             {
38732e0a750SGreg Clayton                 range = pc_line_entry.range;
38832e0a750SGreg Clayton             }
38932e0a750SGreg Clayton             else
39032e0a750SGreg Clayton             {
39132e0a750SGreg Clayton                 m_options.at_pc = true; // No line entry, so just disassemble around the current pc
39232e0a750SGreg Clayton                 m_options.show_mixed = false;
39332e0a750SGreg Clayton             }
39432e0a750SGreg Clayton         }
3951fb2e7dfSGreg Clayton         else if (m_options.current_function)
3963555b5d7SJim Ingham         {
397c14ee32dSGreg Clayton             if (frame == NULL)
3983555b5d7SJim Ingham             {
3993555b5d7SJim Ingham                 result.AppendError ("Cannot disassemble around the current function without a selected frame.\n");
4003555b5d7SJim Ingham                 result.SetStatus (eReturnStatusFailed);
4013555b5d7SJim Ingham                 return false;
4023555b5d7SJim Ingham             }
403c14ee32dSGreg Clayton             Symbol *symbol = frame->GetSymbolContext(eSymbolContextSymbol).symbol;
4043555b5d7SJim Ingham             if (symbol)
405e7612134SGreg Clayton             {
406e7612134SGreg Clayton                 range.GetBaseAddress() = symbol->GetAddress();
407e7612134SGreg Clayton                 range.SetByteSize(symbol->GetByteSize());
408e7612134SGreg Clayton             }
4093555b5d7SJim Ingham         }
41037023b06SJim Ingham 
41132e0a750SGreg Clayton         // Did the "m_options.frame_line" find a valid range already? If so
41232e0a750SGreg Clayton         // skip the rest...
41332e0a750SGreg Clayton         if (range.GetByteSize() == 0)
41432e0a750SGreg Clayton         {
41532e0a750SGreg Clayton             if (m_options.at_pc)
4168651121cSJim Ingham             {
417c14ee32dSGreg Clayton                 if (frame == NULL)
41837023b06SJim Ingham                 {
41937023b06SJim Ingham                     result.AppendError ("Cannot disassemble around the current PC without a selected frame.\n");
42037023b06SJim Ingham                     result.SetStatus (eReturnStatusFailed);
42137023b06SJim Ingham                     return false;
42237023b06SJim Ingham                 }
423c14ee32dSGreg Clayton                 range.GetBaseAddress() = frame->GetFrameCodeAddress();
42437023b06SJim Ingham                 if (m_options.num_instructions == 0)
42537023b06SJim Ingham                 {
42637023b06SJim Ingham                     // Disassembling at the PC always disassembles some number of instructions (not the whole function).
42737023b06SJim Ingham                     m_options.num_instructions = DEFAULT_DISASM_NUM_INS;
42837023b06SJim Ingham                 }
429*2f2c876eSJim Ingham                 ranges.push_back(range);
43037023b06SJim Ingham             }
43137023b06SJim Ingham             else
43237023b06SJim Ingham             {
43332e0a750SGreg Clayton                 range.GetBaseAddress().SetOffset (m_options.start_addr);
43432e0a750SGreg Clayton                 if (range.GetBaseAddress().IsValid())
43537023b06SJim Ingham                 {
43632e0a750SGreg Clayton                     if (m_options.end_addr != LLDB_INVALID_ADDRESS)
4378651121cSJim Ingham                     {
43832e0a750SGreg Clayton                         if (m_options.end_addr <= m_options.start_addr)
4398651121cSJim Ingham                         {
4408651121cSJim Ingham                             result.AppendErrorWithFormat ("End address before start address.\n");
4418651121cSJim Ingham                             result.SetStatus (eReturnStatusFailed);
4428651121cSJim Ingham                             return false;
4438651121cSJim Ingham                         }
44432e0a750SGreg Clayton                         range.SetByteSize (m_options.end_addr - m_options.start_addr);
445*2f2c876eSJim Ingham                         ranges.push_back(range);
44632e0a750SGreg Clayton                     }
44737023b06SJim Ingham                 }
448801237a2SJason Molenda                 else
449801237a2SJason Molenda                 {
450801237a2SJason Molenda                     if (m_options.symbol_containing_addr != LLDB_INVALID_ADDRESS
451*2f2c876eSJim Ingham                         && target)
452*2f2c876eSJim Ingham                     {
453*2f2c876eSJim Ingham                         if (!target->GetSectionLoadList().IsEmpty())
454801237a2SJason Molenda                         {
455801237a2SJason Molenda                             bool failed = false;
456801237a2SJason Molenda                             Address symbol_containing_address;
457801237a2SJason Molenda                             if (target->GetSectionLoadList().ResolveLoadAddress (m_options.symbol_containing_addr, symbol_containing_address))
458801237a2SJason Molenda                             {
459801237a2SJason Molenda                                 ModuleSP module_sp (symbol_containing_address.GetModule());
460801237a2SJason Molenda                                 SymbolContext sc;
46135729bb1SAshok Thirumurthi                                 bool resolve_tail_call_address = true; // PC can be one past the address range of the function.
46235729bb1SAshok Thirumurthi                                 module_sp->ResolveSymbolContextForAddress (symbol_containing_address, eSymbolContextEverything, sc,
46335729bb1SAshok Thirumurthi                                                                            resolve_tail_call_address);
464801237a2SJason Molenda                                 if (sc.function || sc.symbol)
465801237a2SJason Molenda                                 {
466801237a2SJason Molenda                                     sc.GetAddressRange (eSymbolContextFunction | eSymbolContextSymbol, 0, false, range);
467801237a2SJason Molenda                                 }
468801237a2SJason Molenda                                 else
469801237a2SJason Molenda                                 {
470801237a2SJason Molenda                                     failed = true;
471801237a2SJason Molenda                                 }
472801237a2SJason Molenda                             }
473801237a2SJason Molenda                             else
474801237a2SJason Molenda                             {
475801237a2SJason Molenda                                 failed = true;
476801237a2SJason Molenda                             }
477801237a2SJason Molenda                             if (failed)
478801237a2SJason Molenda                             {
479801237a2SJason Molenda                                 result.AppendErrorWithFormat ("Could not find function bounds for address 0x%" PRIx64 "\n", m_options.symbol_containing_addr);
480801237a2SJason Molenda                                 result.SetStatus (eReturnStatusFailed);
481801237a2SJason Molenda                                 return false;
482801237a2SJason Molenda                             }
483*2f2c876eSJim Ingham                             ranges.push_back(range);
484801237a2SJason Molenda                         }
485*2f2c876eSJim Ingham                         else
486*2f2c876eSJim Ingham                         {
487*2f2c876eSJim Ingham                             for (lldb::ModuleSP module_sp : target->GetImages().Modules())
488*2f2c876eSJim Ingham                             {
489*2f2c876eSJim Ingham                                 lldb::addr_t file_addr = m_options.symbol_containing_addr;
490*2f2c876eSJim Ingham                                 Address file_address;
491*2f2c876eSJim Ingham                                 if (module_sp->ResolveFileAddress(file_addr, file_address))
492*2f2c876eSJim Ingham                                 {
493*2f2c876eSJim Ingham                                     SymbolContext sc;
494*2f2c876eSJim Ingham                                     bool resolve_tail_call_address = true; // PC can be one past the address range of the function.
495*2f2c876eSJim Ingham                                     module_sp->ResolveSymbolContextForAddress (file_address, eSymbolContextEverything, sc, resolve_tail_call_address);
496*2f2c876eSJim Ingham                                     if (sc.function || sc.symbol)
497*2f2c876eSJim Ingham                                     {
498*2f2c876eSJim Ingham                                         sc.GetAddressRange (eSymbolContextFunction | eSymbolContextSymbol, 0, false, range);
499*2f2c876eSJim Ingham                                         ranges.push_back(range);
500801237a2SJason Molenda                                     }
50137023b06SJim Ingham                                 }
50237023b06SJim Ingham                             }
50337023b06SJim Ingham 
504*2f2c876eSJim Ingham                         }
505*2f2c876eSJim Ingham                     }
506*2f2c876eSJim Ingham                 }
507*2f2c876eSJim Ingham             }
508*2f2c876eSJim Ingham         }
509*2f2c876eSJim Ingham         else
510*2f2c876eSJim Ingham             ranges.push_back(range);
511*2f2c876eSJim Ingham 
51237023b06SJim Ingham         if (m_options.num_instructions != 0)
51337023b06SJim Ingham         {
514*2f2c876eSJim Ingham             if (ranges.size() == 0)
51537023b06SJim Ingham             {
51637023b06SJim Ingham                 // The default action is to disassemble the current frame function.
517c14ee32dSGreg Clayton                 if (frame)
51837023b06SJim Ingham                 {
519c14ee32dSGreg Clayton                     SymbolContext sc(frame->GetSymbolContext(eSymbolContextFunction | eSymbolContextSymbol));
52037023b06SJim Ingham                     if (sc.function)
52132e0a750SGreg Clayton                         range.GetBaseAddress() = sc.function->GetAddressRange().GetBaseAddress();
522e7612134SGreg Clayton                     else if (sc.symbol && sc.symbol->ValueIsAddress())
523e7612134SGreg Clayton                         range.GetBaseAddress() = sc.symbol->GetAddress();
52437023b06SJim Ingham                     else
525c14ee32dSGreg Clayton                         range.GetBaseAddress() = frame->GetFrameCodeAddress();
52637023b06SJim Ingham                 }
52737023b06SJim Ingham 
52832e0a750SGreg Clayton                 if (!range.GetBaseAddress().IsValid())
52937023b06SJim Ingham                 {
53037023b06SJim Ingham                     result.AppendError ("invalid frame");
53137023b06SJim Ingham                     result.SetStatus (eReturnStatusFailed);
53237023b06SJim Ingham                     return false;
53337023b06SJim Ingham                 }
53437023b06SJim Ingham             }
53537023b06SJim Ingham 
536*2f2c876eSJim Ingham             bool print_sc_header = ranges.size() > 1;
537*2f2c876eSJim Ingham             for (AddressRange cur_range : ranges)
538*2f2c876eSJim Ingham             {
53937023b06SJim Ingham                 if (Disassembler::Disassemble (m_interpreter.GetDebugger(),
54032e0a750SGreg Clayton                                                m_options.arch,
5411080edbcSGreg Clayton                                                plugin_name,
5420f063ba6SJim Ingham                                                flavor_string,
543f9fc609fSGreg Clayton                                                m_exe_ctx,
544*2f2c876eSJim Ingham                                                cur_range.GetBaseAddress(),
54537023b06SJim Ingham                                                m_options.num_instructions,
54637023b06SJim Ingham                                                m_options.show_mixed ? m_options.num_lines_context : 0,
5471da6f9d7SGreg Clayton                                                options,
54837023b06SJim Ingham                                                result.GetOutputStream()))
54937023b06SJim Ingham                 {
55037023b06SJim Ingham                     result.SetStatus (eReturnStatusSuccessFinishResult);
5518651121cSJim Ingham                 }
5528651121cSJim Ingham                 else
55337023b06SJim Ingham                 {
554*2f2c876eSJim Ingham                     if (m_options.start_addr != LLDB_INVALID_ADDRESS)
555d01b2953SDaniel Malea                         result.AppendErrorWithFormat ("Failed to disassemble memory at 0x%8.8" PRIx64 ".\n", m_options.start_addr);
556*2f2c876eSJim Ingham                     else if (m_options.symbol_containing_addr != LLDB_INVALID_ADDRESS)
557*2f2c876eSJim Ingham                         result.AppendErrorWithFormat ("Failed to disassemble memory in function at 0x%8.8" PRIx64 ".\n", m_options.symbol_containing_addr);
55837023b06SJim Ingham                     result.SetStatus (eReturnStatusFailed);
55937023b06SJim Ingham                 }
56037023b06SJim Ingham             }
561*2f2c876eSJim Ingham             if (print_sc_header)
562*2f2c876eSJim Ingham                 result.AppendMessage("\n");
563*2f2c876eSJim Ingham         }
56437023b06SJim Ingham         else
56537023b06SJim Ingham         {
566*2f2c876eSJim Ingham             if (ranges.size() == 0)
56730fdc8d8SChris Lattner             {
5688ceb8ba2SJohnny Chen                 // The default action is to disassemble the current frame function.
569c14ee32dSGreg Clayton                 if (frame)
57030fdc8d8SChris Lattner                 {
571c14ee32dSGreg Clayton                     SymbolContext sc(frame->GetSymbolContext(eSymbolContextFunction | eSymbolContextSymbol));
57230fdc8d8SChris Lattner                     if (sc.function)
573dda4f7b5SGreg Clayton                         range = sc.function->GetAddressRange();
574e7612134SGreg Clayton                     else if (sc.symbol && sc.symbol->ValueIsAddress())
575e7612134SGreg Clayton                     {
576e7612134SGreg Clayton                         range.GetBaseAddress() = sc.symbol->GetAddress();
577e7612134SGreg Clayton                         range.SetByteSize (sc.symbol->GetByteSize());
578e7612134SGreg Clayton                     }
57930fdc8d8SChris Lattner                     else
580c14ee32dSGreg Clayton                         range.GetBaseAddress() = frame->GetFrameCodeAddress();
58130fdc8d8SChris Lattner                 }
58230fdc8d8SChris Lattner                 else
58330fdc8d8SChris Lattner                 {
58430fdc8d8SChris Lattner                     result.AppendError ("invalid frame");
58530fdc8d8SChris Lattner                     result.SetStatus (eReturnStatusFailed);
58630fdc8d8SChris Lattner                     return false;
58730fdc8d8SChris Lattner                 }
588*2f2c876eSJim Ingham                 ranges.push_back(range);
58930fdc8d8SChris Lattner             }
590*2f2c876eSJim Ingham 
591*2f2c876eSJim Ingham             bool print_sc_header = ranges.size() > 1;
592*2f2c876eSJim Ingham             for (AddressRange cur_range : ranges)
593*2f2c876eSJim Ingham             {
594*2f2c876eSJim Ingham                 if (cur_range.GetByteSize() == 0)
595*2f2c876eSJim Ingham                     cur_range.SetByteSize(DEFAULT_DISASM_BYTE_SIZE);
59630fdc8d8SChris Lattner 
597a7015092SGreg Clayton                 if (Disassembler::Disassemble (m_interpreter.GetDebugger(),
59832e0a750SGreg Clayton                                                m_options.arch,
5991080edbcSGreg Clayton                                                plugin_name,
6000f063ba6SJim Ingham                                                flavor_string,
601f9fc609fSGreg Clayton                                                m_exe_ctx,
602*2f2c876eSJim Ingham                                                cur_range,
60337023b06SJim Ingham                                                m_options.num_instructions,
604dda4f7b5SGreg Clayton                                                m_options.show_mixed ? m_options.num_lines_context : 0,
6051da6f9d7SGreg Clayton                                                options,
606dda4f7b5SGreg Clayton                                                result.GetOutputStream()))
60730fdc8d8SChris Lattner                 {
608dda4f7b5SGreg Clayton                     result.SetStatus (eReturnStatusSuccessFinishResult);
60930fdc8d8SChris Lattner                 }
61030fdc8d8SChris Lattner                 else
61130fdc8d8SChris Lattner                 {
612d01b2953SDaniel Malea                     result.AppendErrorWithFormat ("Failed to disassemble memory at 0x%8.8" PRIx64 ".\n", m_options.start_addr);
61330fdc8d8SChris Lattner                     result.SetStatus (eReturnStatusFailed);
61430fdc8d8SChris Lattner                 }
615*2f2c876eSJim Ingham                 if (print_sc_header)
616*2f2c876eSJim Ingham                     result.AppendMessage("\n");
617*2f2c876eSJim Ingham             }
61830fdc8d8SChris Lattner         }
61937023b06SJim Ingham     }
62030fdc8d8SChris Lattner 
62130fdc8d8SChris Lattner     return result.Succeeded();
62230fdc8d8SChris Lattner }
623