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 // C Includes 1130fdc8d8SChris Lattner // C++ Includes 1230fdc8d8SChris Lattner // Other libraries and framework includes 1330fdc8d8SChris Lattner // Project includes 1426cac3afSEugene Zelenko #include "CommandObjectDisassemble.h" 1530fdc8d8SChris Lattner #include "lldb/Core/AddressRange.h" 161f746071SGreg Clayton #include "lldb/Core/Disassembler.h" 17801237a2SJason Molenda #include "lldb/Core/Module.h" 181f746071SGreg Clayton #include "lldb/Core/SourceManager.h" 19*3eb2b44dSZachary Turner #include "lldb/Host/OptionParser.h" 2030fdc8d8SChris Lattner #include "lldb/Interpreter/CommandCompletions.h" 2130fdc8d8SChris Lattner #include "lldb/Interpreter/CommandInterpreter.h" 2230fdc8d8SChris Lattner #include "lldb/Interpreter/CommandReturnObject.h" 2340af72e1SJim Ingham #include "lldb/Interpreter/Options.h" 241f746071SGreg Clayton #include "lldb/Symbol/Function.h" 2530fdc8d8SChris Lattner #include "lldb/Symbol/Symbol.h" 2630fdc8d8SChris Lattner #include "lldb/Target/Process.h" 27d5944cd1SGreg Clayton #include "lldb/Target/SectionLoadList.h" 28b57e4a1bSJason Molenda #include "lldb/Target/StackFrame.h" 2930fdc8d8SChris Lattner #include "lldb/Target/Target.h" 3030fdc8d8SChris Lattner 3130fdc8d8SChris Lattner #define DEFAULT_DISASM_BYTE_SIZE 32 3237023b06SJim Ingham #define DEFAULT_DISASM_NUM_INS 4 3330fdc8d8SChris Lattner 3430fdc8d8SChris Lattner using namespace lldb; 3530fdc8d8SChris Lattner using namespace lldb_private; 3630fdc8d8SChris Lattner 371f0f5b5bSZachary Turner static OptionDefinition g_disassemble_options[] = { 381f0f5b5bSZachary Turner // clang-format off 391f0f5b5bSZachary Turner { LLDB_OPT_SET_ALL, false, "bytes", 'b', OptionParser::eNoArgument, nullptr, nullptr, 0, eArgTypeNone, "Show opcode bytes when disassembling." }, 401f0f5b5bSZachary Turner { LLDB_OPT_SET_ALL, false, "context", 'C', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypeNumLines, "Number of context lines of source to show." }, 411f0f5b5bSZachary Turner { LLDB_OPT_SET_ALL, false, "mixed", 'm', OptionParser::eNoArgument, nullptr, nullptr, 0, eArgTypeNone, "Enable mixed source and assembly display." }, 421f0f5b5bSZachary Turner { LLDB_OPT_SET_ALL, false, "raw", 'r', OptionParser::eNoArgument, nullptr, nullptr, 0, eArgTypeNone, "Print raw disassembly with no symbol information." }, 431f0f5b5bSZachary Turner { LLDB_OPT_SET_ALL, false, "plugin", 'P', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypePlugin, "Name of the disassembler plugin you want to use." }, 441f0f5b5bSZachary Turner { LLDB_OPT_SET_ALL, false, "flavor", 'F', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypeDisassemblyFlavor, "Name of the disassembly flavor you want to use. " 451f0f5b5bSZachary Turner "Currently the only valid options are default, and for Intel " 461f0f5b5bSZachary Turner "architectures, att and intel." }, 471f0f5b5bSZachary Turner { LLDB_OPT_SET_ALL, false, "arch", 'A', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypeArchitecture, "Specify the architecture to use from cross disassembly." }, 481f0f5b5bSZachary Turner { LLDB_OPT_SET_1 | 491f0f5b5bSZachary Turner LLDB_OPT_SET_2, true, "start-address", 's', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypeAddressOrExpression, "Address at which to start disassembling." }, 501f0f5b5bSZachary Turner { LLDB_OPT_SET_1, false, "end-address", 'e', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypeAddressOrExpression, "Address at which to end disassembling." }, 511f0f5b5bSZachary Turner { LLDB_OPT_SET_2 | 521f0f5b5bSZachary Turner LLDB_OPT_SET_3 | 531f0f5b5bSZachary Turner LLDB_OPT_SET_4 | 541f0f5b5bSZachary Turner LLDB_OPT_SET_5, false, "count", 'c', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypeNumLines, "Number of instructions to display." }, 551f0f5b5bSZachary Turner { LLDB_OPT_SET_3, false, "name", 'n', OptionParser::eRequiredArgument, nullptr, nullptr, CommandCompletions::eSymbolCompletion, eArgTypeFunctionName, "Disassemble entire contents of the given function name." }, 561f0f5b5bSZachary Turner { LLDB_OPT_SET_4, false, "frame", 'f', OptionParser::eNoArgument, nullptr, nullptr, 0, eArgTypeNone, "Disassemble from the start of the current frame's function." }, 571f0f5b5bSZachary Turner { LLDB_OPT_SET_5, false, "pc", 'p', OptionParser::eNoArgument, nullptr, nullptr, 0, eArgTypeNone, "Disassemble around the current pc." }, 581f0f5b5bSZachary Turner { LLDB_OPT_SET_6, false, "line", 'l', OptionParser::eNoArgument, nullptr, nullptr, 0, eArgTypeNone, "Disassemble the current frame's current source line instructions if there is debug line " 591f0f5b5bSZachary Turner "table information, else disassemble around the pc." }, 601f0f5b5bSZachary Turner { LLDB_OPT_SET_7, false, "address", 'a', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypeAddressOrExpression, "Disassemble function containing this address." }, 611f0f5b5bSZachary Turner // clang-format on 621f0f5b5bSZachary Turner }; 631f0f5b5bSZachary Turner 64b9c1b51eSKate Stone CommandObjectDisassemble::CommandOptions::CommandOptions() 65b9c1b51eSKate Stone : Options(), num_lines_context(0), num_instructions(0), func_name(), 66b9c1b51eSKate Stone current_function(false), start_addr(), end_addr(), at_pc(false), 67b9c1b51eSKate Stone frame_line(false), plugin_name(), flavor_string(), arch(), 68b9c1b51eSKate Stone some_location_specified(false), symbol_containing_addr() { 69e1cfbc79STodd Fiala OptionParsingStarting(nullptr); 7030fdc8d8SChris Lattner } 7130fdc8d8SChris Lattner 7226cac3afSEugene Zelenko CommandObjectDisassemble::CommandOptions::~CommandOptions() = default; 7330fdc8d8SChris Lattner 74b9c1b51eSKate Stone Error CommandObjectDisassemble::CommandOptions::SetOptionValue( 75fe11483bSZachary Turner uint32_t option_idx, llvm::StringRef option_arg, 76b9c1b51eSKate Stone ExecutionContext *execution_context) { 7730fdc8d8SChris Lattner Error error; 7830fdc8d8SChris Lattner 793bcdfc0eSGreg Clayton const int short_option = m_getopt_table[option_idx].val; 8030fdc8d8SChris Lattner 81b9c1b51eSKate Stone switch (short_option) { 8230fdc8d8SChris Lattner case 'm': 8330fdc8d8SChris Lattner show_mixed = true; 8430fdc8d8SChris Lattner break; 8530fdc8d8SChris Lattner 86357132ebSGreg Clayton case 'C': 87fe11483bSZachary Turner if (option_arg.getAsInteger(0, num_lines_context)) 88b9c1b51eSKate Stone error.SetErrorStringWithFormat("invalid num context lines string: \"%s\"", 89fe11483bSZachary Turner option_arg.str().c_str()); 9037023b06SJim Ingham break; 9137023b06SJim Ingham 9230fdc8d8SChris Lattner case 'c': 93fe11483bSZachary Turner if (option_arg.getAsInteger(0, num_instructions)) 94b9c1b51eSKate Stone error.SetErrorStringWithFormat( 95fe11483bSZachary Turner "invalid num of instructions string: \"%s\"", 96fe11483bSZachary Turner option_arg.str().c_str()); 9730fdc8d8SChris Lattner break; 9830fdc8d8SChris Lattner 9930fdc8d8SChris Lattner case 'b': 10030fdc8d8SChris Lattner show_bytes = true; 10130fdc8d8SChris Lattner break; 10230fdc8d8SChris Lattner 103b9c1b51eSKate Stone case 's': { 104e1cfbc79STodd Fiala start_addr = Args::StringToAddress(execution_context, option_arg, 105e1cfbc79STodd Fiala LLDB_INVALID_ADDRESS, &error); 106b9d5df58SGreg Clayton if (start_addr != LLDB_INVALID_ADDRESS) 1073555b5d7SJim Ingham some_location_specified = true; 108b9c1b51eSKate Stone } break; 109b9c1b51eSKate Stone case 'e': { 110e1cfbc79STodd Fiala end_addr = Args::StringToAddress(execution_context, option_arg, 111e1cfbc79STodd Fiala LLDB_INVALID_ADDRESS, &error); 112b9d5df58SGreg Clayton if (end_addr != LLDB_INVALID_ADDRESS) 1133555b5d7SJim Ingham some_location_specified = true; 114b9c1b51eSKate Stone } break; 11526cac3afSEugene Zelenko 11630fdc8d8SChris Lattner case 'n': 11732e0a750SGreg Clayton func_name.assign(option_arg); 1183555b5d7SJim Ingham some_location_specified = true; 11930fdc8d8SChris Lattner break; 12030fdc8d8SChris Lattner 12137023b06SJim Ingham case 'p': 12232e0a750SGreg Clayton at_pc = true; 1233555b5d7SJim Ingham some_location_specified = true; 12432e0a750SGreg Clayton break; 12532e0a750SGreg Clayton 12632e0a750SGreg Clayton case 'l': 12732e0a750SGreg Clayton frame_line = true; 12832e0a750SGreg Clayton // Disassemble the current source line kind of implies showing mixed 12932e0a750SGreg Clayton // source code context. 13032e0a750SGreg Clayton show_mixed = true; 1313555b5d7SJim Ingham some_location_specified = true; 13237023b06SJim Ingham break; 13337023b06SJim Ingham 1341080edbcSGreg Clayton case 'P': 13532e0a750SGreg Clayton plugin_name.assign(option_arg); 1361080edbcSGreg Clayton break; 1371080edbcSGreg Clayton 138b9c1b51eSKate Stone case 'F': { 139b9c1b51eSKate Stone TargetSP target_sp = 140b9c1b51eSKate Stone execution_context ? execution_context->GetTargetSP() : TargetSP(); 141b9c1b51eSKate Stone if (target_sp && (target_sp->GetArchitecture().GetTriple().getArch() == 142b9c1b51eSKate Stone llvm::Triple::x86 || 143b9c1b51eSKate Stone target_sp->GetArchitecture().GetTriple().getArch() == 144b9c1b51eSKate Stone llvm::Triple::x86_64)) { 1450f063ba6SJim Ingham flavor_string.assign(option_arg); 146b9c1b51eSKate Stone } else 147b9c1b51eSKate Stone error.SetErrorStringWithFormat("Disassembler flavors are currently only " 148b9c1b51eSKate Stone "supported for x86 and x86_64 targets."); 1490f063ba6SJim Ingham break; 1500f063ba6SJim Ingham } 15126cac3afSEugene Zelenko 15230fdc8d8SChris Lattner case 'r': 15330fdc8d8SChris Lattner raw = true; 15430fdc8d8SChris Lattner break; 15530fdc8d8SChris Lattner 1568ceb8ba2SJohnny Chen case 'f': 1571fb2e7dfSGreg Clayton current_function = true; 1583555b5d7SJim Ingham some_location_specified = true; 1598ceb8ba2SJohnny Chen break; 1608ceb8ba2SJohnny Chen 161801237a2SJason Molenda case 'A': 162b9c1b51eSKate Stone if (execution_context) { 163b9c1b51eSKate Stone auto target_sp = 164b9c1b51eSKate Stone execution_context ? execution_context->GetTargetSP() : TargetSP(); 165b9c1b51eSKate Stone auto platform_sp = target_sp ? target_sp->GetPlatform() : PlatformSP(); 166e1cfbc79STodd Fiala if (!arch.SetTriple(option_arg, platform_sp.get())) 16770512317SGreg Clayton arch.SetTriple(option_arg); 168e1cfbc79STodd Fiala } 169357132ebSGreg Clayton break; 170357132ebSGreg Clayton 171b9c1b51eSKate Stone case 'a': { 172b9c1b51eSKate Stone symbol_containing_addr = Args::StringToAddress( 173b9c1b51eSKate Stone execution_context, option_arg, LLDB_INVALID_ADDRESS, &error); 174b9c1b51eSKate Stone if (symbol_containing_addr != LLDB_INVALID_ADDRESS) { 175801237a2SJason Molenda some_location_specified = true; 176801237a2SJason Molenda } 177b9c1b51eSKate Stone } break; 178801237a2SJason Molenda 17930fdc8d8SChris Lattner default: 180b9c1b51eSKate Stone error.SetErrorStringWithFormat("unrecognized short option '%c'", 181b9c1b51eSKate Stone short_option); 18230fdc8d8SChris Lattner break; 18330fdc8d8SChris Lattner } 18430fdc8d8SChris Lattner 18530fdc8d8SChris Lattner return error; 18630fdc8d8SChris Lattner } 18730fdc8d8SChris Lattner 188b9c1b51eSKate Stone void CommandObjectDisassemble::CommandOptions::OptionParsingStarting( 189b9c1b51eSKate Stone ExecutionContext *execution_context) { 19030fdc8d8SChris Lattner show_mixed = false; 19130fdc8d8SChris Lattner show_bytes = false; 19230fdc8d8SChris Lattner num_lines_context = 0; 19337023b06SJim Ingham num_instructions = 0; 19432e0a750SGreg Clayton func_name.clear(); 1951fb2e7dfSGreg Clayton current_function = false; 19632e0a750SGreg Clayton at_pc = false; 19732e0a750SGreg Clayton frame_line = false; 19832e0a750SGreg Clayton start_addr = LLDB_INVALID_ADDRESS; 19932e0a750SGreg Clayton end_addr = LLDB_INVALID_ADDRESS; 200801237a2SJason Molenda symbol_containing_addr = LLDB_INVALID_ADDRESS; 201a68c1a21SSean Callanan raw = false; 20232e0a750SGreg Clayton plugin_name.clear(); 2030f063ba6SJim Ingham 204e1cfbc79STodd Fiala Target *target = 205e1cfbc79STodd Fiala execution_context ? execution_context->GetTargetPtr() : nullptr; 2060f063ba6SJim Ingham 207b9c1b51eSKate Stone // This is a hack till we get the ability to specify features based on 208b9c1b51eSKate Stone // architecture. For now GetDisassemblyFlavor 209b9c1b51eSKate Stone // is really only valid for x86 (and for the llvm assembler plugin, but I'm 210b9c1b51eSKate Stone // papering over that since that is the 2110f063ba6SJim Ingham // only disassembler plugin we have... 212b9c1b51eSKate Stone if (target) { 213b9c1b51eSKate Stone if (target->GetArchitecture().GetTriple().getArch() == llvm::Triple::x86 || 214b9c1b51eSKate Stone target->GetArchitecture().GetTriple().getArch() == 215b9c1b51eSKate Stone llvm::Triple::x86_64) { 2160f063ba6SJim Ingham flavor_string.assign(target->GetDisassemblyFlavor()); 217b9c1b51eSKate Stone } else 2180f063ba6SJim Ingham flavor_string.assign("default"); 2190f063ba6SJim Ingham 220b9c1b51eSKate Stone } else 2210f063ba6SJim Ingham flavor_string.assign("default"); 2220f063ba6SJim Ingham 22332e0a750SGreg Clayton arch.Clear(); 2243555b5d7SJim Ingham some_location_specified = false; 2253555b5d7SJim Ingham } 2263555b5d7SJim Ingham 227b9c1b51eSKate Stone Error CommandObjectDisassemble::CommandOptions::OptionParsingFinished( 228b9c1b51eSKate Stone ExecutionContext *execution_context) { 2293555b5d7SJim Ingham if (!some_location_specified) 2301fb2e7dfSGreg Clayton current_function = true; 231c1edf566SMehdi Amini return Error(); 23230fdc8d8SChris Lattner } 23330fdc8d8SChris Lattner 2341f0f5b5bSZachary Turner llvm::ArrayRef<OptionDefinition> 235b9c1b51eSKate Stone CommandObjectDisassemble::CommandOptions::GetDefinitions() { 23670602439SZachary Turner return llvm::makeArrayRef(g_disassemble_options); 23730fdc8d8SChris Lattner } 23830fdc8d8SChris Lattner 23930fdc8d8SChris Lattner //------------------------------------------------------------------------- 24030fdc8d8SChris Lattner // CommandObjectDisassemble 24130fdc8d8SChris Lattner //------------------------------------------------------------------------- 24230fdc8d8SChris Lattner 243b9c1b51eSKate Stone CommandObjectDisassemble::CommandObjectDisassemble( 244b9c1b51eSKate Stone CommandInterpreter &interpreter) 245b9c1b51eSKate Stone : CommandObjectParsed( 246b9c1b51eSKate Stone interpreter, "disassemble", 247b9c1b51eSKate Stone "Disassemble specified instructions in the current target. " 2487428a18cSKate Stone "Defaults to the current function for the current thread and " 2497428a18cSKate Stone "stack frame.", 250eb0103f2SGreg Clayton "disassemble [<cmd-options>]"), 251b9c1b51eSKate Stone m_options() {} 25230fdc8d8SChris Lattner 25326cac3afSEugene Zelenko CommandObjectDisassemble::~CommandObjectDisassemble() = default; 25430fdc8d8SChris Lattner 255b9c1b51eSKate Stone bool CommandObjectDisassemble::DoExecute(Args &command, 256b9c1b51eSKate Stone CommandReturnObject &result) { 257a7015092SGreg Clayton Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get(); 258b9c1b51eSKate Stone if (target == nullptr) { 259b9c1b51eSKate Stone result.AppendError("invalid target, create a debug target using the " 260b9c1b51eSKate Stone "'target create' command"); 26130fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 26230fdc8d8SChris Lattner return false; 26330fdc8d8SChris Lattner } 26432e0a750SGreg Clayton if (!m_options.arch.IsValid()) 26532e0a750SGreg Clayton m_options.arch = target->GetArchitecture(); 26630fdc8d8SChris Lattner 267b9c1b51eSKate Stone if (!m_options.arch.IsValid()) { 268b9c1b51eSKate Stone result.AppendError( 269b9c1b51eSKate Stone "use the --arch option or set the target architecture to disassemble"); 27030fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 27130fdc8d8SChris Lattner return false; 27230fdc8d8SChris Lattner } 27330fdc8d8SChris Lattner 2741080edbcSGreg Clayton const char *plugin_name = m_options.GetPluginName(); 2750f063ba6SJim Ingham const char *flavor_string = m_options.GetFlavorString(); 2760f063ba6SJim Ingham 277b9c1b51eSKate Stone DisassemblerSP disassembler = 278b9c1b51eSKate Stone Disassembler::FindPlugin(m_options.arch, flavor_string, plugin_name); 27930fdc8d8SChris Lattner 280b9c1b51eSKate Stone if (!disassembler) { 281b9c1b51eSKate Stone if (plugin_name) { 282b9c1b51eSKate Stone result.AppendErrorWithFormat( 283b9c1b51eSKate Stone "Unable to find Disassembler plug-in named '%s' that supports the " 284b9c1b51eSKate Stone "'%s' architecture.\n", 285b9c1b51eSKate Stone plugin_name, m_options.arch.GetArchitectureName()); 286b9c1b51eSKate Stone } else 287b9c1b51eSKate Stone result.AppendErrorWithFormat( 288b9c1b51eSKate Stone "Unable to find Disassembler plug-in for the '%s' architecture.\n", 28932e0a750SGreg Clayton m_options.arch.GetArchitectureName()); 29030fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 29130fdc8d8SChris Lattner return false; 292b9c1b51eSKate Stone } else if (flavor_string != nullptr && 293b9c1b51eSKate Stone !disassembler->FlavorValidForArchSpec(m_options.arch, 294b9c1b51eSKate Stone flavor_string)) 295b9c1b51eSKate Stone result.AppendWarningWithFormat( 296b9c1b51eSKate Stone "invalid disassembler flavor \"%s\", using default.\n", flavor_string); 29730fdc8d8SChris Lattner 29830fdc8d8SChris Lattner result.SetStatus(eReturnStatusSuccessFinishResult); 29930fdc8d8SChris Lattner 30011eb9c64SZachary Turner if (!command.empty()) { 301b9c1b51eSKate Stone result.AppendErrorWithFormat( 302b9c1b51eSKate Stone "\"disassemble\" arguments are specified as options.\n"); 303e1cfbc79STodd Fiala const int terminal_width = 304e1cfbc79STodd Fiala GetCommandInterpreter().GetDebugger().GetTerminalWidth(); 305e1cfbc79STodd Fiala GetOptions()->GenerateOptionUsage(result.GetErrorStream(), this, 306e1cfbc79STodd Fiala terminal_width); 3078651121cSJim Ingham result.SetStatus(eReturnStatusFailed); 3088651121cSJim Ingham return false; 3098651121cSJim Ingham } 3108651121cSJim Ingham 311dda4f7b5SGreg Clayton if (m_options.show_mixed && m_options.num_lines_context == 0) 3120b4c26b2SJason Molenda m_options.num_lines_context = 2; 313dda4f7b5SGreg Clayton 314b10d72f0SGreg Clayton // Always show the PC in the disassembly 315b10d72f0SGreg Clayton uint32_t options = Disassembler::eOptionMarkPCAddress; 3161da6f9d7SGreg Clayton 317b9c1b51eSKate Stone // Mark the source line for the current PC only if we are doing mixed source 318b9c1b51eSKate Stone // and assembly 319b10d72f0SGreg Clayton if (m_options.show_mixed) 320b10d72f0SGreg Clayton options |= Disassembler::eOptionMarkPCSourceLine; 3211da6f9d7SGreg Clayton 3221da6f9d7SGreg Clayton if (m_options.show_bytes) 3231da6f9d7SGreg Clayton options |= Disassembler::eOptionShowBytes; 3241da6f9d7SGreg Clayton 3251da6f9d7SGreg Clayton if (m_options.raw) 3261da6f9d7SGreg Clayton options |= Disassembler::eOptionRawOuput; 32737023b06SJim Ingham 328b9c1b51eSKate Stone if (!m_options.func_name.empty()) { 32932e0a750SGreg Clayton ConstString name(m_options.func_name.c_str()); 330dda4f7b5SGreg Clayton 331b9c1b51eSKate Stone if (Disassembler::Disassemble( 332b9c1b51eSKate Stone m_interpreter.GetDebugger(), m_options.arch, plugin_name, 333b9c1b51eSKate Stone flavor_string, m_exe_ctx, name, 33426cac3afSEugene Zelenko nullptr, // Module * 3350b4c26b2SJason Molenda m_options.num_instructions, m_options.show_mixed, 336b9c1b51eSKate Stone m_options.show_mixed ? m_options.num_lines_context : 0, options, 337b9c1b51eSKate Stone result.GetOutputStream())) { 338dda4f7b5SGreg Clayton result.SetStatus(eReturnStatusSuccessFinishResult); 339b9c1b51eSKate Stone } else { 340b9c1b51eSKate Stone result.AppendErrorWithFormat("Unable to find symbol with name '%s'.\n", 341b9c1b51eSKate Stone name.GetCString()); 342dda4f7b5SGreg Clayton result.SetStatus(eReturnStatusFailed); 343dda4f7b5SGreg Clayton } 344b9c1b51eSKate Stone } else { 3452f2c876eSJim Ingham std::vector<AddressRange> ranges; 34632e0a750SGreg Clayton AddressRange range; 347b57e4a1bSJason Molenda StackFrame *frame = m_exe_ctx.GetFramePtr(); 348b9c1b51eSKate Stone if (m_options.frame_line) { 349b9c1b51eSKate Stone if (frame == nullptr) { 350b9c1b51eSKate Stone result.AppendError("Cannot disassemble around the current line without " 351b9c1b51eSKate Stone "a selected frame.\n"); 3523555b5d7SJim Ingham result.SetStatus(eReturnStatusFailed); 3533555b5d7SJim Ingham return false; 3543555b5d7SJim Ingham } 355b9c1b51eSKate Stone LineEntry pc_line_entry( 356b9c1b51eSKate Stone frame->GetSymbolContext(eSymbolContextLineEntry).line_entry); 357b9c1b51eSKate Stone if (pc_line_entry.IsValid()) { 35832e0a750SGreg Clayton range = pc_line_entry.range; 359b9c1b51eSKate Stone } else { 360b9c1b51eSKate Stone m_options.at_pc = 361b9c1b51eSKate Stone true; // No line entry, so just disassemble around the current pc 36232e0a750SGreg Clayton m_options.show_mixed = false; 36332e0a750SGreg Clayton } 364b9c1b51eSKate Stone } else if (m_options.current_function) { 365b9c1b51eSKate Stone if (frame == nullptr) { 366b9c1b51eSKate Stone result.AppendError("Cannot disassemble around the current function " 367b9c1b51eSKate Stone "without a selected frame.\n"); 3683555b5d7SJim Ingham result.SetStatus(eReturnStatusFailed); 3693555b5d7SJim Ingham return false; 3703555b5d7SJim Ingham } 371c14ee32dSGreg Clayton Symbol *symbol = frame->GetSymbolContext(eSymbolContextSymbol).symbol; 372b9c1b51eSKate Stone if (symbol) { 373e7612134SGreg Clayton range.GetBaseAddress() = symbol->GetAddress(); 374e7612134SGreg Clayton range.SetByteSize(symbol->GetByteSize()); 375e7612134SGreg Clayton } 3763555b5d7SJim Ingham } 37737023b06SJim Ingham 37832e0a750SGreg Clayton // Did the "m_options.frame_line" find a valid range already? If so 37932e0a750SGreg Clayton // skip the rest... 380b9c1b51eSKate Stone if (range.GetByteSize() == 0) { 381b9c1b51eSKate Stone if (m_options.at_pc) { 382b9c1b51eSKate Stone if (frame == nullptr) { 383b9c1b51eSKate Stone result.AppendError("Cannot disassemble around the current PC without " 384b9c1b51eSKate Stone "a selected frame.\n"); 38537023b06SJim Ingham result.SetStatus(eReturnStatusFailed); 38637023b06SJim Ingham return false; 38737023b06SJim Ingham } 388c14ee32dSGreg Clayton range.GetBaseAddress() = frame->GetFrameCodeAddress(); 389b9c1b51eSKate Stone if (m_options.num_instructions == 0) { 390b9c1b51eSKate Stone // Disassembling at the PC always disassembles some number of 391b9c1b51eSKate Stone // instructions (not the whole function). 39237023b06SJim Ingham m_options.num_instructions = DEFAULT_DISASM_NUM_INS; 39337023b06SJim Ingham } 3942f2c876eSJim Ingham ranges.push_back(range); 395b9c1b51eSKate Stone } else { 39632e0a750SGreg Clayton range.GetBaseAddress().SetOffset(m_options.start_addr); 397b9c1b51eSKate Stone if (range.GetBaseAddress().IsValid()) { 398b9c1b51eSKate Stone if (m_options.end_addr != LLDB_INVALID_ADDRESS) { 399b9c1b51eSKate Stone if (m_options.end_addr <= m_options.start_addr) { 400b9c1b51eSKate Stone result.AppendErrorWithFormat( 401b9c1b51eSKate Stone "End address before start address.\n"); 4028651121cSJim Ingham result.SetStatus(eReturnStatusFailed); 4038651121cSJim Ingham return false; 4048651121cSJim Ingham } 40532e0a750SGreg Clayton range.SetByteSize(m_options.end_addr - m_options.start_addr); 40632e0a750SGreg Clayton } 4072f4693aaSJim Ingham ranges.push_back(range); 408b9c1b51eSKate Stone } else { 409b9c1b51eSKate Stone if (m_options.symbol_containing_addr != LLDB_INVALID_ADDRESS && 410b9c1b51eSKate Stone target) { 411b9c1b51eSKate Stone if (!target->GetSectionLoadList().IsEmpty()) { 412801237a2SJason Molenda bool failed = false; 413801237a2SJason Molenda Address symbol_containing_address; 414b9c1b51eSKate Stone if (target->GetSectionLoadList().ResolveLoadAddress( 415b9c1b51eSKate Stone m_options.symbol_containing_addr, 416b9c1b51eSKate Stone symbol_containing_address)) { 417801237a2SJason Molenda ModuleSP module_sp(symbol_containing_address.GetModule()); 418801237a2SJason Molenda SymbolContext sc; 419b9c1b51eSKate Stone bool resolve_tail_call_address = true; // PC can be one past the 420b9c1b51eSKate Stone // address range of the 421b9c1b51eSKate Stone // function. 422b9c1b51eSKate Stone module_sp->ResolveSymbolContextForAddress( 423b9c1b51eSKate Stone symbol_containing_address, eSymbolContextEverything, sc, 42435729bb1SAshok Thirumurthi resolve_tail_call_address); 425b9c1b51eSKate Stone if (sc.function || sc.symbol) { 426b9c1b51eSKate Stone sc.GetAddressRange(eSymbolContextFunction | 427b9c1b51eSKate Stone eSymbolContextSymbol, 428b9c1b51eSKate Stone 0, false, range); 429b9c1b51eSKate Stone } else { 430801237a2SJason Molenda failed = true; 431801237a2SJason Molenda } 432b9c1b51eSKate Stone } else { 433801237a2SJason Molenda failed = true; 434801237a2SJason Molenda } 435b9c1b51eSKate Stone if (failed) { 436b9c1b51eSKate Stone result.AppendErrorWithFormat( 437b9c1b51eSKate Stone "Could not find function bounds for address 0x%" PRIx64 438b9c1b51eSKate Stone "\n", 439b9c1b51eSKate Stone m_options.symbol_containing_addr); 440801237a2SJason Molenda result.SetStatus(eReturnStatusFailed); 441801237a2SJason Molenda return false; 442801237a2SJason Molenda } 4432f2c876eSJim Ingham ranges.push_back(range); 444b9c1b51eSKate Stone } else { 445b9c1b51eSKate Stone for (lldb::ModuleSP module_sp : target->GetImages().Modules()) { 4462f2c876eSJim Ingham lldb::addr_t file_addr = m_options.symbol_containing_addr; 4472f2c876eSJim Ingham Address file_address; 448b9c1b51eSKate Stone if (module_sp->ResolveFileAddress(file_addr, file_address)) { 4492f2c876eSJim Ingham SymbolContext sc; 450b9c1b51eSKate Stone bool resolve_tail_call_address = true; // PC can be one past 451b9c1b51eSKate Stone // the address range of 452b9c1b51eSKate Stone // the function. 453b9c1b51eSKate Stone module_sp->ResolveSymbolContextForAddress( 454b9c1b51eSKate Stone file_address, eSymbolContextEverything, sc, 455b9c1b51eSKate Stone resolve_tail_call_address); 456b9c1b51eSKate Stone if (sc.function || sc.symbol) { 457b9c1b51eSKate Stone sc.GetAddressRange(eSymbolContextFunction | 458b9c1b51eSKate Stone eSymbolContextSymbol, 459b9c1b51eSKate Stone 0, false, range); 4602f2c876eSJim Ingham ranges.push_back(range); 461801237a2SJason Molenda } 46237023b06SJim Ingham } 46337023b06SJim Ingham } 4642f2c876eSJim Ingham } 4652f2c876eSJim Ingham } 4662f2c876eSJim Ingham } 4672f2c876eSJim Ingham } 468b9c1b51eSKate Stone } else 4692f2c876eSJim Ingham ranges.push_back(range); 4702f2c876eSJim Ingham 471b9c1b51eSKate Stone if (m_options.num_instructions != 0) { 472b9c1b51eSKate Stone if (ranges.empty()) { 47337023b06SJim Ingham // The default action is to disassemble the current frame function. 474b9c1b51eSKate Stone if (frame) { 475b9c1b51eSKate Stone SymbolContext sc(frame->GetSymbolContext(eSymbolContextFunction | 476b9c1b51eSKate Stone eSymbolContextSymbol)); 47737023b06SJim Ingham if (sc.function) 478b9c1b51eSKate Stone range.GetBaseAddress() = 479b9c1b51eSKate Stone sc.function->GetAddressRange().GetBaseAddress(); 480e7612134SGreg Clayton else if (sc.symbol && sc.symbol->ValueIsAddress()) 481e7612134SGreg Clayton range.GetBaseAddress() = sc.symbol->GetAddress(); 48237023b06SJim Ingham else 483c14ee32dSGreg Clayton range.GetBaseAddress() = frame->GetFrameCodeAddress(); 48437023b06SJim Ingham } 48537023b06SJim Ingham 486b9c1b51eSKate Stone if (!range.GetBaseAddress().IsValid()) { 48737023b06SJim Ingham result.AppendError("invalid frame"); 48837023b06SJim Ingham result.SetStatus(eReturnStatusFailed); 48937023b06SJim Ingham return false; 49037023b06SJim Ingham } 49137023b06SJim Ingham } 49237023b06SJim Ingham 4932f2c876eSJim Ingham bool print_sc_header = ranges.size() > 1; 494b9c1b51eSKate Stone for (AddressRange cur_range : ranges) { 495b9c1b51eSKate Stone if (Disassembler::Disassemble( 496b9c1b51eSKate Stone m_interpreter.GetDebugger(), m_options.arch, plugin_name, 497b9c1b51eSKate Stone flavor_string, m_exe_ctx, cur_range.GetBaseAddress(), 4980b4c26b2SJason Molenda m_options.num_instructions, m_options.show_mixed, 499b9c1b51eSKate Stone m_options.show_mixed ? m_options.num_lines_context : 0, options, 500b9c1b51eSKate Stone result.GetOutputStream())) { 50137023b06SJim Ingham result.SetStatus(eReturnStatusSuccessFinishResult); 502b9c1b51eSKate Stone } else { 5032f2c876eSJim Ingham if (m_options.start_addr != LLDB_INVALID_ADDRESS) 504b9c1b51eSKate Stone result.AppendErrorWithFormat( 505b9c1b51eSKate Stone "Failed to disassemble memory at 0x%8.8" PRIx64 ".\n", 506b9c1b51eSKate Stone m_options.start_addr); 5072f2c876eSJim Ingham else if (m_options.symbol_containing_addr != LLDB_INVALID_ADDRESS) 508b9c1b51eSKate Stone result.AppendErrorWithFormat( 509b9c1b51eSKate Stone "Failed to disassemble memory in function at 0x%8.8" PRIx64 510b9c1b51eSKate Stone ".\n", 511b9c1b51eSKate Stone m_options.symbol_containing_addr); 51237023b06SJim Ingham result.SetStatus(eReturnStatusFailed); 51337023b06SJim Ingham } 51437023b06SJim Ingham } 5152f2c876eSJim Ingham if (print_sc_header) 5162f2c876eSJim Ingham result.AppendMessage("\n"); 517b9c1b51eSKate Stone } else { 518b9c1b51eSKate Stone if (ranges.empty()) { 5198ceb8ba2SJohnny Chen // The default action is to disassemble the current frame function. 520b9c1b51eSKate Stone if (frame) { 521b9c1b51eSKate Stone SymbolContext sc(frame->GetSymbolContext(eSymbolContextFunction | 522b9c1b51eSKate Stone eSymbolContextSymbol)); 52330fdc8d8SChris Lattner if (sc.function) 524dda4f7b5SGreg Clayton range = sc.function->GetAddressRange(); 525b9c1b51eSKate Stone else if (sc.symbol && sc.symbol->ValueIsAddress()) { 526e7612134SGreg Clayton range.GetBaseAddress() = sc.symbol->GetAddress(); 527e7612134SGreg Clayton range.SetByteSize(sc.symbol->GetByteSize()); 528b9c1b51eSKate Stone } else 529c14ee32dSGreg Clayton range.GetBaseAddress() = frame->GetFrameCodeAddress(); 530b9c1b51eSKate Stone } else { 53130fdc8d8SChris Lattner result.AppendError("invalid frame"); 53230fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 53330fdc8d8SChris Lattner return false; 53430fdc8d8SChris Lattner } 5352f2c876eSJim Ingham ranges.push_back(range); 53630fdc8d8SChris Lattner } 5372f2c876eSJim Ingham 5382f2c876eSJim Ingham bool print_sc_header = ranges.size() > 1; 539b9c1b51eSKate Stone for (AddressRange cur_range : ranges) { 5402f2c876eSJim Ingham if (cur_range.GetByteSize() == 0) 5412f2c876eSJim Ingham cur_range.SetByteSize(DEFAULT_DISASM_BYTE_SIZE); 54230fdc8d8SChris Lattner 543b9c1b51eSKate Stone if (Disassembler::Disassemble( 544b9c1b51eSKate Stone m_interpreter.GetDebugger(), m_options.arch, plugin_name, 545b9c1b51eSKate Stone flavor_string, m_exe_ctx, cur_range, m_options.num_instructions, 5460b4c26b2SJason Molenda m_options.show_mixed, 547b9c1b51eSKate Stone m_options.show_mixed ? m_options.num_lines_context : 0, options, 548b9c1b51eSKate Stone result.GetOutputStream())) { 549dda4f7b5SGreg Clayton result.SetStatus(eReturnStatusSuccessFinishResult); 550b9c1b51eSKate Stone } else { 551b9c1b51eSKate Stone result.AppendErrorWithFormat( 552b9c1b51eSKate Stone "Failed to disassemble memory at 0x%8.8" PRIx64 ".\n", 553b9c1b51eSKate Stone m_options.start_addr); 55430fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 55530fdc8d8SChris Lattner } 5562f2c876eSJim Ingham if (print_sc_header) 5572f2c876eSJim Ingham result.AppendMessage("\n"); 5582f2c876eSJim Ingham } 55930fdc8d8SChris Lattner } 56037023b06SJim Ingham } 56130fdc8d8SChris Lattner 56230fdc8d8SChris Lattner return result.Succeeded(); 56330fdc8d8SChris Lattner } 564