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)
74*86edbf41SGreg 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)
80*86edbf41SGreg 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)
93*86edbf41SGreg 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)
102*86edbf41SGreg 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':
1378b82f087SGreg Clayton             arch.SetTriple (option_arg, m_interpreter.GetPlatform (true).get());
138357132ebSGreg Clayton         break;
139357132ebSGreg Clayton 
14030fdc8d8SChris Lattner     default:
141*86edbf41SGreg Clayton         error.SetErrorStringWithFormat("unrecognized short option '%c'", short_option);
14230fdc8d8SChris Lattner         break;
14330fdc8d8SChris Lattner     }
14430fdc8d8SChris Lattner 
14530fdc8d8SChris Lattner     return error;
14630fdc8d8SChris Lattner }
14730fdc8d8SChris Lattner 
14830fdc8d8SChris Lattner void
149f6b8b581SGreg Clayton CommandObjectDisassemble::CommandOptions::OptionParsingStarting ()
15030fdc8d8SChris Lattner {
15130fdc8d8SChris Lattner     show_mixed = false;
15230fdc8d8SChris Lattner     show_bytes = false;
15330fdc8d8SChris Lattner     num_lines_context = 0;
15437023b06SJim Ingham     num_instructions = 0;
15532e0a750SGreg Clayton     func_name.clear();
1563555b5d7SJim Ingham     cur_function = false;
15732e0a750SGreg Clayton     at_pc = false;
15832e0a750SGreg Clayton     frame_line = false;
15932e0a750SGreg Clayton     start_addr = LLDB_INVALID_ADDRESS;
16032e0a750SGreg Clayton     end_addr = LLDB_INVALID_ADDRESS;
161a68c1a21SSean Callanan     raw = false;
16232e0a750SGreg Clayton     plugin_name.clear();
16332e0a750SGreg Clayton     arch.Clear();
1643555b5d7SJim Ingham     some_location_specified = false;
1653555b5d7SJim Ingham }
1663555b5d7SJim Ingham 
1673555b5d7SJim Ingham Error
1683555b5d7SJim Ingham CommandObjectDisassemble::CommandOptions::OptionParsingFinished ()
1693555b5d7SJim Ingham {
1703555b5d7SJim Ingham     if (!some_location_specified)
1713555b5d7SJim Ingham         at_pc = true;
1723555b5d7SJim Ingham     return Error();
1733555b5d7SJim Ingham 
17430fdc8d8SChris Lattner }
17530fdc8d8SChris Lattner 
176e0d378b3SGreg Clayton const OptionDefinition*
17730fdc8d8SChris Lattner CommandObjectDisassemble::CommandOptions::GetDefinitions ()
17830fdc8d8SChris Lattner {
17930fdc8d8SChris Lattner     return g_option_table;
18030fdc8d8SChris Lattner }
18130fdc8d8SChris Lattner 
182e0d378b3SGreg Clayton OptionDefinition
18330fdc8d8SChris Lattner CommandObjectDisassemble::CommandOptions::g_option_table[] =
18430fdc8d8SChris Lattner {
185deaab222SCaroline Tice { LLDB_OPT_SET_ALL  , false , "bytes",          'b', no_argument        , NULL, 0, eArgTypeNone,        "Show opcode bytes when disassembling."},
186357132ebSGreg Clayton { LLDB_OPT_SET_ALL  , false , "context",        'C', required_argument  , NULL, 0, eArgTypeNumLines,    "Number of context lines of source to show."},
187deaab222SCaroline Tice { LLDB_OPT_SET_ALL  , false , "mixed",          'm', no_argument        , NULL, 0, eArgTypeNone,        "Enable mixed source and assembly display."},
188deaab222SCaroline Tice { LLDB_OPT_SET_ALL  , false , "raw",            'r', no_argument        , NULL, 0, eArgTypeNone,        "Print raw disassembly with no symbol information."},
1891080edbcSGreg Clayton { LLDB_OPT_SET_ALL  , false , "plugin",         'P', required_argument  , NULL, 0, eArgTypePlugin,      "Name of the disassembler plugin you want to use."},
190357132ebSGreg Clayton { LLDB_OPT_SET_ALL  , false , "arch",           'a', required_argument  , NULL, 0, eArgTypeArchitecture,"Specify the architecture to use from cross disassembly."},
191357132ebSGreg Clayton { LLDB_OPT_SET_1 |
192357132ebSGreg Clayton   LLDB_OPT_SET_2    , true  , "start-address" , 's', required_argument  , NULL, 0, eArgTypeStartAddress,"Address at which to start disassembling."},
193405fe67fSCaroline Tice { LLDB_OPT_SET_1    , false , "end-address"  ,  'e', required_argument  , NULL, 0, eArgTypeEndAddress,  "Address at which to end disassembling."},
194357132ebSGreg Clayton { LLDB_OPT_SET_2 |
195357132ebSGreg Clayton   LLDB_OPT_SET_3 |
196357132ebSGreg Clayton   LLDB_OPT_SET_4 |
197357132ebSGreg Clayton   LLDB_OPT_SET_5    , false , "count",          'c', required_argument  , NULL, 0, eArgTypeNumLines,    "Number of instructions to display."},
1983555b5d7SJim Ingham { LLDB_OPT_SET_3    , false  , "name",           'n', required_argument  , NULL, CommandCompletions::eSymbolCompletion, eArgTypeFunctionName,             "Disassemble entire contents of the given function name."},
1993555b5d7SJim Ingham { LLDB_OPT_SET_4    , false  , "frame",          'f', no_argument        , NULL, 0, eArgTypeNone,        "Disassemble from the start of the current frame's function."},
2003555b5d7SJim Ingham { LLDB_OPT_SET_5    , false  , "pc",             'p', no_argument        , NULL, 0, eArgTypeNone,        "Disassemble around the current pc."},
2013555b5d7SJim 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."},
202deaab222SCaroline Tice { 0                 , false , NULL,             0,   0                  , NULL, 0, eArgTypeNone,        NULL }
20330fdc8d8SChris Lattner };
20430fdc8d8SChris Lattner 
20530fdc8d8SChris Lattner 
20630fdc8d8SChris Lattner 
20730fdc8d8SChris Lattner //-------------------------------------------------------------------------
20830fdc8d8SChris Lattner // CommandObjectDisassemble
20930fdc8d8SChris Lattner //-------------------------------------------------------------------------
21030fdc8d8SChris Lattner 
211a7015092SGreg Clayton CommandObjectDisassemble::CommandObjectDisassemble (CommandInterpreter &interpreter) :
212a7015092SGreg Clayton     CommandObject (interpreter,
213a7015092SGreg Clayton                    "disassemble",
2143f4c09c1SCaroline Tice                    "Disassemble bytes in the current function, or elsewhere in the executable program as specified by the user.",
215eb0103f2SGreg Clayton                    "disassemble [<cmd-options>]"),
216eb0103f2SGreg Clayton     m_options (interpreter)
21730fdc8d8SChris Lattner {
21830fdc8d8SChris Lattner }
21930fdc8d8SChris Lattner 
22030fdc8d8SChris Lattner CommandObjectDisassemble::~CommandObjectDisassemble()
22130fdc8d8SChris Lattner {
22230fdc8d8SChris Lattner }
22330fdc8d8SChris Lattner 
22430fdc8d8SChris Lattner bool
22530fdc8d8SChris Lattner CommandObjectDisassemble::Execute
22630fdc8d8SChris Lattner (
22730fdc8d8SChris Lattner     Args& command,
22830fdc8d8SChris Lattner     CommandReturnObject &result
22930fdc8d8SChris Lattner )
23030fdc8d8SChris Lattner {
231a7015092SGreg Clayton     Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get();
23230fdc8d8SChris Lattner     if (target == NULL)
23330fdc8d8SChris Lattner     {
234effe5c95SGreg Clayton         result.AppendError ("invalid target, create a debug target using the 'target create' command");
23530fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
23630fdc8d8SChris Lattner         return false;
23730fdc8d8SChris Lattner     }
23832e0a750SGreg Clayton     if (!m_options.arch.IsValid())
23932e0a750SGreg Clayton         m_options.arch = target->GetArchitecture();
24030fdc8d8SChris Lattner 
24132e0a750SGreg Clayton     if (!m_options.arch.IsValid())
24230fdc8d8SChris Lattner     {
243357132ebSGreg Clayton         result.AppendError ("use the --arch option or set the target architecure to disassemble");
24430fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
24530fdc8d8SChris Lattner         return false;
24630fdc8d8SChris Lattner     }
24730fdc8d8SChris Lattner 
2481080edbcSGreg Clayton     const char *plugin_name = m_options.GetPluginName ();
24932e0a750SGreg Clayton     Disassembler *disassembler = Disassembler::FindPlugin(m_options.arch, plugin_name);
25030fdc8d8SChris Lattner 
25130fdc8d8SChris Lattner     if (disassembler == NULL)
25230fdc8d8SChris Lattner     {
2531080edbcSGreg Clayton         if (plugin_name)
254357132ebSGreg Clayton             result.AppendErrorWithFormat ("Unable to find Disassembler plug-in named '%s' that supports the '%s' architecture.\n",
255357132ebSGreg Clayton                                           plugin_name,
25632e0a750SGreg Clayton                                           m_options.arch.GetArchitectureName());
2571080edbcSGreg Clayton         else
258357132ebSGreg Clayton             result.AppendErrorWithFormat ("Unable to find Disassembler plug-in for the '%s' architecture.\n",
25932e0a750SGreg Clayton                                           m_options.arch.GetArchitectureName());
26030fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
26130fdc8d8SChris Lattner         return false;
26230fdc8d8SChris Lattner     }
26330fdc8d8SChris Lattner 
26430fdc8d8SChris Lattner     result.SetStatus (eReturnStatusSuccessFinishResult);
26530fdc8d8SChris Lattner 
266dda4f7b5SGreg Clayton     if (command.GetArgumentCount() != 0)
26730fdc8d8SChris Lattner     {
268a7015092SGreg Clayton         result.AppendErrorWithFormat ("\"disassemble\" arguments are specified as options.\n");
269eb0103f2SGreg Clayton         GetOptions()->GenerateOptionUsage (result.GetErrorStream(), this);
2708651121cSJim Ingham         result.SetStatus (eReturnStatusFailed);
2718651121cSJim Ingham         return false;
2728651121cSJim Ingham     }
2738651121cSJim Ingham 
274dda4f7b5SGreg Clayton     if (m_options.show_mixed && m_options.num_lines_context == 0)
2756dbd3983SGreg Clayton         m_options.num_lines_context = 1;
276dda4f7b5SGreg Clayton 
2778b82f087SGreg Clayton     ExecutionContext exe_ctx(m_interpreter.GetExecutionContext());
278b10d72f0SGreg Clayton     // Always show the PC in the disassembly
279b10d72f0SGreg Clayton     uint32_t options = Disassembler::eOptionMarkPCAddress;
2801da6f9d7SGreg Clayton 
281b10d72f0SGreg Clayton     // Mark the source line for the current PC only if we are doing mixed source and assembly
282b10d72f0SGreg Clayton     if (m_options.show_mixed)
283b10d72f0SGreg Clayton         options |= Disassembler::eOptionMarkPCSourceLine;
2841da6f9d7SGreg Clayton 
2851da6f9d7SGreg Clayton     if (m_options.show_bytes)
2861da6f9d7SGreg Clayton         options |= Disassembler::eOptionShowBytes;
2871da6f9d7SGreg Clayton 
2881da6f9d7SGreg Clayton     if (m_options.raw)
2891da6f9d7SGreg Clayton         options |= Disassembler::eOptionRawOuput;
29037023b06SJim Ingham 
29132e0a750SGreg Clayton     if (!m_options.func_name.empty())
292dda4f7b5SGreg Clayton     {
29332e0a750SGreg Clayton         ConstString name(m_options.func_name.c_str());
294dda4f7b5SGreg Clayton 
295a7015092SGreg Clayton         if (Disassembler::Disassemble (m_interpreter.GetDebugger(),
29632e0a750SGreg Clayton                                        m_options.arch,
2971080edbcSGreg Clayton                                        plugin_name,
298dda4f7b5SGreg Clayton                                        exe_ctx,
299dda4f7b5SGreg Clayton                                        name,
300dda4f7b5SGreg Clayton                                        NULL,    // Module *
30137023b06SJim Ingham                                        m_options.num_instructions,
302dda4f7b5SGreg Clayton                                        m_options.show_mixed ? m_options.num_lines_context : 0,
3031da6f9d7SGreg Clayton                                        options,
304dda4f7b5SGreg Clayton                                        result.GetOutputStream()))
305dda4f7b5SGreg Clayton         {
306dda4f7b5SGreg Clayton             result.SetStatus (eReturnStatusSuccessFinishResult);
307dda4f7b5SGreg Clayton         }
308dda4f7b5SGreg Clayton         else
309dda4f7b5SGreg Clayton         {
310dda4f7b5SGreg Clayton             result.AppendErrorWithFormat ("Unable to find symbol with name '%s'.\n", name.GetCString());
311dda4f7b5SGreg Clayton             result.SetStatus (eReturnStatusFailed);
312dda4f7b5SGreg Clayton         }
313dda4f7b5SGreg Clayton     }
314dda4f7b5SGreg Clayton     else
315dda4f7b5SGreg Clayton     {
31632e0a750SGreg Clayton         AddressRange range;
317c14ee32dSGreg Clayton         StackFrame *frame = exe_ctx.GetFramePtr();
31832e0a750SGreg Clayton         if (m_options.frame_line)
31932e0a750SGreg Clayton         {
320c14ee32dSGreg Clayton             if (frame == NULL)
3213555b5d7SJim Ingham             {
3223555b5d7SJim Ingham                 result.AppendError ("Cannot disassemble around the current line without a selected frame.\n");
3233555b5d7SJim Ingham                 result.SetStatus (eReturnStatusFailed);
3243555b5d7SJim Ingham                 return false;
3253555b5d7SJim Ingham             }
326c14ee32dSGreg Clayton             LineEntry pc_line_entry (frame->GetSymbolContext(eSymbolContextLineEntry).line_entry);
32732e0a750SGreg Clayton             if (pc_line_entry.IsValid())
32832e0a750SGreg Clayton             {
32932e0a750SGreg Clayton                 range = pc_line_entry.range;
33032e0a750SGreg Clayton             }
33132e0a750SGreg Clayton             else
33232e0a750SGreg Clayton             {
33332e0a750SGreg Clayton                 m_options.at_pc = true; // No line entry, so just disassemble around the current pc
33432e0a750SGreg Clayton                 m_options.show_mixed = false;
33532e0a750SGreg Clayton             }
33632e0a750SGreg Clayton         }
3373555b5d7SJim Ingham         else if (m_options.cur_function)
3383555b5d7SJim Ingham         {
339c14ee32dSGreg Clayton             if (frame == NULL)
3403555b5d7SJim Ingham             {
3413555b5d7SJim Ingham                 result.AppendError ("Cannot disassemble around the current function without a selected frame.\n");
3423555b5d7SJim Ingham                 result.SetStatus (eReturnStatusFailed);
3433555b5d7SJim Ingham                 return false;
3443555b5d7SJim Ingham             }
345c14ee32dSGreg Clayton             Symbol *symbol = frame->GetSymbolContext(eSymbolContextSymbol).symbol;
3463555b5d7SJim Ingham             if (symbol)
3473555b5d7SJim Ingham                 range = symbol->GetAddressRangeRef();
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();
39837023b06SJim Ingham                     else if (sc.symbol && sc.symbol->GetAddressRangePtr())
39932e0a750SGreg Clayton                         range.GetBaseAddress() = sc.symbol->GetAddressRangePtr()->GetBaseAddress();
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();
44030fdc8d8SChris Lattner                     else if (sc.symbol && sc.symbol->GetAddressRangePtr())
441dda4f7b5SGreg Clayton                         range = *sc.symbol->GetAddressRangePtr();
44230fdc8d8SChris Lattner                     else
443c14ee32dSGreg Clayton                         range.GetBaseAddress() = frame->GetFrameCodeAddress();
44430fdc8d8SChris Lattner                 }
44530fdc8d8SChris Lattner                 else
44630fdc8d8SChris Lattner                 {
44730fdc8d8SChris Lattner                     result.AppendError ("invalid frame");
44830fdc8d8SChris Lattner                     result.SetStatus (eReturnStatusFailed);
44930fdc8d8SChris Lattner                     return false;
45030fdc8d8SChris Lattner                 }
45130fdc8d8SChris Lattner             }
452dda4f7b5SGreg Clayton             if (range.GetByteSize() == 0)
453dda4f7b5SGreg Clayton                 range.SetByteSize(DEFAULT_DISASM_BYTE_SIZE);
45430fdc8d8SChris Lattner 
455a7015092SGreg Clayton             if (Disassembler::Disassemble (m_interpreter.GetDebugger(),
45632e0a750SGreg Clayton                                            m_options.arch,
4571080edbcSGreg Clayton                                            plugin_name,
458dda4f7b5SGreg Clayton                                            exe_ctx,
459dda4f7b5SGreg Clayton                                            range,
46037023b06SJim Ingham                                            m_options.num_instructions,
461dda4f7b5SGreg Clayton                                            m_options.show_mixed ? m_options.num_lines_context : 0,
4621da6f9d7SGreg Clayton                                            options,
463dda4f7b5SGreg Clayton                                            result.GetOutputStream()))
46430fdc8d8SChris Lattner             {
465dda4f7b5SGreg Clayton                 result.SetStatus (eReturnStatusSuccessFinishResult);
46630fdc8d8SChris Lattner             }
46730fdc8d8SChris Lattner             else
46830fdc8d8SChris Lattner             {
46932e0a750SGreg Clayton                 result.AppendErrorWithFormat ("Failed to disassemble memory at 0x%8.8llx.\n", m_options.start_addr);
47030fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusFailed);
47130fdc8d8SChris Lattner             }
47230fdc8d8SChris Lattner         }
47337023b06SJim Ingham     }
47430fdc8d8SChris Lattner 
47530fdc8d8SChris Lattner     return result.Succeeded();
47630fdc8d8SChris Lattner }
477