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" 193eb2b44dSZachary 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" 2347cbf4a0SPavel Labath #include "lldb/Interpreter/OptionArgParser.h" 2440af72e1SJim Ingham #include "lldb/Interpreter/Options.h" 251f746071SGreg Clayton #include "lldb/Symbol/Function.h" 2630fdc8d8SChris Lattner #include "lldb/Symbol/Symbol.h" 2730fdc8d8SChris Lattner #include "lldb/Target/Process.h" 28d5944cd1SGreg Clayton #include "lldb/Target/SectionLoadList.h" 29b57e4a1bSJason Molenda #include "lldb/Target/StackFrame.h" 3030fdc8d8SChris Lattner #include "lldb/Target/Target.h" 3130fdc8d8SChris Lattner 3230fdc8d8SChris Lattner #define DEFAULT_DISASM_BYTE_SIZE 32 3337023b06SJim Ingham #define DEFAULT_DISASM_NUM_INS 4 3430fdc8d8SChris Lattner 3530fdc8d8SChris Lattner using namespace lldb; 3630fdc8d8SChris Lattner using namespace lldb_private; 3730fdc8d8SChris Lattner 381f0f5b5bSZachary Turner static OptionDefinition g_disassemble_options[] = { 391f0f5b5bSZachary Turner // clang-format off 401f0f5b5bSZachary Turner { LLDB_OPT_SET_ALL, false, "bytes", 'b', OptionParser::eNoArgument, nullptr, nullptr, 0, eArgTypeNone, "Show opcode bytes when disassembling." }, 411f0f5b5bSZachary Turner { LLDB_OPT_SET_ALL, false, "context", 'C', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypeNumLines, "Number of context lines of source to show." }, 421f0f5b5bSZachary Turner { LLDB_OPT_SET_ALL, false, "mixed", 'm', OptionParser::eNoArgument, nullptr, nullptr, 0, eArgTypeNone, "Enable mixed source and assembly display." }, 431f0f5b5bSZachary Turner { LLDB_OPT_SET_ALL, false, "raw", 'r', OptionParser::eNoArgument, nullptr, nullptr, 0, eArgTypeNone, "Print raw disassembly with no symbol information." }, 441f0f5b5bSZachary Turner { LLDB_OPT_SET_ALL, false, "plugin", 'P', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypePlugin, "Name of the disassembler plugin you want to use." }, 451f0f5b5bSZachary Turner { LLDB_OPT_SET_ALL, false, "flavor", 'F', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypeDisassemblyFlavor, "Name of the disassembly flavor you want to use. " 461f0f5b5bSZachary Turner "Currently the only valid options are default, and for Intel " 471f0f5b5bSZachary Turner "architectures, att and intel." }, 481f0f5b5bSZachary Turner { LLDB_OPT_SET_ALL, false, "arch", 'A', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypeArchitecture, "Specify the architecture to use from cross disassembly." }, 491f0f5b5bSZachary Turner { LLDB_OPT_SET_1 | 501f0f5b5bSZachary Turner LLDB_OPT_SET_2, true, "start-address", 's', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypeAddressOrExpression, "Address at which to start disassembling." }, 511f0f5b5bSZachary Turner { LLDB_OPT_SET_1, false, "end-address", 'e', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypeAddressOrExpression, "Address at which to end disassembling." }, 521f0f5b5bSZachary Turner { LLDB_OPT_SET_2 | 531f0f5b5bSZachary Turner LLDB_OPT_SET_3 | 541f0f5b5bSZachary Turner LLDB_OPT_SET_4 | 551f0f5b5bSZachary Turner LLDB_OPT_SET_5, false, "count", 'c', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypeNumLines, "Number of instructions to display." }, 561f0f5b5bSZachary Turner { LLDB_OPT_SET_3, false, "name", 'n', OptionParser::eRequiredArgument, nullptr, nullptr, CommandCompletions::eSymbolCompletion, eArgTypeFunctionName, "Disassemble entire contents of the given function name." }, 571f0f5b5bSZachary Turner { LLDB_OPT_SET_4, false, "frame", 'f', OptionParser::eNoArgument, nullptr, nullptr, 0, eArgTypeNone, "Disassemble from the start of the current frame's function." }, 581f0f5b5bSZachary Turner { LLDB_OPT_SET_5, false, "pc", 'p', OptionParser::eNoArgument, nullptr, nullptr, 0, eArgTypeNone, "Disassemble around the current pc." }, 591f0f5b5bSZachary 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 " 601f0f5b5bSZachary Turner "table information, else disassemble around the pc." }, 611f0f5b5bSZachary Turner { LLDB_OPT_SET_7, false, "address", 'a', OptionParser::eRequiredArgument, nullptr, nullptr, 0, eArgTypeAddressOrExpression, "Disassemble function containing this address." }, 621f0f5b5bSZachary Turner // clang-format on 631f0f5b5bSZachary Turner }; 641f0f5b5bSZachary Turner 65b9c1b51eSKate Stone CommandObjectDisassemble::CommandOptions::CommandOptions() 66b9c1b51eSKate Stone : Options(), num_lines_context(0), num_instructions(0), func_name(), 67b9c1b51eSKate Stone current_function(false), start_addr(), end_addr(), at_pc(false), 68b9c1b51eSKate Stone frame_line(false), plugin_name(), flavor_string(), arch(), 69b9c1b51eSKate Stone some_location_specified(false), symbol_containing_addr() { 70e1cfbc79STodd Fiala OptionParsingStarting(nullptr); 7130fdc8d8SChris Lattner } 7230fdc8d8SChris Lattner 7326cac3afSEugene Zelenko CommandObjectDisassemble::CommandOptions::~CommandOptions() = default; 7430fdc8d8SChris Lattner 7597206d57SZachary Turner Status CommandObjectDisassemble::CommandOptions::SetOptionValue( 76fe11483bSZachary Turner uint32_t option_idx, llvm::StringRef option_arg, 77b9c1b51eSKate Stone ExecutionContext *execution_context) { 7897206d57SZachary Turner Status error; 7930fdc8d8SChris Lattner 803bcdfc0eSGreg Clayton const int short_option = m_getopt_table[option_idx].val; 8130fdc8d8SChris Lattner 82b9c1b51eSKate Stone switch (short_option) { 8330fdc8d8SChris Lattner case 'm': 8430fdc8d8SChris Lattner show_mixed = true; 8530fdc8d8SChris Lattner break; 8630fdc8d8SChris Lattner 87357132ebSGreg Clayton case 'C': 88fe11483bSZachary Turner if (option_arg.getAsInteger(0, num_lines_context)) 89b9c1b51eSKate Stone error.SetErrorStringWithFormat("invalid num context lines string: \"%s\"", 90fe11483bSZachary Turner option_arg.str().c_str()); 9137023b06SJim Ingham break; 9237023b06SJim Ingham 9330fdc8d8SChris Lattner case 'c': 94fe11483bSZachary Turner if (option_arg.getAsInteger(0, num_instructions)) 95b9c1b51eSKate Stone error.SetErrorStringWithFormat( 96fe11483bSZachary Turner "invalid num of instructions string: \"%s\"", 97fe11483bSZachary Turner option_arg.str().c_str()); 9830fdc8d8SChris Lattner break; 9930fdc8d8SChris Lattner 10030fdc8d8SChris Lattner case 'b': 10130fdc8d8SChris Lattner show_bytes = true; 10230fdc8d8SChris Lattner break; 10330fdc8d8SChris Lattner 104b9c1b51eSKate Stone case 's': { 10547cbf4a0SPavel Labath start_addr = OptionArgParser::ToAddress(execution_context, option_arg, 106e1cfbc79STodd Fiala LLDB_INVALID_ADDRESS, &error); 107b9d5df58SGreg Clayton if (start_addr != LLDB_INVALID_ADDRESS) 1083555b5d7SJim Ingham some_location_specified = true; 109b9c1b51eSKate Stone } break; 110b9c1b51eSKate Stone case 'e': { 11147cbf4a0SPavel Labath end_addr = OptionArgParser::ToAddress(execution_context, option_arg, 112e1cfbc79STodd Fiala LLDB_INVALID_ADDRESS, &error); 113b9d5df58SGreg Clayton if (end_addr != LLDB_INVALID_ADDRESS) 1143555b5d7SJim Ingham some_location_specified = true; 115b9c1b51eSKate Stone } break; 11626cac3afSEugene Zelenko 11730fdc8d8SChris Lattner case 'n': 11832e0a750SGreg Clayton func_name.assign(option_arg); 1193555b5d7SJim Ingham some_location_specified = true; 12030fdc8d8SChris Lattner break; 12130fdc8d8SChris Lattner 12237023b06SJim Ingham case 'p': 12332e0a750SGreg Clayton at_pc = true; 1243555b5d7SJim Ingham some_location_specified = true; 12532e0a750SGreg Clayton break; 12632e0a750SGreg Clayton 12732e0a750SGreg Clayton case 'l': 12832e0a750SGreg Clayton frame_line = true; 12905097246SAdrian Prantl // Disassemble the current source line kind of implies showing mixed source 13005097246SAdrian Prantl // code context. 13132e0a750SGreg Clayton show_mixed = true; 1323555b5d7SJim Ingham some_location_specified = true; 13337023b06SJim Ingham break; 13437023b06SJim Ingham 1351080edbcSGreg Clayton case 'P': 13632e0a750SGreg Clayton plugin_name.assign(option_arg); 1371080edbcSGreg Clayton break; 1381080edbcSGreg Clayton 139b9c1b51eSKate Stone case 'F': { 140b9c1b51eSKate Stone TargetSP target_sp = 141b9c1b51eSKate Stone execution_context ? execution_context->GetTargetSP() : TargetSP(); 142b9c1b51eSKate Stone if (target_sp && (target_sp->GetArchitecture().GetTriple().getArch() == 143b9c1b51eSKate Stone llvm::Triple::x86 || 144b9c1b51eSKate Stone target_sp->GetArchitecture().GetTriple().getArch() == 145b9c1b51eSKate Stone llvm::Triple::x86_64)) { 1460f063ba6SJim Ingham flavor_string.assign(option_arg); 147b9c1b51eSKate Stone } else 148b9c1b51eSKate Stone error.SetErrorStringWithFormat("Disassembler flavors are currently only " 149b9c1b51eSKate Stone "supported for x86 and x86_64 targets."); 1500f063ba6SJim Ingham break; 1510f063ba6SJim Ingham } 15226cac3afSEugene Zelenko 15330fdc8d8SChris Lattner case 'r': 15430fdc8d8SChris Lattner raw = true; 15530fdc8d8SChris Lattner break; 15630fdc8d8SChris Lattner 1578ceb8ba2SJohnny Chen case 'f': 1581fb2e7dfSGreg Clayton current_function = true; 1593555b5d7SJim Ingham some_location_specified = true; 1608ceb8ba2SJohnny Chen break; 1618ceb8ba2SJohnny Chen 162801237a2SJason Molenda case 'A': 163b9c1b51eSKate Stone if (execution_context) { 164*be73fd6bSTatyana Krasnukha const auto &target_sp = execution_context->GetTargetSP(); 165*be73fd6bSTatyana Krasnukha auto platform_ptr = target_sp ? target_sp->GetPlatform().get() : nullptr; 166*be73fd6bSTatyana Krasnukha arch = Platform::GetAugmentedArchSpec(platform_ptr, option_arg); 167e1cfbc79STodd Fiala } 168357132ebSGreg Clayton break; 169357132ebSGreg Clayton 170b9c1b51eSKate Stone case 'a': { 17147cbf4a0SPavel Labath symbol_containing_addr = OptionArgParser::ToAddress( 172b9c1b51eSKate Stone execution_context, option_arg, LLDB_INVALID_ADDRESS, &error); 173b9c1b51eSKate Stone if (symbol_containing_addr != LLDB_INVALID_ADDRESS) { 174801237a2SJason Molenda some_location_specified = true; 175801237a2SJason Molenda } 176b9c1b51eSKate Stone } break; 177801237a2SJason Molenda 17830fdc8d8SChris Lattner default: 179b9c1b51eSKate Stone error.SetErrorStringWithFormat("unrecognized short option '%c'", 180b9c1b51eSKate Stone short_option); 18130fdc8d8SChris Lattner break; 18230fdc8d8SChris Lattner } 18330fdc8d8SChris Lattner 18430fdc8d8SChris Lattner return error; 18530fdc8d8SChris Lattner } 18630fdc8d8SChris Lattner 187b9c1b51eSKate Stone void CommandObjectDisassemble::CommandOptions::OptionParsingStarting( 188b9c1b51eSKate Stone ExecutionContext *execution_context) { 18930fdc8d8SChris Lattner show_mixed = false; 19030fdc8d8SChris Lattner show_bytes = false; 19130fdc8d8SChris Lattner num_lines_context = 0; 19237023b06SJim Ingham num_instructions = 0; 19332e0a750SGreg Clayton func_name.clear(); 1941fb2e7dfSGreg Clayton current_function = false; 19532e0a750SGreg Clayton at_pc = false; 19632e0a750SGreg Clayton frame_line = false; 19732e0a750SGreg Clayton start_addr = LLDB_INVALID_ADDRESS; 19832e0a750SGreg Clayton end_addr = LLDB_INVALID_ADDRESS; 199801237a2SJason Molenda symbol_containing_addr = LLDB_INVALID_ADDRESS; 200a68c1a21SSean Callanan raw = false; 20132e0a750SGreg Clayton plugin_name.clear(); 2020f063ba6SJim Ingham 203e1cfbc79STodd Fiala Target *target = 204e1cfbc79STodd Fiala execution_context ? execution_context->GetTargetPtr() : nullptr; 2050f063ba6SJim Ingham 206b9c1b51eSKate Stone // This is a hack till we get the ability to specify features based on 20705097246SAdrian Prantl // architecture. For now GetDisassemblyFlavor is really only valid for x86 20805097246SAdrian Prantl // (and for the llvm assembler plugin, but I'm papering over that since that 20905097246SAdrian Prantl // is the only disassembler plugin we have... 210b9c1b51eSKate Stone if (target) { 211b9c1b51eSKate Stone if (target->GetArchitecture().GetTriple().getArch() == llvm::Triple::x86 || 212b9c1b51eSKate Stone target->GetArchitecture().GetTriple().getArch() == 213b9c1b51eSKate Stone llvm::Triple::x86_64) { 2140f063ba6SJim Ingham flavor_string.assign(target->GetDisassemblyFlavor()); 215b9c1b51eSKate Stone } else 2160f063ba6SJim Ingham flavor_string.assign("default"); 2170f063ba6SJim Ingham 218b9c1b51eSKate Stone } else 2190f063ba6SJim Ingham flavor_string.assign("default"); 2200f063ba6SJim Ingham 22132e0a750SGreg Clayton arch.Clear(); 2223555b5d7SJim Ingham some_location_specified = false; 2233555b5d7SJim Ingham } 2243555b5d7SJim Ingham 22597206d57SZachary Turner Status CommandObjectDisassemble::CommandOptions::OptionParsingFinished( 226b9c1b51eSKate Stone ExecutionContext *execution_context) { 2273555b5d7SJim Ingham if (!some_location_specified) 2281fb2e7dfSGreg Clayton current_function = true; 22997206d57SZachary Turner return Status(); 23030fdc8d8SChris Lattner } 23130fdc8d8SChris Lattner 2321f0f5b5bSZachary Turner llvm::ArrayRef<OptionDefinition> 233b9c1b51eSKate Stone CommandObjectDisassemble::CommandOptions::GetDefinitions() { 23470602439SZachary Turner return llvm::makeArrayRef(g_disassemble_options); 23530fdc8d8SChris Lattner } 23630fdc8d8SChris Lattner 23730fdc8d8SChris Lattner //------------------------------------------------------------------------- 23830fdc8d8SChris Lattner // CommandObjectDisassemble 23930fdc8d8SChris Lattner //------------------------------------------------------------------------- 24030fdc8d8SChris Lattner 241b9c1b51eSKate Stone CommandObjectDisassemble::CommandObjectDisassemble( 242b9c1b51eSKate Stone CommandInterpreter &interpreter) 243b9c1b51eSKate Stone : CommandObjectParsed( 244b9c1b51eSKate Stone interpreter, "disassemble", 245b9c1b51eSKate Stone "Disassemble specified instructions in the current target. " 2467428a18cSKate Stone "Defaults to the current function for the current thread and " 2477428a18cSKate Stone "stack frame.", 248eb0103f2SGreg Clayton "disassemble [<cmd-options>]"), 249b9c1b51eSKate Stone m_options() {} 25030fdc8d8SChris Lattner 25126cac3afSEugene Zelenko CommandObjectDisassemble::~CommandObjectDisassemble() = default; 25230fdc8d8SChris Lattner 253b9c1b51eSKate Stone bool CommandObjectDisassemble::DoExecute(Args &command, 254b9c1b51eSKate Stone CommandReturnObject &result) { 255a7015092SGreg Clayton Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get(); 256b9c1b51eSKate Stone if (target == nullptr) { 257b9c1b51eSKate Stone result.AppendError("invalid target, create a debug target using the " 258b9c1b51eSKate Stone "'target create' command"); 25930fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 26030fdc8d8SChris Lattner return false; 26130fdc8d8SChris Lattner } 26232e0a750SGreg Clayton if (!m_options.arch.IsValid()) 26332e0a750SGreg Clayton m_options.arch = target->GetArchitecture(); 26430fdc8d8SChris Lattner 265b9c1b51eSKate Stone if (!m_options.arch.IsValid()) { 266b9c1b51eSKate Stone result.AppendError( 267b9c1b51eSKate Stone "use the --arch option or set the target architecture to disassemble"); 26830fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 26930fdc8d8SChris Lattner return false; 27030fdc8d8SChris Lattner } 27130fdc8d8SChris Lattner 2721080edbcSGreg Clayton const char *plugin_name = m_options.GetPluginName(); 2730f063ba6SJim Ingham const char *flavor_string = m_options.GetFlavorString(); 2740f063ba6SJim Ingham 275b9c1b51eSKate Stone DisassemblerSP disassembler = 276b9c1b51eSKate Stone Disassembler::FindPlugin(m_options.arch, flavor_string, plugin_name); 27730fdc8d8SChris Lattner 278b9c1b51eSKate Stone if (!disassembler) { 279b9c1b51eSKate Stone if (plugin_name) { 280b9c1b51eSKate Stone result.AppendErrorWithFormat( 281b9c1b51eSKate Stone "Unable to find Disassembler plug-in named '%s' that supports the " 282b9c1b51eSKate Stone "'%s' architecture.\n", 283b9c1b51eSKate Stone plugin_name, m_options.arch.GetArchitectureName()); 284b9c1b51eSKate Stone } else 285b9c1b51eSKate Stone result.AppendErrorWithFormat( 286b9c1b51eSKate Stone "Unable to find Disassembler plug-in for the '%s' architecture.\n", 28732e0a750SGreg Clayton m_options.arch.GetArchitectureName()); 28830fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 28930fdc8d8SChris Lattner return false; 290b9c1b51eSKate Stone } else if (flavor_string != nullptr && 291b9c1b51eSKate Stone !disassembler->FlavorValidForArchSpec(m_options.arch, 292b9c1b51eSKate Stone flavor_string)) 293b9c1b51eSKate Stone result.AppendWarningWithFormat( 294b9c1b51eSKate Stone "invalid disassembler flavor \"%s\", using default.\n", flavor_string); 29530fdc8d8SChris Lattner 29630fdc8d8SChris Lattner result.SetStatus(eReturnStatusSuccessFinishResult); 29730fdc8d8SChris Lattner 29811eb9c64SZachary Turner if (!command.empty()) { 299b9c1b51eSKate Stone result.AppendErrorWithFormat( 300b9c1b51eSKate Stone "\"disassemble\" arguments are specified as options.\n"); 301e1cfbc79STodd Fiala const int terminal_width = 302e1cfbc79STodd Fiala GetCommandInterpreter().GetDebugger().GetTerminalWidth(); 303e1cfbc79STodd Fiala GetOptions()->GenerateOptionUsage(result.GetErrorStream(), this, 304e1cfbc79STodd Fiala terminal_width); 3058651121cSJim Ingham result.SetStatus(eReturnStatusFailed); 3068651121cSJim Ingham return false; 3078651121cSJim Ingham } 3088651121cSJim Ingham 309dda4f7b5SGreg Clayton if (m_options.show_mixed && m_options.num_lines_context == 0) 3100b4c26b2SJason Molenda m_options.num_lines_context = 2; 311dda4f7b5SGreg Clayton 312b10d72f0SGreg Clayton // Always show the PC in the disassembly 313b10d72f0SGreg Clayton uint32_t options = Disassembler::eOptionMarkPCAddress; 3141da6f9d7SGreg Clayton 315b9c1b51eSKate Stone // Mark the source line for the current PC only if we are doing mixed source 316b9c1b51eSKate Stone // and assembly 317b10d72f0SGreg Clayton if (m_options.show_mixed) 318b10d72f0SGreg Clayton options |= Disassembler::eOptionMarkPCSourceLine; 3191da6f9d7SGreg Clayton 3201da6f9d7SGreg Clayton if (m_options.show_bytes) 3211da6f9d7SGreg Clayton options |= Disassembler::eOptionShowBytes; 3221da6f9d7SGreg Clayton 3231da6f9d7SGreg Clayton if (m_options.raw) 3241da6f9d7SGreg Clayton options |= Disassembler::eOptionRawOuput; 32537023b06SJim Ingham 326b9c1b51eSKate Stone if (!m_options.func_name.empty()) { 32732e0a750SGreg Clayton ConstString name(m_options.func_name.c_str()); 328dda4f7b5SGreg Clayton 329b9c1b51eSKate Stone if (Disassembler::Disassemble( 330b9c1b51eSKate Stone m_interpreter.GetDebugger(), m_options.arch, plugin_name, 331b9c1b51eSKate Stone flavor_string, m_exe_ctx, name, 33226cac3afSEugene Zelenko nullptr, // Module * 3330b4c26b2SJason Molenda m_options.num_instructions, m_options.show_mixed, 334b9c1b51eSKate Stone m_options.show_mixed ? m_options.num_lines_context : 0, options, 335b9c1b51eSKate Stone result.GetOutputStream())) { 336dda4f7b5SGreg Clayton result.SetStatus(eReturnStatusSuccessFinishResult); 337b9c1b51eSKate Stone } else { 338b9c1b51eSKate Stone result.AppendErrorWithFormat("Unable to find symbol with name '%s'.\n", 339b9c1b51eSKate Stone name.GetCString()); 340dda4f7b5SGreg Clayton result.SetStatus(eReturnStatusFailed); 341dda4f7b5SGreg Clayton } 342b9c1b51eSKate Stone } else { 3432f2c876eSJim Ingham std::vector<AddressRange> ranges; 34432e0a750SGreg Clayton AddressRange range; 345b57e4a1bSJason Molenda StackFrame *frame = m_exe_ctx.GetFramePtr(); 346b9c1b51eSKate Stone if (m_options.frame_line) { 347b9c1b51eSKate Stone if (frame == nullptr) { 348b9c1b51eSKate Stone result.AppendError("Cannot disassemble around the current line without " 349b9c1b51eSKate Stone "a selected frame.\n"); 3503555b5d7SJim Ingham result.SetStatus(eReturnStatusFailed); 3513555b5d7SJim Ingham return false; 3523555b5d7SJim Ingham } 353b9c1b51eSKate Stone LineEntry pc_line_entry( 354b9c1b51eSKate Stone frame->GetSymbolContext(eSymbolContextLineEntry).line_entry); 355b9c1b51eSKate Stone if (pc_line_entry.IsValid()) { 35632e0a750SGreg Clayton range = pc_line_entry.range; 357b9c1b51eSKate Stone } else { 358b9c1b51eSKate Stone m_options.at_pc = 359b9c1b51eSKate Stone true; // No line entry, so just disassemble around the current pc 36032e0a750SGreg Clayton m_options.show_mixed = false; 36132e0a750SGreg Clayton } 362b9c1b51eSKate Stone } else if (m_options.current_function) { 363b9c1b51eSKate Stone if (frame == nullptr) { 364b9c1b51eSKate Stone result.AppendError("Cannot disassemble around the current function " 365b9c1b51eSKate Stone "without a selected frame.\n"); 3663555b5d7SJim Ingham result.SetStatus(eReturnStatusFailed); 3673555b5d7SJim Ingham return false; 3683555b5d7SJim Ingham } 369c14ee32dSGreg Clayton Symbol *symbol = frame->GetSymbolContext(eSymbolContextSymbol).symbol; 370b9c1b51eSKate Stone if (symbol) { 371e7612134SGreg Clayton range.GetBaseAddress() = symbol->GetAddress(); 372e7612134SGreg Clayton range.SetByteSize(symbol->GetByteSize()); 373e7612134SGreg Clayton } 3743555b5d7SJim Ingham } 37537023b06SJim Ingham 37605097246SAdrian Prantl // Did the "m_options.frame_line" find a valid range already? If so skip 37705097246SAdrian Prantl // the rest... 378b9c1b51eSKate Stone if (range.GetByteSize() == 0) { 379b9c1b51eSKate Stone if (m_options.at_pc) { 380b9c1b51eSKate Stone if (frame == nullptr) { 381b9c1b51eSKate Stone result.AppendError("Cannot disassemble around the current PC without " 382b9c1b51eSKate Stone "a selected frame.\n"); 38337023b06SJim Ingham result.SetStatus(eReturnStatusFailed); 38437023b06SJim Ingham return false; 38537023b06SJim Ingham } 386c14ee32dSGreg Clayton range.GetBaseAddress() = frame->GetFrameCodeAddress(); 387b9c1b51eSKate Stone if (m_options.num_instructions == 0) { 388b9c1b51eSKate Stone // Disassembling at the PC always disassembles some number of 389b9c1b51eSKate Stone // instructions (not the whole function). 39037023b06SJim Ingham m_options.num_instructions = DEFAULT_DISASM_NUM_INS; 39137023b06SJim Ingham } 3922f2c876eSJim Ingham ranges.push_back(range); 393b9c1b51eSKate Stone } else { 39432e0a750SGreg Clayton range.GetBaseAddress().SetOffset(m_options.start_addr); 395b9c1b51eSKate Stone if (range.GetBaseAddress().IsValid()) { 396b9c1b51eSKate Stone if (m_options.end_addr != LLDB_INVALID_ADDRESS) { 397b9c1b51eSKate Stone if (m_options.end_addr <= m_options.start_addr) { 398b9c1b51eSKate Stone result.AppendErrorWithFormat( 399b9c1b51eSKate Stone "End address before start address.\n"); 4008651121cSJim Ingham result.SetStatus(eReturnStatusFailed); 4018651121cSJim Ingham return false; 4028651121cSJim Ingham } 40332e0a750SGreg Clayton range.SetByteSize(m_options.end_addr - m_options.start_addr); 40432e0a750SGreg Clayton } 4052f4693aaSJim Ingham ranges.push_back(range); 406b9c1b51eSKate Stone } else { 407b9c1b51eSKate Stone if (m_options.symbol_containing_addr != LLDB_INVALID_ADDRESS && 408b9c1b51eSKate Stone target) { 409b9c1b51eSKate Stone if (!target->GetSectionLoadList().IsEmpty()) { 410801237a2SJason Molenda bool failed = false; 411801237a2SJason Molenda Address symbol_containing_address; 412b9c1b51eSKate Stone if (target->GetSectionLoadList().ResolveLoadAddress( 413b9c1b51eSKate Stone m_options.symbol_containing_addr, 414b9c1b51eSKate Stone symbol_containing_address)) { 415801237a2SJason Molenda ModuleSP module_sp(symbol_containing_address.GetModule()); 416801237a2SJason Molenda SymbolContext sc; 417b9c1b51eSKate Stone bool resolve_tail_call_address = true; // PC can be one past the 418b9c1b51eSKate Stone // address range of the 419b9c1b51eSKate Stone // function. 420b9c1b51eSKate Stone module_sp->ResolveSymbolContextForAddress( 421b9c1b51eSKate Stone symbol_containing_address, eSymbolContextEverything, sc, 42235729bb1SAshok Thirumurthi resolve_tail_call_address); 423b9c1b51eSKate Stone if (sc.function || sc.symbol) { 424b9c1b51eSKate Stone sc.GetAddressRange(eSymbolContextFunction | 425b9c1b51eSKate Stone eSymbolContextSymbol, 426b9c1b51eSKate Stone 0, false, range); 427b9c1b51eSKate Stone } else { 428801237a2SJason Molenda failed = true; 429801237a2SJason Molenda } 430b9c1b51eSKate Stone } else { 431801237a2SJason Molenda failed = true; 432801237a2SJason Molenda } 433b9c1b51eSKate Stone if (failed) { 434b9c1b51eSKate Stone result.AppendErrorWithFormat( 435b9c1b51eSKate Stone "Could not find function bounds for address 0x%" PRIx64 436b9c1b51eSKate Stone "\n", 437b9c1b51eSKate Stone m_options.symbol_containing_addr); 438801237a2SJason Molenda result.SetStatus(eReturnStatusFailed); 439801237a2SJason Molenda return false; 440801237a2SJason Molenda } 4412f2c876eSJim Ingham ranges.push_back(range); 442b9c1b51eSKate Stone } else { 443b9c1b51eSKate Stone for (lldb::ModuleSP module_sp : target->GetImages().Modules()) { 4442f2c876eSJim Ingham lldb::addr_t file_addr = m_options.symbol_containing_addr; 4452f2c876eSJim Ingham Address file_address; 446b9c1b51eSKate Stone if (module_sp->ResolveFileAddress(file_addr, file_address)) { 4472f2c876eSJim Ingham SymbolContext sc; 448b9c1b51eSKate Stone bool resolve_tail_call_address = true; // PC can be one past 449b9c1b51eSKate Stone // the address range of 450b9c1b51eSKate Stone // the function. 451b9c1b51eSKate Stone module_sp->ResolveSymbolContextForAddress( 452b9c1b51eSKate Stone file_address, eSymbolContextEverything, sc, 453b9c1b51eSKate Stone resolve_tail_call_address); 454b9c1b51eSKate Stone if (sc.function || sc.symbol) { 455b9c1b51eSKate Stone sc.GetAddressRange(eSymbolContextFunction | 456b9c1b51eSKate Stone eSymbolContextSymbol, 457b9c1b51eSKate Stone 0, false, range); 4582f2c876eSJim Ingham ranges.push_back(range); 459801237a2SJason Molenda } 46037023b06SJim Ingham } 46137023b06SJim Ingham } 4622f2c876eSJim Ingham } 4632f2c876eSJim Ingham } 4642f2c876eSJim Ingham } 4652f2c876eSJim Ingham } 466b9c1b51eSKate Stone } else 4672f2c876eSJim Ingham ranges.push_back(range); 4682f2c876eSJim Ingham 469b9c1b51eSKate Stone if (m_options.num_instructions != 0) { 470b9c1b51eSKate Stone if (ranges.empty()) { 47137023b06SJim Ingham // The default action is to disassemble the current frame function. 472b9c1b51eSKate Stone if (frame) { 473b9c1b51eSKate Stone SymbolContext sc(frame->GetSymbolContext(eSymbolContextFunction | 474b9c1b51eSKate Stone eSymbolContextSymbol)); 47537023b06SJim Ingham if (sc.function) 476b9c1b51eSKate Stone range.GetBaseAddress() = 477b9c1b51eSKate Stone sc.function->GetAddressRange().GetBaseAddress(); 478e7612134SGreg Clayton else if (sc.symbol && sc.symbol->ValueIsAddress()) 479e7612134SGreg Clayton range.GetBaseAddress() = sc.symbol->GetAddress(); 48037023b06SJim Ingham else 481c14ee32dSGreg Clayton range.GetBaseAddress() = frame->GetFrameCodeAddress(); 48237023b06SJim Ingham } 48337023b06SJim Ingham 484b9c1b51eSKate Stone if (!range.GetBaseAddress().IsValid()) { 48537023b06SJim Ingham result.AppendError("invalid frame"); 48637023b06SJim Ingham result.SetStatus(eReturnStatusFailed); 48737023b06SJim Ingham return false; 48837023b06SJim Ingham } 48937023b06SJim Ingham } 49037023b06SJim Ingham 4912f2c876eSJim Ingham bool print_sc_header = ranges.size() > 1; 492b9c1b51eSKate Stone for (AddressRange cur_range : ranges) { 493b9c1b51eSKate Stone if (Disassembler::Disassemble( 494b9c1b51eSKate Stone m_interpreter.GetDebugger(), m_options.arch, plugin_name, 495b9c1b51eSKate Stone flavor_string, m_exe_ctx, cur_range.GetBaseAddress(), 4960b4c26b2SJason Molenda m_options.num_instructions, m_options.show_mixed, 497b9c1b51eSKate Stone m_options.show_mixed ? m_options.num_lines_context : 0, options, 498b9c1b51eSKate Stone result.GetOutputStream())) { 49937023b06SJim Ingham result.SetStatus(eReturnStatusSuccessFinishResult); 500b9c1b51eSKate Stone } else { 5012f2c876eSJim Ingham if (m_options.start_addr != LLDB_INVALID_ADDRESS) 502b9c1b51eSKate Stone result.AppendErrorWithFormat( 503b9c1b51eSKate Stone "Failed to disassemble memory at 0x%8.8" PRIx64 ".\n", 504b9c1b51eSKate Stone m_options.start_addr); 5052f2c876eSJim Ingham else if (m_options.symbol_containing_addr != LLDB_INVALID_ADDRESS) 506b9c1b51eSKate Stone result.AppendErrorWithFormat( 507b9c1b51eSKate Stone "Failed to disassemble memory in function at 0x%8.8" PRIx64 508b9c1b51eSKate Stone ".\n", 509b9c1b51eSKate Stone m_options.symbol_containing_addr); 51037023b06SJim Ingham result.SetStatus(eReturnStatusFailed); 51137023b06SJim Ingham } 51237023b06SJim Ingham } 5132f2c876eSJim Ingham if (print_sc_header) 5142f2c876eSJim Ingham result.AppendMessage("\n"); 515b9c1b51eSKate Stone } else { 516b9c1b51eSKate Stone if (ranges.empty()) { 5178ceb8ba2SJohnny Chen // The default action is to disassemble the current frame function. 518b9c1b51eSKate Stone if (frame) { 519b9c1b51eSKate Stone SymbolContext sc(frame->GetSymbolContext(eSymbolContextFunction | 520b9c1b51eSKate Stone eSymbolContextSymbol)); 52130fdc8d8SChris Lattner if (sc.function) 522dda4f7b5SGreg Clayton range = sc.function->GetAddressRange(); 523b9c1b51eSKate Stone else if (sc.symbol && sc.symbol->ValueIsAddress()) { 524e7612134SGreg Clayton range.GetBaseAddress() = sc.symbol->GetAddress(); 525e7612134SGreg Clayton range.SetByteSize(sc.symbol->GetByteSize()); 526b9c1b51eSKate Stone } else 527c14ee32dSGreg Clayton range.GetBaseAddress() = frame->GetFrameCodeAddress(); 528b9c1b51eSKate Stone } else { 52930fdc8d8SChris Lattner result.AppendError("invalid frame"); 53030fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 53130fdc8d8SChris Lattner return false; 53230fdc8d8SChris Lattner } 5332f2c876eSJim Ingham ranges.push_back(range); 53430fdc8d8SChris Lattner } 5352f2c876eSJim Ingham 5362f2c876eSJim Ingham bool print_sc_header = ranges.size() > 1; 537b9c1b51eSKate Stone for (AddressRange cur_range : ranges) { 5382f2c876eSJim Ingham if (cur_range.GetByteSize() == 0) 5392f2c876eSJim Ingham cur_range.SetByteSize(DEFAULT_DISASM_BYTE_SIZE); 54030fdc8d8SChris Lattner 541b9c1b51eSKate Stone if (Disassembler::Disassemble( 542b9c1b51eSKate Stone m_interpreter.GetDebugger(), m_options.arch, plugin_name, 543b9c1b51eSKate Stone flavor_string, m_exe_ctx, cur_range, m_options.num_instructions, 5440b4c26b2SJason Molenda m_options.show_mixed, 545b9c1b51eSKate Stone m_options.show_mixed ? m_options.num_lines_context : 0, options, 546b9c1b51eSKate Stone result.GetOutputStream())) { 547dda4f7b5SGreg Clayton result.SetStatus(eReturnStatusSuccessFinishResult); 548b9c1b51eSKate Stone } else { 549b9c1b51eSKate Stone result.AppendErrorWithFormat( 550b9c1b51eSKate Stone "Failed to disassemble memory at 0x%8.8" PRIx64 ".\n", 551b9c1b51eSKate Stone m_options.start_addr); 55230fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 55330fdc8d8SChris Lattner } 5542f2c876eSJim Ingham if (print_sc_header) 5552f2c876eSJim Ingham result.AppendMessage("\n"); 5562f2c876eSJim Ingham } 55730fdc8d8SChris Lattner } 55837023b06SJim Ingham } 55930fdc8d8SChris Lattner 56030fdc8d8SChris Lattner return result.Succeeded(); 56130fdc8d8SChris Lattner } 562