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"
1740af72e1SJim Ingham #include "lldb/Interpreter/Args.h"
1830fdc8d8SChris Lattner #include "lldb/Interpreter/CommandCompletions.h"
1930fdc8d8SChris Lattner #include "lldb/Interpreter/CommandInterpreter.h"
2030fdc8d8SChris Lattner #include "lldb/Interpreter/CommandReturnObject.h"
2130fdc8d8SChris Lattner #include "lldb/Core/Disassembler.h"
2240af72e1SJim Ingham #include "lldb/Interpreter/Options.h"
2330fdc8d8SChris Lattner #include "lldb/Core/SourceManager.h"
2430fdc8d8SChris Lattner #include "lldb/Target/StackFrame.h"
2530fdc8d8SChris Lattner #include "lldb/Symbol/Symbol.h"
2630fdc8d8SChris Lattner #include "lldb/Target/Process.h"
2730fdc8d8SChris Lattner #include "lldb/Target/Target.h"
2830fdc8d8SChris Lattner 
2930fdc8d8SChris Lattner #define DEFAULT_DISASM_BYTE_SIZE 32
3037023b06SJim Ingham #define DEFAULT_DISASM_NUM_INS  4
3130fdc8d8SChris Lattner 
3230fdc8d8SChris Lattner using namespace lldb;
3330fdc8d8SChris Lattner using namespace lldb_private;
3430fdc8d8SChris Lattner 
35eb0103f2SGreg Clayton CommandObjectDisassemble::CommandOptions::CommandOptions (CommandInterpreter &interpreter) :
361f1b269bSJohnny Chen     Options(interpreter),
3737023b06SJim Ingham     num_lines_context(0),
3837023b06SJim Ingham     num_instructions (0),
3932e0a750SGreg Clayton     func_name(),
403555b5d7SJim Ingham     cur_function (false),
4132e0a750SGreg Clayton     start_addr(),
4232e0a750SGreg Clayton     end_addr (),
4332e0a750SGreg Clayton     at_pc (false),
4432e0a750SGreg Clayton     frame_line (false),
4532e0a750SGreg Clayton     plugin_name (),
463555b5d7SJim Ingham     arch(),
473555b5d7SJim Ingham     some_location_specified (false)
4830fdc8d8SChris Lattner {
49f6b8b581SGreg Clayton     OptionParsingStarting();
5030fdc8d8SChris Lattner }
5130fdc8d8SChris Lattner 
5230fdc8d8SChris Lattner CommandObjectDisassemble::CommandOptions::~CommandOptions ()
5330fdc8d8SChris Lattner {
5430fdc8d8SChris Lattner }
5530fdc8d8SChris Lattner 
5630fdc8d8SChris Lattner Error
57f6b8b581SGreg Clayton CommandObjectDisassemble::CommandOptions::SetOptionValue (uint32_t option_idx, const char *option_arg)
5830fdc8d8SChris Lattner {
5930fdc8d8SChris Lattner     Error error;
6030fdc8d8SChris Lattner 
6130fdc8d8SChris Lattner     char short_option = (char) m_getopt_table[option_idx].val;
6230fdc8d8SChris Lattner 
6337023b06SJim Ingham     bool success;
6437023b06SJim Ingham 
6530fdc8d8SChris Lattner     switch (short_option)
6630fdc8d8SChris Lattner     {
6730fdc8d8SChris Lattner     case 'm':
6830fdc8d8SChris Lattner         show_mixed = true;
6930fdc8d8SChris Lattner         break;
7030fdc8d8SChris Lattner 
71357132ebSGreg Clayton     case 'C':
7237023b06SJim Ingham         num_lines_context = Args::StringToUInt32(option_arg, 0, 0, &success);
7337023b06SJim Ingham         if (!success)
7486edbf41SGreg Clayton             error.SetErrorStringWithFormat ("invalid num context lines string: \"%s\"", option_arg);
7537023b06SJim Ingham         break;
7637023b06SJim Ingham 
7730fdc8d8SChris Lattner     case 'c':
7837023b06SJim Ingham         num_instructions = Args::StringToUInt32(option_arg, 0, 0, &success);
7937023b06SJim Ingham         if (!success)
8086edbf41SGreg Clayton             error.SetErrorStringWithFormat ("invalid num of instructions string: \"%s\"", option_arg);
8130fdc8d8SChris Lattner         break;
8230fdc8d8SChris Lattner 
8330fdc8d8SChris Lattner     case 'b':
8430fdc8d8SChris Lattner         show_bytes = true;
8530fdc8d8SChris Lattner         break;
8630fdc8d8SChris Lattner 
878651121cSJim Ingham     case 's':
8832e0a750SGreg Clayton         start_addr = Args::StringToUInt64(option_arg, LLDB_INVALID_ADDRESS, 0);
8932e0a750SGreg Clayton         if (start_addr == LLDB_INVALID_ADDRESS)
9032e0a750SGreg Clayton             start_addr = Args::StringToUInt64(option_arg, LLDB_INVALID_ADDRESS, 16);
9130fdc8d8SChris Lattner 
9232e0a750SGreg Clayton         if (start_addr == LLDB_INVALID_ADDRESS)
9386edbf41SGreg Clayton             error.SetErrorStringWithFormat ("invalid start address string '%s'", option_arg);
943555b5d7SJim Ingham         some_location_specified = true;
958651121cSJim Ingham         break;
968651121cSJim Ingham     case 'e':
9732e0a750SGreg Clayton         end_addr = Args::StringToUInt64(option_arg, LLDB_INVALID_ADDRESS, 0);
9832e0a750SGreg Clayton         if (end_addr == LLDB_INVALID_ADDRESS)
9932e0a750SGreg Clayton             end_addr = Args::StringToUInt64(option_arg, LLDB_INVALID_ADDRESS, 16);
1008651121cSJim Ingham 
10132e0a750SGreg Clayton         if (end_addr == LLDB_INVALID_ADDRESS)
10286edbf41SGreg Clayton             error.SetErrorStringWithFormat ("invalid end address string '%s'", option_arg);
10330fdc8d8SChris Lattner         break;
1043555b5d7SJim Ingham         some_location_specified = true;
10530fdc8d8SChris Lattner     case 'n':
10632e0a750SGreg Clayton         func_name.assign (option_arg);
1073555b5d7SJim Ingham         some_location_specified = true;
10830fdc8d8SChris Lattner         break;
10930fdc8d8SChris Lattner 
11037023b06SJim Ingham     case 'p':
11132e0a750SGreg Clayton         at_pc = true;
1123555b5d7SJim Ingham         some_location_specified = true;
11332e0a750SGreg Clayton         break;
11432e0a750SGreg Clayton 
11532e0a750SGreg Clayton     case 'l':
11632e0a750SGreg Clayton         frame_line = true;
11732e0a750SGreg Clayton         // Disassemble the current source line kind of implies showing mixed
11832e0a750SGreg Clayton         // source code context.
11932e0a750SGreg Clayton         show_mixed = true;
1203555b5d7SJim Ingham         some_location_specified = true;
12137023b06SJim Ingham         break;
12237023b06SJim Ingham 
1231080edbcSGreg Clayton     case 'P':
12432e0a750SGreg Clayton         plugin_name.assign (option_arg);
1251080edbcSGreg Clayton         break;
1261080edbcSGreg Clayton 
12730fdc8d8SChris Lattner     case 'r':
12830fdc8d8SChris Lattner         raw = true;
12930fdc8d8SChris Lattner         break;
13030fdc8d8SChris Lattner 
1318ceb8ba2SJohnny Chen     case 'f':
1323555b5d7SJim Ingham         cur_function = true;
1333555b5d7SJim Ingham         some_location_specified = true;
1348ceb8ba2SJohnny Chen         break;
1358ceb8ba2SJohnny Chen 
136357132ebSGreg Clayton     case 'a':
13770512317SGreg Clayton         if (!arch.SetTriple (option_arg, m_interpreter.GetPlatform (true).get()))
13870512317SGreg Clayton             arch.SetTriple (option_arg);
139357132ebSGreg Clayton         break;
140357132ebSGreg Clayton 
14130fdc8d8SChris Lattner     default:
14286edbf41SGreg Clayton         error.SetErrorStringWithFormat("unrecognized short option '%c'", short_option);
14330fdc8d8SChris Lattner         break;
14430fdc8d8SChris Lattner     }
14530fdc8d8SChris Lattner 
14630fdc8d8SChris Lattner     return error;
14730fdc8d8SChris Lattner }
14830fdc8d8SChris Lattner 
14930fdc8d8SChris Lattner void
150f6b8b581SGreg Clayton CommandObjectDisassemble::CommandOptions::OptionParsingStarting ()
15130fdc8d8SChris Lattner {
15230fdc8d8SChris Lattner     show_mixed = false;
15330fdc8d8SChris Lattner     show_bytes = false;
15430fdc8d8SChris Lattner     num_lines_context = 0;
15537023b06SJim Ingham     num_instructions = 0;
15632e0a750SGreg Clayton     func_name.clear();
1573555b5d7SJim Ingham     cur_function = false;
15832e0a750SGreg Clayton     at_pc = false;
15932e0a750SGreg Clayton     frame_line = false;
16032e0a750SGreg Clayton     start_addr = LLDB_INVALID_ADDRESS;
16132e0a750SGreg Clayton     end_addr = LLDB_INVALID_ADDRESS;
162a68c1a21SSean Callanan     raw = false;
16332e0a750SGreg Clayton     plugin_name.clear();
16432e0a750SGreg Clayton     arch.Clear();
1653555b5d7SJim Ingham     some_location_specified = false;
1663555b5d7SJim Ingham }
1673555b5d7SJim Ingham 
1683555b5d7SJim Ingham Error
1693555b5d7SJim Ingham CommandObjectDisassemble::CommandOptions::OptionParsingFinished ()
1703555b5d7SJim Ingham {
1713555b5d7SJim Ingham     if (!some_location_specified)
1723555b5d7SJim Ingham         at_pc = true;
1733555b5d7SJim Ingham     return Error();
1743555b5d7SJim Ingham 
17530fdc8d8SChris Lattner }
17630fdc8d8SChris Lattner 
177e0d378b3SGreg Clayton const OptionDefinition*
17830fdc8d8SChris Lattner CommandObjectDisassemble::CommandOptions::GetDefinitions ()
17930fdc8d8SChris Lattner {
18030fdc8d8SChris Lattner     return g_option_table;
18130fdc8d8SChris Lattner }
18230fdc8d8SChris Lattner 
183e0d378b3SGreg Clayton OptionDefinition
18430fdc8d8SChris Lattner CommandObjectDisassemble::CommandOptions::g_option_table[] =
18530fdc8d8SChris Lattner {
186deaab222SCaroline Tice { LLDB_OPT_SET_ALL  , false , "bytes",          'b', no_argument        , NULL, 0, eArgTypeNone,        "Show opcode bytes when disassembling."},
187357132ebSGreg Clayton { LLDB_OPT_SET_ALL  , false , "context",        'C', required_argument  , NULL, 0, eArgTypeNumLines,    "Number of context lines of source to show."},
188deaab222SCaroline Tice { LLDB_OPT_SET_ALL  , false , "mixed",          'm', no_argument        , NULL, 0, eArgTypeNone,        "Enable mixed source and assembly display."},
189deaab222SCaroline Tice { LLDB_OPT_SET_ALL  , false , "raw",            'r', no_argument        , NULL, 0, eArgTypeNone,        "Print raw disassembly with no symbol information."},
1901080edbcSGreg Clayton { LLDB_OPT_SET_ALL  , false , "plugin",         'P', required_argument  , NULL, 0, eArgTypePlugin,      "Name of the disassembler plugin you want to use."},
191357132ebSGreg Clayton { LLDB_OPT_SET_ALL  , false , "arch",           'a', required_argument  , NULL, 0, eArgTypeArchitecture,"Specify the architecture to use from cross disassembly."},
192357132ebSGreg Clayton { LLDB_OPT_SET_1 |
193357132ebSGreg Clayton   LLDB_OPT_SET_2    , true  , "start-address" , 's', required_argument  , NULL, 0, eArgTypeStartAddress,"Address at which to start disassembling."},
194405fe67fSCaroline Tice { LLDB_OPT_SET_1    , false , "end-address"  ,  'e', required_argument  , NULL, 0, eArgTypeEndAddress,  "Address at which to end disassembling."},
195357132ebSGreg Clayton { LLDB_OPT_SET_2 |
196357132ebSGreg Clayton   LLDB_OPT_SET_3 |
197357132ebSGreg Clayton   LLDB_OPT_SET_4 |
198357132ebSGreg Clayton   LLDB_OPT_SET_5    , false , "count",          'c', required_argument  , NULL, 0, eArgTypeNumLines,    "Number of instructions to display."},
1993555b5d7SJim Ingham { LLDB_OPT_SET_3    , false  , "name",           'n', required_argument  , NULL, CommandCompletions::eSymbolCompletion, eArgTypeFunctionName,             "Disassemble entire contents of the given function name."},
2003555b5d7SJim Ingham { LLDB_OPT_SET_4    , false  , "frame",          'f', no_argument        , NULL, 0, eArgTypeNone,        "Disassemble from the start of the current frame's function."},
2013555b5d7SJim Ingham { LLDB_OPT_SET_5    , false  , "pc",             'p', no_argument        , NULL, 0, eArgTypeNone,        "Disassemble around the current pc."},
2023555b5d7SJim 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."},
203deaab222SCaroline Tice { 0                 , false , NULL,             0,   0                  , NULL, 0, eArgTypeNone,        NULL }
20430fdc8d8SChris Lattner };
20530fdc8d8SChris Lattner 
20630fdc8d8SChris Lattner 
20730fdc8d8SChris Lattner 
20830fdc8d8SChris Lattner //-------------------------------------------------------------------------
20930fdc8d8SChris Lattner // CommandObjectDisassemble
21030fdc8d8SChris Lattner //-------------------------------------------------------------------------
21130fdc8d8SChris Lattner 
212a7015092SGreg Clayton CommandObjectDisassemble::CommandObjectDisassemble (CommandInterpreter &interpreter) :
2135a988416SJim Ingham     CommandObjectParsed (interpreter,
214a7015092SGreg Clayton                          "disassemble",
2153f4c09c1SCaroline Tice                          "Disassemble bytes in the current function, or elsewhere in the executable program as specified by the user.",
216eb0103f2SGreg Clayton                          "disassemble [<cmd-options>]"),
217eb0103f2SGreg Clayton     m_options (interpreter)
21830fdc8d8SChris Lattner {
21930fdc8d8SChris Lattner }
22030fdc8d8SChris Lattner 
22130fdc8d8SChris Lattner CommandObjectDisassemble::~CommandObjectDisassemble()
22230fdc8d8SChris Lattner {
22330fdc8d8SChris Lattner }
22430fdc8d8SChris Lattner 
22530fdc8d8SChris Lattner bool
2265a988416SJim Ingham CommandObjectDisassemble::DoExecute (Args& command, CommandReturnObject &result)
22730fdc8d8SChris Lattner {
228a7015092SGreg Clayton     Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get();
22930fdc8d8SChris Lattner     if (target == NULL)
23030fdc8d8SChris Lattner     {
231effe5c95SGreg Clayton         result.AppendError ("invalid target, create a debug target using the 'target create' command");
23230fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
23330fdc8d8SChris Lattner         return false;
23430fdc8d8SChris Lattner     }
23532e0a750SGreg Clayton     if (!m_options.arch.IsValid())
23632e0a750SGreg Clayton         m_options.arch = target->GetArchitecture();
23730fdc8d8SChris Lattner 
23832e0a750SGreg Clayton     if (!m_options.arch.IsValid())
23930fdc8d8SChris Lattner     {
240357132ebSGreg Clayton         result.AppendError ("use the --arch option or set the target architecure to disassemble");
24130fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
24230fdc8d8SChris Lattner         return false;
24330fdc8d8SChris Lattner     }
24430fdc8d8SChris Lattner 
2451080edbcSGreg Clayton     const char *plugin_name = m_options.GetPluginName ();
246*7e6d4e5aSSean Callanan     DisassemblerSP disassembler = Disassembler::FindPlugin(m_options.arch, plugin_name);
24730fdc8d8SChris Lattner 
24830fdc8d8SChris Lattner     if (disassembler == NULL)
24930fdc8d8SChris Lattner     {
2501080edbcSGreg Clayton         if (plugin_name)
251357132ebSGreg Clayton             result.AppendErrorWithFormat ("Unable to find Disassembler plug-in named '%s' that supports the '%s' architecture.\n",
252357132ebSGreg Clayton                                           plugin_name,
25332e0a750SGreg Clayton                                           m_options.arch.GetArchitectureName());
2541080edbcSGreg Clayton         else
255357132ebSGreg Clayton             result.AppendErrorWithFormat ("Unable to find Disassembler plug-in for the '%s' architecture.\n",
25632e0a750SGreg Clayton                                           m_options.arch.GetArchitectureName());
25730fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
25830fdc8d8SChris Lattner         return false;
25930fdc8d8SChris Lattner     }
26030fdc8d8SChris Lattner 
26130fdc8d8SChris Lattner     result.SetStatus (eReturnStatusSuccessFinishResult);
26230fdc8d8SChris Lattner 
263dda4f7b5SGreg Clayton     if (command.GetArgumentCount() != 0)
26430fdc8d8SChris Lattner     {
265a7015092SGreg Clayton         result.AppendErrorWithFormat ("\"disassemble\" arguments are specified as options.\n");
266eb0103f2SGreg Clayton         GetOptions()->GenerateOptionUsage (result.GetErrorStream(), this);
2678651121cSJim Ingham         result.SetStatus (eReturnStatusFailed);
2688651121cSJim Ingham         return false;
2698651121cSJim Ingham     }
2708651121cSJim Ingham 
271dda4f7b5SGreg Clayton     if (m_options.show_mixed && m_options.num_lines_context == 0)
2726dbd3983SGreg Clayton         m_options.num_lines_context = 1;
273dda4f7b5SGreg Clayton 
2748b82f087SGreg Clayton     ExecutionContext exe_ctx(m_interpreter.GetExecutionContext());
275b10d72f0SGreg Clayton     // Always show the PC in the disassembly
276b10d72f0SGreg Clayton     uint32_t options = Disassembler::eOptionMarkPCAddress;
2771da6f9d7SGreg Clayton 
278b10d72f0SGreg Clayton     // Mark the source line for the current PC only if we are doing mixed source and assembly
279b10d72f0SGreg Clayton     if (m_options.show_mixed)
280b10d72f0SGreg Clayton         options |= Disassembler::eOptionMarkPCSourceLine;
2811da6f9d7SGreg Clayton 
2821da6f9d7SGreg Clayton     if (m_options.show_bytes)
2831da6f9d7SGreg Clayton         options |= Disassembler::eOptionShowBytes;
2841da6f9d7SGreg Clayton 
2851da6f9d7SGreg Clayton     if (m_options.raw)
2861da6f9d7SGreg Clayton         options |= Disassembler::eOptionRawOuput;
28737023b06SJim Ingham 
28832e0a750SGreg Clayton     if (!m_options.func_name.empty())
289dda4f7b5SGreg Clayton     {
29032e0a750SGreg Clayton         ConstString name(m_options.func_name.c_str());
291dda4f7b5SGreg Clayton 
292a7015092SGreg Clayton         if (Disassembler::Disassemble (m_interpreter.GetDebugger(),
29332e0a750SGreg Clayton                                        m_options.arch,
2941080edbcSGreg Clayton                                        plugin_name,
295dda4f7b5SGreg Clayton                                        exe_ctx,
296dda4f7b5SGreg Clayton                                        name,
297dda4f7b5SGreg Clayton                                        NULL,    // Module *
29837023b06SJim Ingham                                        m_options.num_instructions,
299dda4f7b5SGreg Clayton                                        m_options.show_mixed ? m_options.num_lines_context : 0,
3001da6f9d7SGreg Clayton                                        options,
301dda4f7b5SGreg Clayton                                        result.GetOutputStream()))
302dda4f7b5SGreg Clayton         {
303dda4f7b5SGreg Clayton             result.SetStatus (eReturnStatusSuccessFinishResult);
304dda4f7b5SGreg Clayton         }
305dda4f7b5SGreg Clayton         else
306dda4f7b5SGreg Clayton         {
307dda4f7b5SGreg Clayton             result.AppendErrorWithFormat ("Unable to find symbol with name '%s'.\n", name.GetCString());
308dda4f7b5SGreg Clayton             result.SetStatus (eReturnStatusFailed);
309dda4f7b5SGreg Clayton         }
310dda4f7b5SGreg Clayton     }
311dda4f7b5SGreg Clayton     else
312dda4f7b5SGreg Clayton     {
31332e0a750SGreg Clayton         AddressRange range;
314c14ee32dSGreg Clayton         StackFrame *frame = exe_ctx.GetFramePtr();
31532e0a750SGreg Clayton         if (m_options.frame_line)
31632e0a750SGreg Clayton         {
317c14ee32dSGreg Clayton             if (frame == NULL)
3183555b5d7SJim Ingham             {
3193555b5d7SJim Ingham                 result.AppendError ("Cannot disassemble around the current line without a selected frame.\n");
3203555b5d7SJim Ingham                 result.SetStatus (eReturnStatusFailed);
3213555b5d7SJim Ingham                 return false;
3223555b5d7SJim Ingham             }
323c14ee32dSGreg Clayton             LineEntry pc_line_entry (frame->GetSymbolContext(eSymbolContextLineEntry).line_entry);
32432e0a750SGreg Clayton             if (pc_line_entry.IsValid())
32532e0a750SGreg Clayton             {
32632e0a750SGreg Clayton                 range = pc_line_entry.range;
32732e0a750SGreg Clayton             }
32832e0a750SGreg Clayton             else
32932e0a750SGreg Clayton             {
33032e0a750SGreg Clayton                 m_options.at_pc = true; // No line entry, so just disassemble around the current pc
33132e0a750SGreg Clayton                 m_options.show_mixed = false;
33232e0a750SGreg Clayton             }
33332e0a750SGreg Clayton         }
3343555b5d7SJim Ingham         else if (m_options.cur_function)
3353555b5d7SJim Ingham         {
336c14ee32dSGreg Clayton             if (frame == NULL)
3373555b5d7SJim Ingham             {
3383555b5d7SJim Ingham                 result.AppendError ("Cannot disassemble around the current function without a selected frame.\n");
3393555b5d7SJim Ingham                 result.SetStatus (eReturnStatusFailed);
3403555b5d7SJim Ingham                 return false;
3413555b5d7SJim Ingham             }
342c14ee32dSGreg Clayton             Symbol *symbol = frame->GetSymbolContext(eSymbolContextSymbol).symbol;
3433555b5d7SJim Ingham             if (symbol)
344e7612134SGreg Clayton             {
345e7612134SGreg Clayton                 range.GetBaseAddress() = symbol->GetAddress();
346e7612134SGreg Clayton                 range.SetByteSize(symbol->GetByteSize());
347e7612134SGreg Clayton             }
3483555b5d7SJim Ingham         }
34937023b06SJim Ingham 
35032e0a750SGreg Clayton         // Did the "m_options.frame_line" find a valid range already? If so
35132e0a750SGreg Clayton         // skip the rest...
35232e0a750SGreg Clayton         if (range.GetByteSize() == 0)
35332e0a750SGreg Clayton         {
35432e0a750SGreg Clayton             if (m_options.at_pc)
3558651121cSJim Ingham             {
356c14ee32dSGreg Clayton                 if (frame == NULL)
35737023b06SJim Ingham                 {
35837023b06SJim Ingham                     result.AppendError ("Cannot disassemble around the current PC without a selected frame.\n");
35937023b06SJim Ingham                     result.SetStatus (eReturnStatusFailed);
36037023b06SJim Ingham                     return false;
36137023b06SJim Ingham                 }
362c14ee32dSGreg Clayton                 range.GetBaseAddress() = frame->GetFrameCodeAddress();
36337023b06SJim Ingham                 if (m_options.num_instructions == 0)
36437023b06SJim Ingham                 {
36537023b06SJim Ingham                     // Disassembling at the PC always disassembles some number of instructions (not the whole function).
36637023b06SJim Ingham                     m_options.num_instructions = DEFAULT_DISASM_NUM_INS;
36737023b06SJim Ingham                 }
36837023b06SJim Ingham             }
36937023b06SJim Ingham             else
37037023b06SJim Ingham             {
37132e0a750SGreg Clayton                 range.GetBaseAddress().SetOffset (m_options.start_addr);
37232e0a750SGreg Clayton                 if (range.GetBaseAddress().IsValid())
37337023b06SJim Ingham                 {
37432e0a750SGreg Clayton                     if (m_options.end_addr != LLDB_INVALID_ADDRESS)
3758651121cSJim Ingham                     {
37632e0a750SGreg Clayton                         if (m_options.end_addr <= m_options.start_addr)
3778651121cSJim Ingham                         {
3788651121cSJim Ingham                             result.AppendErrorWithFormat ("End address before start address.\n");
3798651121cSJim Ingham                             result.SetStatus (eReturnStatusFailed);
3808651121cSJim Ingham                             return false;
3818651121cSJim Ingham                         }
38232e0a750SGreg Clayton                         range.SetByteSize (m_options.end_addr - m_options.start_addr);
38332e0a750SGreg Clayton                     }
38437023b06SJim Ingham                 }
38537023b06SJim Ingham             }
38637023b06SJim Ingham         }
38737023b06SJim Ingham 
38837023b06SJim Ingham         if (m_options.num_instructions != 0)
38937023b06SJim Ingham         {
39032e0a750SGreg Clayton             if (!range.GetBaseAddress().IsValid())
39137023b06SJim Ingham             {
39237023b06SJim Ingham                 // The default action is to disassemble the current frame function.
393c14ee32dSGreg Clayton                 if (frame)
39437023b06SJim Ingham                 {
395c14ee32dSGreg Clayton                     SymbolContext sc(frame->GetSymbolContext(eSymbolContextFunction | eSymbolContextSymbol));
39637023b06SJim Ingham                     if (sc.function)
39732e0a750SGreg Clayton                         range.GetBaseAddress() = sc.function->GetAddressRange().GetBaseAddress();
398e7612134SGreg Clayton                     else if (sc.symbol && sc.symbol->ValueIsAddress())
399e7612134SGreg Clayton                         range.GetBaseAddress() = sc.symbol->GetAddress();
40037023b06SJim Ingham                     else
401c14ee32dSGreg Clayton                         range.GetBaseAddress() = frame->GetFrameCodeAddress();
40237023b06SJim Ingham                 }
40337023b06SJim Ingham 
40432e0a750SGreg Clayton                 if (!range.GetBaseAddress().IsValid())
40537023b06SJim Ingham                 {
40637023b06SJim Ingham                     result.AppendError ("invalid frame");
40737023b06SJim Ingham                     result.SetStatus (eReturnStatusFailed);
40837023b06SJim Ingham                     return false;
40937023b06SJim Ingham                 }
41037023b06SJim Ingham             }
41137023b06SJim Ingham 
41237023b06SJim Ingham             if (Disassembler::Disassemble (m_interpreter.GetDebugger(),
41332e0a750SGreg Clayton                                            m_options.arch,
4141080edbcSGreg Clayton                                            plugin_name,
41537023b06SJim Ingham                                            exe_ctx,
41632e0a750SGreg Clayton                                            range.GetBaseAddress(),
41737023b06SJim Ingham                                            m_options.num_instructions,
41837023b06SJim Ingham                                            m_options.show_mixed ? m_options.num_lines_context : 0,
4191da6f9d7SGreg Clayton                                            options,
42037023b06SJim Ingham                                            result.GetOutputStream()))
42137023b06SJim Ingham             {
42237023b06SJim Ingham                 result.SetStatus (eReturnStatusSuccessFinishResult);
4238651121cSJim Ingham             }
4248651121cSJim Ingham             else
42537023b06SJim Ingham             {
42632e0a750SGreg Clayton                 result.AppendErrorWithFormat ("Failed to disassemble memory at 0x%8.8llx.\n", m_options.start_addr);
42737023b06SJim Ingham                 result.SetStatus (eReturnStatusFailed);
42837023b06SJim Ingham             }
42937023b06SJim Ingham         }
43037023b06SJim Ingham         else
43137023b06SJim Ingham         {
43232e0a750SGreg Clayton             if (!range.GetBaseAddress().IsValid())
43330fdc8d8SChris Lattner             {
4348ceb8ba2SJohnny Chen                 // The default action is to disassemble the current frame function.
435c14ee32dSGreg Clayton                 if (frame)
43630fdc8d8SChris Lattner                 {
437c14ee32dSGreg Clayton                     SymbolContext sc(frame->GetSymbolContext(eSymbolContextFunction | eSymbolContextSymbol));
43830fdc8d8SChris Lattner                     if (sc.function)
439dda4f7b5SGreg Clayton                         range = sc.function->GetAddressRange();
440e7612134SGreg Clayton                     else if (sc.symbol && sc.symbol->ValueIsAddress())
441e7612134SGreg Clayton                     {
442e7612134SGreg Clayton                         range.GetBaseAddress() = sc.symbol->GetAddress();
443e7612134SGreg Clayton                         range.SetByteSize (sc.symbol->GetByteSize());
444e7612134SGreg Clayton                     }
44530fdc8d8SChris Lattner                     else
446c14ee32dSGreg Clayton                         range.GetBaseAddress() = frame->GetFrameCodeAddress();
44730fdc8d8SChris Lattner                 }
44830fdc8d8SChris Lattner                 else
44930fdc8d8SChris Lattner                 {
45030fdc8d8SChris Lattner                     result.AppendError ("invalid frame");
45130fdc8d8SChris Lattner                     result.SetStatus (eReturnStatusFailed);
45230fdc8d8SChris Lattner                     return false;
45330fdc8d8SChris Lattner                 }
45430fdc8d8SChris Lattner             }
455dda4f7b5SGreg Clayton             if (range.GetByteSize() == 0)
456dda4f7b5SGreg Clayton                 range.SetByteSize(DEFAULT_DISASM_BYTE_SIZE);
45730fdc8d8SChris Lattner 
458a7015092SGreg Clayton             if (Disassembler::Disassemble (m_interpreter.GetDebugger(),
45932e0a750SGreg Clayton                                            m_options.arch,
4601080edbcSGreg Clayton                                            plugin_name,
461dda4f7b5SGreg Clayton                                            exe_ctx,
462dda4f7b5SGreg Clayton                                            range,
46337023b06SJim Ingham                                            m_options.num_instructions,
464dda4f7b5SGreg Clayton                                            m_options.show_mixed ? m_options.num_lines_context : 0,
4651da6f9d7SGreg Clayton                                            options,
466dda4f7b5SGreg Clayton                                            result.GetOutputStream()))
46730fdc8d8SChris Lattner             {
468dda4f7b5SGreg Clayton                 result.SetStatus (eReturnStatusSuccessFinishResult);
46930fdc8d8SChris Lattner             }
47030fdc8d8SChris Lattner             else
47130fdc8d8SChris Lattner             {
47232e0a750SGreg Clayton                 result.AppendErrorWithFormat ("Failed to disassemble memory at 0x%8.8llx.\n", m_options.start_addr);
47330fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusFailed);
47430fdc8d8SChris Lattner             }
47530fdc8d8SChris Lattner         }
47637023b06SJim Ingham     }
47730fdc8d8SChris Lattner 
47830fdc8d8SChris Lattner     return result.Succeeded();
47930fdc8d8SChris Lattner }
480