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