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 
1030fdc8d8SChris Lattner #include "CommandObjectDisassemble.h"
1130fdc8d8SChris Lattner 
1230fdc8d8SChris Lattner // C Includes
1330fdc8d8SChris Lattner // C++ Includes
1430fdc8d8SChris Lattner // Other libraries and framework includes
1530fdc8d8SChris Lattner // Project includes
1630fdc8d8SChris Lattner #include "lldb/Core/AddressRange.h"
171f746071SGreg Clayton #include "lldb/Core/Disassembler.h"
181f746071SGreg Clayton #include "lldb/Core/SourceManager.h"
1940af72e1SJim Ingham #include "lldb/Interpreter/Args.h"
2030fdc8d8SChris Lattner #include "lldb/Interpreter/CommandCompletions.h"
2130fdc8d8SChris Lattner #include "lldb/Interpreter/CommandInterpreter.h"
2230fdc8d8SChris Lattner #include "lldb/Interpreter/CommandReturnObject.h"
2340af72e1SJim Ingham #include "lldb/Interpreter/Options.h"
241f746071SGreg Clayton #include "lldb/Symbol/Function.h"
2530fdc8d8SChris Lattner #include "lldb/Symbol/Symbol.h"
2630fdc8d8SChris Lattner #include "lldb/Target/Process.h"
271f746071SGreg Clayton #include "lldb/Target/StackFrame.h"
2830fdc8d8SChris Lattner #include "lldb/Target/Target.h"
2930fdc8d8SChris Lattner 
3030fdc8d8SChris Lattner #define DEFAULT_DISASM_BYTE_SIZE 32
3137023b06SJim Ingham #define DEFAULT_DISASM_NUM_INS  4
3230fdc8d8SChris Lattner 
3330fdc8d8SChris Lattner using namespace lldb;
3430fdc8d8SChris Lattner using namespace lldb_private;
3530fdc8d8SChris Lattner 
36eb0103f2SGreg Clayton CommandObjectDisassemble::CommandOptions::CommandOptions (CommandInterpreter &interpreter) :
371f1b269bSJohnny Chen     Options(interpreter),
3837023b06SJim Ingham     num_lines_context(0),
3937023b06SJim Ingham     num_instructions (0),
4032e0a750SGreg Clayton     func_name(),
413555b5d7SJim Ingham     cur_function (false),
4232e0a750SGreg Clayton     start_addr(),
4332e0a750SGreg Clayton     end_addr (),
4432e0a750SGreg Clayton     at_pc (false),
4532e0a750SGreg Clayton     frame_line (false),
4632e0a750SGreg Clayton     plugin_name (),
473555b5d7SJim Ingham     arch(),
483555b5d7SJim Ingham     some_location_specified (false)
4930fdc8d8SChris Lattner {
50f6b8b581SGreg Clayton     OptionParsingStarting();
5130fdc8d8SChris Lattner }
5230fdc8d8SChris Lattner 
5330fdc8d8SChris Lattner CommandObjectDisassemble::CommandOptions::~CommandOptions ()
5430fdc8d8SChris Lattner {
5530fdc8d8SChris Lattner }
5630fdc8d8SChris Lattner 
5730fdc8d8SChris Lattner Error
58f6b8b581SGreg Clayton CommandObjectDisassemble::CommandOptions::SetOptionValue (uint32_t option_idx, const char *option_arg)
5930fdc8d8SChris Lattner {
6030fdc8d8SChris Lattner     Error error;
6130fdc8d8SChris Lattner 
62*3bcdfc0eSGreg Clayton     const int short_option = m_getopt_table[option_idx].val;
6330fdc8d8SChris Lattner 
6437023b06SJim Ingham     bool success;
6537023b06SJim Ingham 
6630fdc8d8SChris Lattner     switch (short_option)
6730fdc8d8SChris Lattner     {
6830fdc8d8SChris Lattner     case 'm':
6930fdc8d8SChris Lattner         show_mixed = true;
7030fdc8d8SChris Lattner         break;
7130fdc8d8SChris Lattner 
72357132ebSGreg Clayton     case 'C':
7337023b06SJim Ingham         num_lines_context = Args::StringToUInt32(option_arg, 0, 0, &success);
7437023b06SJim Ingham         if (!success)
7586edbf41SGreg Clayton             error.SetErrorStringWithFormat ("invalid num context lines string: \"%s\"", option_arg);
7637023b06SJim Ingham         break;
7737023b06SJim Ingham 
7830fdc8d8SChris Lattner     case 'c':
7937023b06SJim Ingham         num_instructions = Args::StringToUInt32(option_arg, 0, 0, &success);
8037023b06SJim Ingham         if (!success)
8186edbf41SGreg Clayton             error.SetErrorStringWithFormat ("invalid num of instructions string: \"%s\"", option_arg);
8230fdc8d8SChris Lattner         break;
8330fdc8d8SChris Lattner 
8430fdc8d8SChris Lattner     case 'b':
8530fdc8d8SChris Lattner         show_bytes = true;
8630fdc8d8SChris Lattner         break;
8730fdc8d8SChris Lattner 
888651121cSJim Ingham     case 's':
8932e0a750SGreg Clayton         start_addr = Args::StringToUInt64(option_arg, LLDB_INVALID_ADDRESS, 0);
9032e0a750SGreg Clayton         if (start_addr == LLDB_INVALID_ADDRESS)
9132e0a750SGreg Clayton             start_addr = Args::StringToUInt64(option_arg, LLDB_INVALID_ADDRESS, 16);
9230fdc8d8SChris Lattner 
9332e0a750SGreg Clayton         if (start_addr == LLDB_INVALID_ADDRESS)
9486edbf41SGreg Clayton             error.SetErrorStringWithFormat ("invalid start address string '%s'", option_arg);
953555b5d7SJim Ingham         some_location_specified = true;
968651121cSJim Ingham         break;
978651121cSJim Ingham     case 'e':
9832e0a750SGreg Clayton         end_addr = Args::StringToUInt64(option_arg, LLDB_INVALID_ADDRESS, 0);
9932e0a750SGreg Clayton         if (end_addr == LLDB_INVALID_ADDRESS)
10032e0a750SGreg Clayton             end_addr = Args::StringToUInt64(option_arg, LLDB_INVALID_ADDRESS, 16);
1018651121cSJim Ingham 
10232e0a750SGreg Clayton         if (end_addr == LLDB_INVALID_ADDRESS)
10386edbf41SGreg Clayton             error.SetErrorStringWithFormat ("invalid end address string '%s'", option_arg);
10430fdc8d8SChris Lattner         break;
1053555b5d7SJim Ingham         some_location_specified = true;
10630fdc8d8SChris Lattner     case 'n':
10732e0a750SGreg Clayton         func_name.assign (option_arg);
1083555b5d7SJim Ingham         some_location_specified = true;
10930fdc8d8SChris Lattner         break;
11030fdc8d8SChris Lattner 
11137023b06SJim Ingham     case 'p':
11232e0a750SGreg Clayton         at_pc = true;
1133555b5d7SJim Ingham         some_location_specified = true;
11432e0a750SGreg Clayton         break;
11532e0a750SGreg Clayton 
11632e0a750SGreg Clayton     case 'l':
11732e0a750SGreg Clayton         frame_line = true;
11832e0a750SGreg Clayton         // Disassemble the current source line kind of implies showing mixed
11932e0a750SGreg Clayton         // source code context.
12032e0a750SGreg Clayton         show_mixed = true;
1213555b5d7SJim Ingham         some_location_specified = true;
12237023b06SJim Ingham         break;
12337023b06SJim Ingham 
1241080edbcSGreg Clayton     case 'P':
12532e0a750SGreg Clayton         plugin_name.assign (option_arg);
1261080edbcSGreg Clayton         break;
1271080edbcSGreg Clayton 
12830fdc8d8SChris Lattner     case 'r':
12930fdc8d8SChris Lattner         raw = true;
13030fdc8d8SChris Lattner         break;
13130fdc8d8SChris Lattner 
1328ceb8ba2SJohnny Chen     case 'f':
1333555b5d7SJim Ingham         cur_function = true;
1343555b5d7SJim Ingham         some_location_specified = true;
1358ceb8ba2SJohnny Chen         break;
1368ceb8ba2SJohnny Chen 
137357132ebSGreg Clayton     case 'a':
13870512317SGreg Clayton         if (!arch.SetTriple (option_arg, m_interpreter.GetPlatform (true).get()))
13970512317SGreg Clayton             arch.SetTriple (option_arg);
140357132ebSGreg Clayton         break;
141357132ebSGreg Clayton 
14230fdc8d8SChris Lattner     default:
14386edbf41SGreg Clayton         error.SetErrorStringWithFormat("unrecognized short option '%c'", short_option);
14430fdc8d8SChris Lattner         break;
14530fdc8d8SChris Lattner     }
14630fdc8d8SChris Lattner 
14730fdc8d8SChris Lattner     return error;
14830fdc8d8SChris Lattner }
14930fdc8d8SChris Lattner 
15030fdc8d8SChris Lattner void
151f6b8b581SGreg Clayton CommandObjectDisassemble::CommandOptions::OptionParsingStarting ()
15230fdc8d8SChris Lattner {
15330fdc8d8SChris Lattner     show_mixed = false;
15430fdc8d8SChris Lattner     show_bytes = false;
15530fdc8d8SChris Lattner     num_lines_context = 0;
15637023b06SJim Ingham     num_instructions = 0;
15732e0a750SGreg Clayton     func_name.clear();
1583555b5d7SJim Ingham     cur_function = false;
15932e0a750SGreg Clayton     at_pc = false;
16032e0a750SGreg Clayton     frame_line = false;
16132e0a750SGreg Clayton     start_addr = LLDB_INVALID_ADDRESS;
16232e0a750SGreg Clayton     end_addr = LLDB_INVALID_ADDRESS;
163a68c1a21SSean Callanan     raw = false;
16432e0a750SGreg Clayton     plugin_name.clear();
16532e0a750SGreg Clayton     arch.Clear();
1663555b5d7SJim Ingham     some_location_specified = false;
1673555b5d7SJim Ingham }
1683555b5d7SJim Ingham 
1693555b5d7SJim Ingham Error
1703555b5d7SJim Ingham CommandObjectDisassemble::CommandOptions::OptionParsingFinished ()
1713555b5d7SJim Ingham {
1723555b5d7SJim Ingham     if (!some_location_specified)
1733555b5d7SJim Ingham         at_pc = true;
1743555b5d7SJim Ingham     return Error();
1753555b5d7SJim Ingham 
17630fdc8d8SChris Lattner }
17730fdc8d8SChris Lattner 
178e0d378b3SGreg Clayton const OptionDefinition*
17930fdc8d8SChris Lattner CommandObjectDisassemble::CommandOptions::GetDefinitions ()
18030fdc8d8SChris Lattner {
18130fdc8d8SChris Lattner     return g_option_table;
18230fdc8d8SChris Lattner }
18330fdc8d8SChris Lattner 
184e0d378b3SGreg Clayton OptionDefinition
18530fdc8d8SChris Lattner CommandObjectDisassemble::CommandOptions::g_option_table[] =
18630fdc8d8SChris Lattner {
187deaab222SCaroline Tice { LLDB_OPT_SET_ALL  , false , "bytes",          'b', no_argument        , NULL, 0, eArgTypeNone,        "Show opcode bytes when disassembling."},
188357132ebSGreg Clayton { LLDB_OPT_SET_ALL  , false , "context",        'C', required_argument  , NULL, 0, eArgTypeNumLines,    "Number of context lines of source to show."},
189deaab222SCaroline Tice { LLDB_OPT_SET_ALL  , false , "mixed",          'm', no_argument        , NULL, 0, eArgTypeNone,        "Enable mixed source and assembly display."},
190deaab222SCaroline Tice { LLDB_OPT_SET_ALL  , false , "raw",            'r', no_argument        , NULL, 0, eArgTypeNone,        "Print raw disassembly with no symbol information."},
1911080edbcSGreg Clayton { LLDB_OPT_SET_ALL  , false , "plugin",         'P', required_argument  , NULL, 0, eArgTypePlugin,      "Name of the disassembler plugin you want to use."},
192357132ebSGreg Clayton { LLDB_OPT_SET_ALL  , false , "arch",           'a', required_argument  , NULL, 0, eArgTypeArchitecture,"Specify the architecture to use from cross disassembly."},
193357132ebSGreg Clayton { LLDB_OPT_SET_1 |
194357132ebSGreg Clayton   LLDB_OPT_SET_2    , true  , "start-address" , 's', required_argument  , NULL, 0, eArgTypeStartAddress,"Address at which to start disassembling."},
195405fe67fSCaroline Tice { LLDB_OPT_SET_1    , false , "end-address"  ,  'e', required_argument  , NULL, 0, eArgTypeEndAddress,  "Address at which to end disassembling."},
196357132ebSGreg Clayton { LLDB_OPT_SET_2 |
197357132ebSGreg Clayton   LLDB_OPT_SET_3 |
198357132ebSGreg Clayton   LLDB_OPT_SET_4 |
199357132ebSGreg Clayton   LLDB_OPT_SET_5    , false , "count",          'c', required_argument  , NULL, 0, eArgTypeNumLines,    "Number of instructions to display."},
2003555b5d7SJim Ingham { LLDB_OPT_SET_3    , false  , "name",           'n', required_argument  , NULL, CommandCompletions::eSymbolCompletion, eArgTypeFunctionName,             "Disassemble entire contents of the given function name."},
2013555b5d7SJim Ingham { LLDB_OPT_SET_4    , false  , "frame",          'f', no_argument        , NULL, 0, eArgTypeNone,        "Disassemble from the start of the current frame's function."},
2023555b5d7SJim Ingham { LLDB_OPT_SET_5    , false  , "pc",             'p', no_argument        , NULL, 0, eArgTypeNone,        "Disassemble around the current pc."},
2033555b5d7SJim Ingham { LLDB_OPT_SET_6    , false  , "line",           'l', no_argument        , NULL, 0, eArgTypeNone,        "Disassemble the current frame's current source line instructions if there debug line table information, else disasemble around the pc."},
204deaab222SCaroline Tice { 0                 , false , NULL,             0,   0                  , NULL, 0, eArgTypeNone,        NULL }
20530fdc8d8SChris Lattner };
20630fdc8d8SChris Lattner 
20730fdc8d8SChris Lattner 
20830fdc8d8SChris Lattner 
20930fdc8d8SChris Lattner //-------------------------------------------------------------------------
21030fdc8d8SChris Lattner // CommandObjectDisassemble
21130fdc8d8SChris Lattner //-------------------------------------------------------------------------
21230fdc8d8SChris Lattner 
213a7015092SGreg Clayton CommandObjectDisassemble::CommandObjectDisassemble (CommandInterpreter &interpreter) :
2145a988416SJim Ingham     CommandObjectParsed (interpreter,
215a7015092SGreg Clayton                          "disassemble",
2163f4c09c1SCaroline Tice                          "Disassemble bytes in the current function, or elsewhere in the executable program as specified by the user.",
217eb0103f2SGreg Clayton                          "disassemble [<cmd-options>]"),
218eb0103f2SGreg Clayton     m_options (interpreter)
21930fdc8d8SChris Lattner {
22030fdc8d8SChris Lattner }
22130fdc8d8SChris Lattner 
22230fdc8d8SChris Lattner CommandObjectDisassemble::~CommandObjectDisassemble()
22330fdc8d8SChris Lattner {
22430fdc8d8SChris Lattner }
22530fdc8d8SChris Lattner 
22630fdc8d8SChris Lattner bool
2275a988416SJim Ingham CommandObjectDisassemble::DoExecute (Args& command, CommandReturnObject &result)
22830fdc8d8SChris Lattner {
229a7015092SGreg Clayton     Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get();
23030fdc8d8SChris Lattner     if (target == NULL)
23130fdc8d8SChris Lattner     {
232effe5c95SGreg Clayton         result.AppendError ("invalid target, create a debug target using the 'target create' command");
23330fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
23430fdc8d8SChris Lattner         return false;
23530fdc8d8SChris Lattner     }
23632e0a750SGreg Clayton     if (!m_options.arch.IsValid())
23732e0a750SGreg Clayton         m_options.arch = target->GetArchitecture();
23830fdc8d8SChris Lattner 
23932e0a750SGreg Clayton     if (!m_options.arch.IsValid())
24030fdc8d8SChris Lattner     {
241357132ebSGreg Clayton         result.AppendError ("use the --arch option or set the target architecure to disassemble");
24230fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
24330fdc8d8SChris Lattner         return false;
24430fdc8d8SChris Lattner     }
24530fdc8d8SChris Lattner 
2461080edbcSGreg Clayton     const char *plugin_name = m_options.GetPluginName ();
2477e6d4e5aSSean Callanan     DisassemblerSP disassembler = Disassembler::FindPlugin(m_options.arch, plugin_name);
24830fdc8d8SChris Lattner 
2499a028519SSean Callanan     if (!disassembler)
25030fdc8d8SChris Lattner     {
2511080edbcSGreg Clayton         if (plugin_name)
252357132ebSGreg Clayton             result.AppendErrorWithFormat ("Unable to find Disassembler plug-in named '%s' that supports the '%s' architecture.\n",
253357132ebSGreg Clayton                                           plugin_name,
25432e0a750SGreg Clayton                                           m_options.arch.GetArchitectureName());
2551080edbcSGreg Clayton         else
256357132ebSGreg Clayton             result.AppendErrorWithFormat ("Unable to find Disassembler plug-in for the '%s' architecture.\n",
25732e0a750SGreg Clayton                                           m_options.arch.GetArchitectureName());
25830fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
25930fdc8d8SChris Lattner         return false;
26030fdc8d8SChris Lattner     }
26130fdc8d8SChris Lattner 
26230fdc8d8SChris Lattner     result.SetStatus (eReturnStatusSuccessFinishResult);
26330fdc8d8SChris Lattner 
264dda4f7b5SGreg Clayton     if (command.GetArgumentCount() != 0)
26530fdc8d8SChris Lattner     {
266a7015092SGreg Clayton         result.AppendErrorWithFormat ("\"disassemble\" arguments are specified as options.\n");
267eb0103f2SGreg Clayton         GetOptions()->GenerateOptionUsage (result.GetErrorStream(), this);
2688651121cSJim Ingham         result.SetStatus (eReturnStatusFailed);
2698651121cSJim Ingham         return false;
2708651121cSJim Ingham     }
2718651121cSJim Ingham 
272dda4f7b5SGreg Clayton     if (m_options.show_mixed && m_options.num_lines_context == 0)
2736dbd3983SGreg Clayton         m_options.num_lines_context = 1;
274dda4f7b5SGreg Clayton 
2758b82f087SGreg Clayton     ExecutionContext exe_ctx(m_interpreter.GetExecutionContext());
276b10d72f0SGreg Clayton     // Always show the PC in the disassembly
277b10d72f0SGreg Clayton     uint32_t options = Disassembler::eOptionMarkPCAddress;
2781da6f9d7SGreg Clayton 
279b10d72f0SGreg Clayton     // Mark the source line for the current PC only if we are doing mixed source and assembly
280b10d72f0SGreg Clayton     if (m_options.show_mixed)
281b10d72f0SGreg Clayton         options |= Disassembler::eOptionMarkPCSourceLine;
2821da6f9d7SGreg Clayton 
2831da6f9d7SGreg Clayton     if (m_options.show_bytes)
2841da6f9d7SGreg Clayton         options |= Disassembler::eOptionShowBytes;
2851da6f9d7SGreg Clayton 
2861da6f9d7SGreg Clayton     if (m_options.raw)
2871da6f9d7SGreg Clayton         options |= Disassembler::eOptionRawOuput;
28837023b06SJim Ingham 
28932e0a750SGreg Clayton     if (!m_options.func_name.empty())
290dda4f7b5SGreg Clayton     {
29132e0a750SGreg Clayton         ConstString name(m_options.func_name.c_str());
292dda4f7b5SGreg Clayton 
293a7015092SGreg Clayton         if (Disassembler::Disassemble (m_interpreter.GetDebugger(),
29432e0a750SGreg Clayton                                        m_options.arch,
2951080edbcSGreg Clayton                                        plugin_name,
296dda4f7b5SGreg Clayton                                        exe_ctx,
297dda4f7b5SGreg Clayton                                        name,
298dda4f7b5SGreg Clayton                                        NULL,    // Module *
29937023b06SJim Ingham                                        m_options.num_instructions,
300dda4f7b5SGreg Clayton                                        m_options.show_mixed ? m_options.num_lines_context : 0,
3011da6f9d7SGreg Clayton                                        options,
302dda4f7b5SGreg Clayton                                        result.GetOutputStream()))
303dda4f7b5SGreg Clayton         {
304dda4f7b5SGreg Clayton             result.SetStatus (eReturnStatusSuccessFinishResult);
305dda4f7b5SGreg Clayton         }
306dda4f7b5SGreg Clayton         else
307dda4f7b5SGreg Clayton         {
308dda4f7b5SGreg Clayton             result.AppendErrorWithFormat ("Unable to find symbol with name '%s'.\n", name.GetCString());
309dda4f7b5SGreg Clayton             result.SetStatus (eReturnStatusFailed);
310dda4f7b5SGreg Clayton         }
311dda4f7b5SGreg Clayton     }
312dda4f7b5SGreg Clayton     else
313dda4f7b5SGreg Clayton     {
31432e0a750SGreg Clayton         AddressRange range;
315c14ee32dSGreg Clayton         StackFrame *frame = exe_ctx.GetFramePtr();
31632e0a750SGreg Clayton         if (m_options.frame_line)
31732e0a750SGreg Clayton         {
318c14ee32dSGreg Clayton             if (frame == NULL)
3193555b5d7SJim Ingham             {
3203555b5d7SJim Ingham                 result.AppendError ("Cannot disassemble around the current line without a selected frame.\n");
3213555b5d7SJim Ingham                 result.SetStatus (eReturnStatusFailed);
3223555b5d7SJim Ingham                 return false;
3233555b5d7SJim Ingham             }
324c14ee32dSGreg Clayton             LineEntry pc_line_entry (frame->GetSymbolContext(eSymbolContextLineEntry).line_entry);
32532e0a750SGreg Clayton             if (pc_line_entry.IsValid())
32632e0a750SGreg Clayton             {
32732e0a750SGreg Clayton                 range = pc_line_entry.range;
32832e0a750SGreg Clayton             }
32932e0a750SGreg Clayton             else
33032e0a750SGreg Clayton             {
33132e0a750SGreg Clayton                 m_options.at_pc = true; // No line entry, so just disassemble around the current pc
33232e0a750SGreg Clayton                 m_options.show_mixed = false;
33332e0a750SGreg Clayton             }
33432e0a750SGreg Clayton         }
3353555b5d7SJim Ingham         else if (m_options.cur_function)
3363555b5d7SJim Ingham         {
337c14ee32dSGreg Clayton             if (frame == NULL)
3383555b5d7SJim Ingham             {
3393555b5d7SJim Ingham                 result.AppendError ("Cannot disassemble around the current function without a selected frame.\n");
3403555b5d7SJim Ingham                 result.SetStatus (eReturnStatusFailed);
3413555b5d7SJim Ingham                 return false;
3423555b5d7SJim Ingham             }
343c14ee32dSGreg Clayton             Symbol *symbol = frame->GetSymbolContext(eSymbolContextSymbol).symbol;
3443555b5d7SJim Ingham             if (symbol)
345e7612134SGreg Clayton             {
346e7612134SGreg Clayton                 range.GetBaseAddress() = symbol->GetAddress();
347e7612134SGreg Clayton                 range.SetByteSize(symbol->GetByteSize());
348e7612134SGreg Clayton             }
3493555b5d7SJim Ingham         }
35037023b06SJim Ingham 
35132e0a750SGreg Clayton         // Did the "m_options.frame_line" find a valid range already? If so
35232e0a750SGreg Clayton         // skip the rest...
35332e0a750SGreg Clayton         if (range.GetByteSize() == 0)
35432e0a750SGreg Clayton         {
35532e0a750SGreg Clayton             if (m_options.at_pc)
3568651121cSJim Ingham             {
357c14ee32dSGreg Clayton                 if (frame == NULL)
35837023b06SJim Ingham                 {
35937023b06SJim Ingham                     result.AppendError ("Cannot disassemble around the current PC without a selected frame.\n");
36037023b06SJim Ingham                     result.SetStatus (eReturnStatusFailed);
36137023b06SJim Ingham                     return false;
36237023b06SJim Ingham                 }
363c14ee32dSGreg Clayton                 range.GetBaseAddress() = frame->GetFrameCodeAddress();
36437023b06SJim Ingham                 if (m_options.num_instructions == 0)
36537023b06SJim Ingham                 {
36637023b06SJim Ingham                     // Disassembling at the PC always disassembles some number of instructions (not the whole function).
36737023b06SJim Ingham                     m_options.num_instructions = DEFAULT_DISASM_NUM_INS;
36837023b06SJim Ingham                 }
36937023b06SJim Ingham             }
37037023b06SJim Ingham             else
37137023b06SJim Ingham             {
37232e0a750SGreg Clayton                 range.GetBaseAddress().SetOffset (m_options.start_addr);
37332e0a750SGreg Clayton                 if (range.GetBaseAddress().IsValid())
37437023b06SJim Ingham                 {
37532e0a750SGreg Clayton                     if (m_options.end_addr != LLDB_INVALID_ADDRESS)
3768651121cSJim Ingham                     {
37732e0a750SGreg Clayton                         if (m_options.end_addr <= m_options.start_addr)
3788651121cSJim Ingham                         {
3798651121cSJim Ingham                             result.AppendErrorWithFormat ("End address before start address.\n");
3808651121cSJim Ingham                             result.SetStatus (eReturnStatusFailed);
3818651121cSJim Ingham                             return false;
3828651121cSJim Ingham                         }
38332e0a750SGreg Clayton                         range.SetByteSize (m_options.end_addr - m_options.start_addr);
38432e0a750SGreg Clayton                     }
38537023b06SJim Ingham                 }
38637023b06SJim Ingham             }
38737023b06SJim Ingham         }
38837023b06SJim Ingham 
38937023b06SJim Ingham         if (m_options.num_instructions != 0)
39037023b06SJim Ingham         {
39132e0a750SGreg Clayton             if (!range.GetBaseAddress().IsValid())
39237023b06SJim Ingham             {
39337023b06SJim Ingham                 // The default action is to disassemble the current frame function.
394c14ee32dSGreg Clayton                 if (frame)
39537023b06SJim Ingham                 {
396c14ee32dSGreg Clayton                     SymbolContext sc(frame->GetSymbolContext(eSymbolContextFunction | eSymbolContextSymbol));
39737023b06SJim Ingham                     if (sc.function)
39832e0a750SGreg Clayton                         range.GetBaseAddress() = sc.function->GetAddressRange().GetBaseAddress();
399e7612134SGreg Clayton                     else if (sc.symbol && sc.symbol->ValueIsAddress())
400e7612134SGreg Clayton                         range.GetBaseAddress() = sc.symbol->GetAddress();
40137023b06SJim Ingham                     else
402c14ee32dSGreg Clayton                         range.GetBaseAddress() = frame->GetFrameCodeAddress();
40337023b06SJim Ingham                 }
40437023b06SJim Ingham 
40532e0a750SGreg Clayton                 if (!range.GetBaseAddress().IsValid())
40637023b06SJim Ingham                 {
40737023b06SJim Ingham                     result.AppendError ("invalid frame");
40837023b06SJim Ingham                     result.SetStatus (eReturnStatusFailed);
40937023b06SJim Ingham                     return false;
41037023b06SJim Ingham                 }
41137023b06SJim Ingham             }
41237023b06SJim Ingham 
41337023b06SJim Ingham             if (Disassembler::Disassemble (m_interpreter.GetDebugger(),
41432e0a750SGreg Clayton                                            m_options.arch,
4151080edbcSGreg Clayton                                            plugin_name,
41637023b06SJim Ingham                                            exe_ctx,
41732e0a750SGreg Clayton                                            range.GetBaseAddress(),
41837023b06SJim Ingham                                            m_options.num_instructions,
41937023b06SJim Ingham                                            m_options.show_mixed ? m_options.num_lines_context : 0,
4201da6f9d7SGreg Clayton                                            options,
42137023b06SJim Ingham                                            result.GetOutputStream()))
42237023b06SJim Ingham             {
42337023b06SJim Ingham                 result.SetStatus (eReturnStatusSuccessFinishResult);
4248651121cSJim Ingham             }
4258651121cSJim Ingham             else
42637023b06SJim Ingham             {
427d01b2953SDaniel Malea                 result.AppendErrorWithFormat ("Failed to disassemble memory at 0x%8.8" PRIx64 ".\n", m_options.start_addr);
42837023b06SJim Ingham                 result.SetStatus (eReturnStatusFailed);
42937023b06SJim Ingham             }
43037023b06SJim Ingham         }
43137023b06SJim Ingham         else
43237023b06SJim Ingham         {
43332e0a750SGreg Clayton             if (!range.GetBaseAddress().IsValid())
43430fdc8d8SChris Lattner             {
4358ceb8ba2SJohnny Chen                 // The default action is to disassemble the current frame function.
436c14ee32dSGreg Clayton                 if (frame)
43730fdc8d8SChris Lattner                 {
438c14ee32dSGreg Clayton                     SymbolContext sc(frame->GetSymbolContext(eSymbolContextFunction | eSymbolContextSymbol));
43930fdc8d8SChris Lattner                     if (sc.function)
440dda4f7b5SGreg Clayton                         range = sc.function->GetAddressRange();
441e7612134SGreg Clayton                     else if (sc.symbol && sc.symbol->ValueIsAddress())
442e7612134SGreg Clayton                     {
443e7612134SGreg Clayton                         range.GetBaseAddress() = sc.symbol->GetAddress();
444e7612134SGreg Clayton                         range.SetByteSize (sc.symbol->GetByteSize());
445e7612134SGreg Clayton                     }
44630fdc8d8SChris Lattner                     else
447c14ee32dSGreg Clayton                         range.GetBaseAddress() = frame->GetFrameCodeAddress();
44830fdc8d8SChris Lattner                 }
44930fdc8d8SChris Lattner                 else
45030fdc8d8SChris Lattner                 {
45130fdc8d8SChris Lattner                     result.AppendError ("invalid frame");
45230fdc8d8SChris Lattner                     result.SetStatus (eReturnStatusFailed);
45330fdc8d8SChris Lattner                     return false;
45430fdc8d8SChris Lattner                 }
45530fdc8d8SChris Lattner             }
456dda4f7b5SGreg Clayton             if (range.GetByteSize() == 0)
457dda4f7b5SGreg Clayton                 range.SetByteSize(DEFAULT_DISASM_BYTE_SIZE);
45830fdc8d8SChris Lattner 
459a7015092SGreg Clayton             if (Disassembler::Disassemble (m_interpreter.GetDebugger(),
46032e0a750SGreg Clayton                                            m_options.arch,
4611080edbcSGreg Clayton                                            plugin_name,
462dda4f7b5SGreg Clayton                                            exe_ctx,
463dda4f7b5SGreg Clayton                                            range,
46437023b06SJim Ingham                                            m_options.num_instructions,
465dda4f7b5SGreg Clayton                                            m_options.show_mixed ? m_options.num_lines_context : 0,
4661da6f9d7SGreg Clayton                                            options,
467dda4f7b5SGreg Clayton                                            result.GetOutputStream()))
46830fdc8d8SChris Lattner             {
469dda4f7b5SGreg Clayton                 result.SetStatus (eReturnStatusSuccessFinishResult);
47030fdc8d8SChris Lattner             }
47130fdc8d8SChris Lattner             else
47230fdc8d8SChris Lattner             {
473d01b2953SDaniel Malea                 result.AppendErrorWithFormat ("Failed to disassemble memory at 0x%8.8" PRIx64 ".\n", m_options.start_addr);
47430fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusFailed);
47530fdc8d8SChris Lattner             }
47630fdc8d8SChris Lattner         }
47737023b06SJim Ingham     }
47830fdc8d8SChris Lattner 
47930fdc8d8SChris Lattner     return result.Succeeded();
48030fdc8d8SChris Lattner }
481