180814287SRaphael Isemann //===-- CommandObjectDisassemble.cpp --------------------------------------===// 230fdc8d8SChris Lattner // 32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information. 52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 630fdc8d8SChris Lattner // 730fdc8d8SChris Lattner //===----------------------------------------------------------------------===// 830fdc8d8SChris Lattner 926cac3afSEugene Zelenko #include "CommandObjectDisassemble.h" 1030fdc8d8SChris Lattner #include "lldb/Core/AddressRange.h" 111f746071SGreg Clayton #include "lldb/Core/Disassembler.h" 12801237a2SJason Molenda #include "lldb/Core/Module.h" 133eb2b44dSZachary Turner #include "lldb/Host/OptionParser.h" 1430fdc8d8SChris Lattner #include "lldb/Interpreter/CommandInterpreter.h" 1530fdc8d8SChris Lattner #include "lldb/Interpreter/CommandReturnObject.h" 1647cbf4a0SPavel Labath #include "lldb/Interpreter/OptionArgParser.h" 1740af72e1SJim Ingham #include "lldb/Interpreter/Options.h" 181f746071SGreg Clayton #include "lldb/Symbol/Function.h" 1930fdc8d8SChris Lattner #include "lldb/Symbol/Symbol.h" 20d5944cd1SGreg Clayton #include "lldb/Target/SectionLoadList.h" 21b57e4a1bSJason Molenda #include "lldb/Target/StackFrame.h" 2230fdc8d8SChris Lattner #include "lldb/Target/Target.h" 2330fdc8d8SChris Lattner 2430fdc8d8SChris Lattner #define DEFAULT_DISASM_BYTE_SIZE 32 2537023b06SJim Ingham #define DEFAULT_DISASM_NUM_INS 4 2630fdc8d8SChris Lattner 2730fdc8d8SChris Lattner using namespace lldb; 2830fdc8d8SChris Lattner using namespace lldb_private; 2930fdc8d8SChris Lattner 30f1883637SRaphael Isemann #define LLDB_OPTIONS_disassemble 31f1883637SRaphael Isemann #include "CommandOptions.inc" 321f0f5b5bSZachary Turner 33b9c1b51eSKate Stone CommandObjectDisassemble::CommandOptions::CommandOptions() 34b9c1b51eSKate Stone : Options(), num_lines_context(0), num_instructions(0), func_name(), 35b9c1b51eSKate Stone current_function(false), start_addr(), end_addr(), at_pc(false), 36b9c1b51eSKate Stone frame_line(false), plugin_name(), flavor_string(), arch(), 37b9c1b51eSKate Stone some_location_specified(false), symbol_containing_addr() { 38e1cfbc79STodd Fiala OptionParsingStarting(nullptr); 3930fdc8d8SChris Lattner } 4030fdc8d8SChris Lattner 4126cac3afSEugene Zelenko CommandObjectDisassemble::CommandOptions::~CommandOptions() = default; 4230fdc8d8SChris Lattner 4397206d57SZachary Turner Status CommandObjectDisassemble::CommandOptions::SetOptionValue( 44fe11483bSZachary Turner uint32_t option_idx, llvm::StringRef option_arg, 45b9c1b51eSKate Stone ExecutionContext *execution_context) { 4697206d57SZachary Turner Status error; 4730fdc8d8SChris Lattner 483bcdfc0eSGreg Clayton const int short_option = m_getopt_table[option_idx].val; 4930fdc8d8SChris Lattner 50b9c1b51eSKate Stone switch (short_option) { 5130fdc8d8SChris Lattner case 'm': 5230fdc8d8SChris Lattner show_mixed = true; 5330fdc8d8SChris Lattner break; 5430fdc8d8SChris Lattner 55357132ebSGreg Clayton case 'C': 56fe11483bSZachary Turner if (option_arg.getAsInteger(0, num_lines_context)) 57b9c1b51eSKate Stone error.SetErrorStringWithFormat("invalid num context lines string: \"%s\"", 58fe11483bSZachary Turner option_arg.str().c_str()); 5937023b06SJim Ingham break; 6037023b06SJim Ingham 6130fdc8d8SChris Lattner case 'c': 62fe11483bSZachary Turner if (option_arg.getAsInteger(0, num_instructions)) 63b9c1b51eSKate Stone error.SetErrorStringWithFormat( 64fe11483bSZachary Turner "invalid num of instructions string: \"%s\"", 65fe11483bSZachary Turner option_arg.str().c_str()); 6630fdc8d8SChris Lattner break; 6730fdc8d8SChris Lattner 6830fdc8d8SChris Lattner case 'b': 6930fdc8d8SChris Lattner show_bytes = true; 7030fdc8d8SChris Lattner break; 7130fdc8d8SChris Lattner 72b9c1b51eSKate Stone case 's': { 7347cbf4a0SPavel Labath start_addr = OptionArgParser::ToAddress(execution_context, option_arg, 74e1cfbc79STodd Fiala LLDB_INVALID_ADDRESS, &error); 75b9d5df58SGreg Clayton if (start_addr != LLDB_INVALID_ADDRESS) 763555b5d7SJim Ingham some_location_specified = true; 77b9c1b51eSKate Stone } break; 78b9c1b51eSKate Stone case 'e': { 7947cbf4a0SPavel Labath end_addr = OptionArgParser::ToAddress(execution_context, option_arg, 80e1cfbc79STodd Fiala LLDB_INVALID_ADDRESS, &error); 81b9d5df58SGreg Clayton if (end_addr != LLDB_INVALID_ADDRESS) 823555b5d7SJim Ingham some_location_specified = true; 83b9c1b51eSKate Stone } break; 8426cac3afSEugene Zelenko 8530fdc8d8SChris Lattner case 'n': 86*adcd0268SBenjamin Kramer func_name.assign(std::string(option_arg)); 873555b5d7SJim Ingham some_location_specified = true; 8830fdc8d8SChris Lattner break; 8930fdc8d8SChris Lattner 9037023b06SJim Ingham case 'p': 9132e0a750SGreg Clayton at_pc = true; 923555b5d7SJim Ingham some_location_specified = true; 9332e0a750SGreg Clayton break; 9432e0a750SGreg Clayton 9532e0a750SGreg Clayton case 'l': 9632e0a750SGreg Clayton frame_line = true; 9705097246SAdrian Prantl // Disassemble the current source line kind of implies showing mixed source 9805097246SAdrian Prantl // code context. 9932e0a750SGreg Clayton show_mixed = true; 1003555b5d7SJim Ingham some_location_specified = true; 10137023b06SJim Ingham break; 10237023b06SJim Ingham 1031080edbcSGreg Clayton case 'P': 104*adcd0268SBenjamin Kramer plugin_name.assign(std::string(option_arg)); 1051080edbcSGreg Clayton break; 1061080edbcSGreg Clayton 107b9c1b51eSKate Stone case 'F': { 108b9c1b51eSKate Stone TargetSP target_sp = 109b9c1b51eSKate Stone execution_context ? execution_context->GetTargetSP() : TargetSP(); 110b9c1b51eSKate Stone if (target_sp && (target_sp->GetArchitecture().GetTriple().getArch() == 111b9c1b51eSKate Stone llvm::Triple::x86 || 112b9c1b51eSKate Stone target_sp->GetArchitecture().GetTriple().getArch() == 113b9c1b51eSKate Stone llvm::Triple::x86_64)) { 114*adcd0268SBenjamin Kramer flavor_string.assign(std::string(option_arg)); 115b9c1b51eSKate Stone } else 116b9c1b51eSKate Stone error.SetErrorStringWithFormat("Disassembler flavors are currently only " 117b9c1b51eSKate Stone "supported for x86 and x86_64 targets."); 1180f063ba6SJim Ingham break; 1190f063ba6SJim Ingham } 12026cac3afSEugene Zelenko 12130fdc8d8SChris Lattner case 'r': 12230fdc8d8SChris Lattner raw = true; 12330fdc8d8SChris Lattner break; 12430fdc8d8SChris Lattner 1258ceb8ba2SJohnny Chen case 'f': 1261fb2e7dfSGreg Clayton current_function = true; 1273555b5d7SJim Ingham some_location_specified = true; 1288ceb8ba2SJohnny Chen break; 1298ceb8ba2SJohnny Chen 130801237a2SJason Molenda case 'A': 131b9c1b51eSKate Stone if (execution_context) { 132be73fd6bSTatyana Krasnukha const auto &target_sp = execution_context->GetTargetSP(); 133be73fd6bSTatyana Krasnukha auto platform_ptr = target_sp ? target_sp->GetPlatform().get() : nullptr; 134be73fd6bSTatyana Krasnukha arch = Platform::GetAugmentedArchSpec(platform_ptr, option_arg); 135e1cfbc79STodd Fiala } 136357132ebSGreg Clayton break; 137357132ebSGreg Clayton 138b9c1b51eSKate Stone case 'a': { 13947cbf4a0SPavel Labath symbol_containing_addr = OptionArgParser::ToAddress( 140b9c1b51eSKate Stone execution_context, option_arg, LLDB_INVALID_ADDRESS, &error); 141b9c1b51eSKate Stone if (symbol_containing_addr != LLDB_INVALID_ADDRESS) { 142801237a2SJason Molenda some_location_specified = true; 143801237a2SJason Molenda } 144b9c1b51eSKate Stone } break; 145801237a2SJason Molenda 14630fdc8d8SChris Lattner default: 14736162014SRaphael Isemann llvm_unreachable("Unimplemented option"); 14830fdc8d8SChris Lattner } 14930fdc8d8SChris Lattner 15030fdc8d8SChris Lattner return error; 15130fdc8d8SChris Lattner } 15230fdc8d8SChris Lattner 153b9c1b51eSKate Stone void CommandObjectDisassemble::CommandOptions::OptionParsingStarting( 154b9c1b51eSKate Stone ExecutionContext *execution_context) { 15530fdc8d8SChris Lattner show_mixed = false; 15630fdc8d8SChris Lattner show_bytes = false; 15730fdc8d8SChris Lattner num_lines_context = 0; 15837023b06SJim Ingham num_instructions = 0; 15932e0a750SGreg Clayton func_name.clear(); 1601fb2e7dfSGreg Clayton current_function = false; 16132e0a750SGreg Clayton at_pc = false; 16232e0a750SGreg Clayton frame_line = false; 16332e0a750SGreg Clayton start_addr = LLDB_INVALID_ADDRESS; 16432e0a750SGreg Clayton end_addr = LLDB_INVALID_ADDRESS; 165801237a2SJason Molenda symbol_containing_addr = LLDB_INVALID_ADDRESS; 166a68c1a21SSean Callanan raw = false; 16732e0a750SGreg Clayton plugin_name.clear(); 1680f063ba6SJim Ingham 169e1cfbc79STodd Fiala Target *target = 170e1cfbc79STodd Fiala execution_context ? execution_context->GetTargetPtr() : nullptr; 1710f063ba6SJim Ingham 172b9c1b51eSKate Stone // This is a hack till we get the ability to specify features based on 17305097246SAdrian Prantl // architecture. For now GetDisassemblyFlavor is really only valid for x86 17405097246SAdrian Prantl // (and for the llvm assembler plugin, but I'm papering over that since that 17505097246SAdrian Prantl // is the only disassembler plugin we have... 176b9c1b51eSKate Stone if (target) { 177b9c1b51eSKate Stone if (target->GetArchitecture().GetTriple().getArch() == llvm::Triple::x86 || 178b9c1b51eSKate Stone target->GetArchitecture().GetTriple().getArch() == 179b9c1b51eSKate Stone llvm::Triple::x86_64) { 1800f063ba6SJim Ingham flavor_string.assign(target->GetDisassemblyFlavor()); 181b9c1b51eSKate Stone } else 1820f063ba6SJim Ingham flavor_string.assign("default"); 1830f063ba6SJim Ingham 184b9c1b51eSKate Stone } else 1850f063ba6SJim Ingham flavor_string.assign("default"); 1860f063ba6SJim Ingham 18732e0a750SGreg Clayton arch.Clear(); 1883555b5d7SJim Ingham some_location_specified = false; 1893555b5d7SJim Ingham } 1903555b5d7SJim Ingham 19197206d57SZachary Turner Status CommandObjectDisassemble::CommandOptions::OptionParsingFinished( 192b9c1b51eSKate Stone ExecutionContext *execution_context) { 1933555b5d7SJim Ingham if (!some_location_specified) 1941fb2e7dfSGreg Clayton current_function = true; 19597206d57SZachary Turner return Status(); 19630fdc8d8SChris Lattner } 19730fdc8d8SChris Lattner 1981f0f5b5bSZachary Turner llvm::ArrayRef<OptionDefinition> 199b9c1b51eSKate Stone CommandObjectDisassemble::CommandOptions::GetDefinitions() { 20070602439SZachary Turner return llvm::makeArrayRef(g_disassemble_options); 20130fdc8d8SChris Lattner } 20230fdc8d8SChris Lattner 20330fdc8d8SChris Lattner // CommandObjectDisassemble 20430fdc8d8SChris Lattner 205b9c1b51eSKate Stone CommandObjectDisassemble::CommandObjectDisassemble( 206b9c1b51eSKate Stone CommandInterpreter &interpreter) 207b9c1b51eSKate Stone : CommandObjectParsed( 208b9c1b51eSKate Stone interpreter, "disassemble", 209b9c1b51eSKate Stone "Disassemble specified instructions in the current target. " 2107428a18cSKate Stone "Defaults to the current function for the current thread and " 2117428a18cSKate Stone "stack frame.", 21204a4c091SRaphael Isemann "disassemble [<cmd-options>]", eCommandRequiresTarget), 213b9c1b51eSKate Stone m_options() {} 21430fdc8d8SChris Lattner 21526cac3afSEugene Zelenko CommandObjectDisassemble::~CommandObjectDisassemble() = default; 21630fdc8d8SChris Lattner 217b9c1b51eSKate Stone bool CommandObjectDisassemble::DoExecute(Args &command, 218b9c1b51eSKate Stone CommandReturnObject &result) { 21904a4c091SRaphael Isemann Target *target = &GetSelectedTarget(); 22004a4c091SRaphael Isemann 22132e0a750SGreg Clayton if (!m_options.arch.IsValid()) 22232e0a750SGreg Clayton m_options.arch = target->GetArchitecture(); 22330fdc8d8SChris Lattner 224b9c1b51eSKate Stone if (!m_options.arch.IsValid()) { 225b9c1b51eSKate Stone result.AppendError( 226b9c1b51eSKate Stone "use the --arch option or set the target architecture to disassemble"); 22730fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 22830fdc8d8SChris Lattner return false; 22930fdc8d8SChris Lattner } 23030fdc8d8SChris Lattner 2311080edbcSGreg Clayton const char *plugin_name = m_options.GetPluginName(); 2320f063ba6SJim Ingham const char *flavor_string = m_options.GetFlavorString(); 2330f063ba6SJim Ingham 234b9c1b51eSKate Stone DisassemblerSP disassembler = 235b9c1b51eSKate Stone Disassembler::FindPlugin(m_options.arch, flavor_string, plugin_name); 23630fdc8d8SChris Lattner 237b9c1b51eSKate Stone if (!disassembler) { 238b9c1b51eSKate Stone if (plugin_name) { 239b9c1b51eSKate Stone result.AppendErrorWithFormat( 240b9c1b51eSKate Stone "Unable to find Disassembler plug-in named '%s' that supports the " 241b9c1b51eSKate Stone "'%s' architecture.\n", 242b9c1b51eSKate Stone plugin_name, m_options.arch.GetArchitectureName()); 243b9c1b51eSKate Stone } else 244b9c1b51eSKate Stone result.AppendErrorWithFormat( 245b9c1b51eSKate Stone "Unable to find Disassembler plug-in for the '%s' architecture.\n", 24632e0a750SGreg Clayton m_options.arch.GetArchitectureName()); 24730fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 24830fdc8d8SChris Lattner return false; 249a925974bSAdrian Prantl } else if (flavor_string != nullptr && !disassembler->FlavorValidForArchSpec( 250a925974bSAdrian Prantl m_options.arch, flavor_string)) 251b9c1b51eSKate Stone result.AppendWarningWithFormat( 252b9c1b51eSKate Stone "invalid disassembler flavor \"%s\", using default.\n", flavor_string); 25330fdc8d8SChris Lattner 25430fdc8d8SChris Lattner result.SetStatus(eReturnStatusSuccessFinishResult); 25530fdc8d8SChris Lattner 25611eb9c64SZachary Turner if (!command.empty()) { 257b9c1b51eSKate Stone result.AppendErrorWithFormat( 258b9c1b51eSKate Stone "\"disassemble\" arguments are specified as options.\n"); 259e1cfbc79STodd Fiala const int terminal_width = 260e1cfbc79STodd Fiala GetCommandInterpreter().GetDebugger().GetTerminalWidth(); 261e1cfbc79STodd Fiala GetOptions()->GenerateOptionUsage(result.GetErrorStream(), this, 262e1cfbc79STodd Fiala terminal_width); 2638651121cSJim Ingham result.SetStatus(eReturnStatusFailed); 2648651121cSJim Ingham return false; 2658651121cSJim Ingham } 2668651121cSJim Ingham 267dda4f7b5SGreg Clayton if (m_options.show_mixed && m_options.num_lines_context == 0) 2680b4c26b2SJason Molenda m_options.num_lines_context = 2; 269dda4f7b5SGreg Clayton 270b10d72f0SGreg Clayton // Always show the PC in the disassembly 271b10d72f0SGreg Clayton uint32_t options = Disassembler::eOptionMarkPCAddress; 2721da6f9d7SGreg Clayton 273b9c1b51eSKate Stone // Mark the source line for the current PC only if we are doing mixed source 274b9c1b51eSKate Stone // and assembly 275b10d72f0SGreg Clayton if (m_options.show_mixed) 276b10d72f0SGreg Clayton options |= Disassembler::eOptionMarkPCSourceLine; 2771da6f9d7SGreg Clayton 2781da6f9d7SGreg Clayton if (m_options.show_bytes) 2791da6f9d7SGreg Clayton options |= Disassembler::eOptionShowBytes; 2801da6f9d7SGreg Clayton 2811da6f9d7SGreg Clayton if (m_options.raw) 2821da6f9d7SGreg Clayton options |= Disassembler::eOptionRawOuput; 28337023b06SJim Ingham 284b9c1b51eSKate Stone if (!m_options.func_name.empty()) { 28532e0a750SGreg Clayton ConstString name(m_options.func_name.c_str()); 286dda4f7b5SGreg Clayton 287b9c1b51eSKate Stone if (Disassembler::Disassemble( 28857179860SJonas Devlieghere GetDebugger(), m_options.arch, plugin_name, flavor_string, 28957179860SJonas Devlieghere m_exe_ctx, name, 29026cac3afSEugene Zelenko nullptr, // Module * 2910b4c26b2SJason Molenda m_options.num_instructions, m_options.show_mixed, 292b9c1b51eSKate Stone m_options.show_mixed ? m_options.num_lines_context : 0, options, 293b9c1b51eSKate Stone result.GetOutputStream())) { 294dda4f7b5SGreg Clayton result.SetStatus(eReturnStatusSuccessFinishResult); 295b9c1b51eSKate Stone } else { 296b9c1b51eSKate Stone result.AppendErrorWithFormat("Unable to find symbol with name '%s'.\n", 297b9c1b51eSKate Stone name.GetCString()); 298dda4f7b5SGreg Clayton result.SetStatus(eReturnStatusFailed); 299dda4f7b5SGreg Clayton } 300b9c1b51eSKate Stone } else { 3012f2c876eSJim Ingham std::vector<AddressRange> ranges; 30232e0a750SGreg Clayton AddressRange range; 303b57e4a1bSJason Molenda StackFrame *frame = m_exe_ctx.GetFramePtr(); 304b9c1b51eSKate Stone if (m_options.frame_line) { 305b9c1b51eSKate Stone if (frame == nullptr) { 306b9c1b51eSKate Stone result.AppendError("Cannot disassemble around the current line without " 307b9c1b51eSKate Stone "a selected frame.\n"); 3083555b5d7SJim Ingham result.SetStatus(eReturnStatusFailed); 3093555b5d7SJim Ingham return false; 3103555b5d7SJim Ingham } 311b9c1b51eSKate Stone LineEntry pc_line_entry( 312b9c1b51eSKate Stone frame->GetSymbolContext(eSymbolContextLineEntry).line_entry); 313b9c1b51eSKate Stone if (pc_line_entry.IsValid()) { 31432e0a750SGreg Clayton range = pc_line_entry.range; 315b9c1b51eSKate Stone } else { 316b9c1b51eSKate Stone m_options.at_pc = 317b9c1b51eSKate Stone true; // No line entry, so just disassemble around the current pc 31832e0a750SGreg Clayton m_options.show_mixed = false; 31932e0a750SGreg Clayton } 320b9c1b51eSKate Stone } else if (m_options.current_function) { 321b9c1b51eSKate Stone if (frame == nullptr) { 322b9c1b51eSKate Stone result.AppendError("Cannot disassemble around the current function " 323b9c1b51eSKate Stone "without a selected frame.\n"); 3243555b5d7SJim Ingham result.SetStatus(eReturnStatusFailed); 3253555b5d7SJim Ingham return false; 3263555b5d7SJim Ingham } 327c14ee32dSGreg Clayton Symbol *symbol = frame->GetSymbolContext(eSymbolContextSymbol).symbol; 328b9c1b51eSKate Stone if (symbol) { 329e7612134SGreg Clayton range.GetBaseAddress() = symbol->GetAddress(); 330e7612134SGreg Clayton range.SetByteSize(symbol->GetByteSize()); 331e7612134SGreg Clayton } 3323555b5d7SJim Ingham } 33337023b06SJim Ingham 33405097246SAdrian Prantl // Did the "m_options.frame_line" find a valid range already? If so skip 33505097246SAdrian Prantl // the rest... 336b9c1b51eSKate Stone if (range.GetByteSize() == 0) { 337b9c1b51eSKate Stone if (m_options.at_pc) { 338b9c1b51eSKate Stone if (frame == nullptr) { 339b9c1b51eSKate Stone result.AppendError("Cannot disassemble around the current PC without " 340b9c1b51eSKate Stone "a selected frame.\n"); 34137023b06SJim Ingham result.SetStatus(eReturnStatusFailed); 34237023b06SJim Ingham return false; 34337023b06SJim Ingham } 344c14ee32dSGreg Clayton range.GetBaseAddress() = frame->GetFrameCodeAddress(); 345b9c1b51eSKate Stone if (m_options.num_instructions == 0) { 346b9c1b51eSKate Stone // Disassembling at the PC always disassembles some number of 347b9c1b51eSKate Stone // instructions (not the whole function). 34837023b06SJim Ingham m_options.num_instructions = DEFAULT_DISASM_NUM_INS; 34937023b06SJim Ingham } 3502f2c876eSJim Ingham ranges.push_back(range); 351b9c1b51eSKate Stone } else { 35232e0a750SGreg Clayton range.GetBaseAddress().SetOffset(m_options.start_addr); 353b9c1b51eSKate Stone if (range.GetBaseAddress().IsValid()) { 354b9c1b51eSKate Stone if (m_options.end_addr != LLDB_INVALID_ADDRESS) { 355b9c1b51eSKate Stone if (m_options.end_addr <= m_options.start_addr) { 356b9c1b51eSKate Stone result.AppendErrorWithFormat( 357b9c1b51eSKate Stone "End address before start address.\n"); 3588651121cSJim Ingham result.SetStatus(eReturnStatusFailed); 3598651121cSJim Ingham return false; 3608651121cSJim Ingham } 36132e0a750SGreg Clayton range.SetByteSize(m_options.end_addr - m_options.start_addr); 36232e0a750SGreg Clayton } 3632f4693aaSJim Ingham ranges.push_back(range); 364b9c1b51eSKate Stone } else { 365b9c1b51eSKate Stone if (m_options.symbol_containing_addr != LLDB_INVALID_ADDRESS && 366b9c1b51eSKate Stone target) { 367b9c1b51eSKate Stone if (!target->GetSectionLoadList().IsEmpty()) { 368801237a2SJason Molenda bool failed = false; 369801237a2SJason Molenda Address symbol_containing_address; 370b9c1b51eSKate Stone if (target->GetSectionLoadList().ResolveLoadAddress( 371b9c1b51eSKate Stone m_options.symbol_containing_addr, 372b9c1b51eSKate Stone symbol_containing_address)) { 373801237a2SJason Molenda ModuleSP module_sp(symbol_containing_address.GetModule()); 374801237a2SJason Molenda SymbolContext sc; 375b9c1b51eSKate Stone bool resolve_tail_call_address = true; // PC can be one past the 376b9c1b51eSKate Stone // address range of the 377b9c1b51eSKate Stone // function. 378b9c1b51eSKate Stone module_sp->ResolveSymbolContextForAddress( 379b9c1b51eSKate Stone symbol_containing_address, eSymbolContextEverything, sc, 38035729bb1SAshok Thirumurthi resolve_tail_call_address); 381b9c1b51eSKate Stone if (sc.function || sc.symbol) { 382b9c1b51eSKate Stone sc.GetAddressRange(eSymbolContextFunction | 383b9c1b51eSKate Stone eSymbolContextSymbol, 384b9c1b51eSKate Stone 0, false, range); 385b9c1b51eSKate Stone } else { 386801237a2SJason Molenda failed = true; 387801237a2SJason Molenda } 388b9c1b51eSKate Stone } else { 389801237a2SJason Molenda failed = true; 390801237a2SJason Molenda } 391b9c1b51eSKate Stone if (failed) { 392b9c1b51eSKate Stone result.AppendErrorWithFormat( 393b9c1b51eSKate Stone "Could not find function bounds for address 0x%" PRIx64 394b9c1b51eSKate Stone "\n", 395b9c1b51eSKate Stone m_options.symbol_containing_addr); 396801237a2SJason Molenda result.SetStatus(eReturnStatusFailed); 397801237a2SJason Molenda return false; 398801237a2SJason Molenda } 3992f2c876eSJim Ingham ranges.push_back(range); 400b9c1b51eSKate Stone } else { 401b9c1b51eSKate Stone for (lldb::ModuleSP module_sp : target->GetImages().Modules()) { 4022f2c876eSJim Ingham lldb::addr_t file_addr = m_options.symbol_containing_addr; 4032f2c876eSJim Ingham Address file_address; 404b9c1b51eSKate Stone if (module_sp->ResolveFileAddress(file_addr, file_address)) { 4052f2c876eSJim Ingham SymbolContext sc; 406b9c1b51eSKate Stone bool resolve_tail_call_address = true; // PC can be one past 407b9c1b51eSKate Stone // the address range of 408b9c1b51eSKate Stone // the function. 409b9c1b51eSKate Stone module_sp->ResolveSymbolContextForAddress( 410b9c1b51eSKate Stone file_address, eSymbolContextEverything, sc, 411b9c1b51eSKate Stone resolve_tail_call_address); 412b9c1b51eSKate Stone if (sc.function || sc.symbol) { 413b9c1b51eSKate Stone sc.GetAddressRange(eSymbolContextFunction | 414b9c1b51eSKate Stone eSymbolContextSymbol, 415b9c1b51eSKate Stone 0, false, range); 4162f2c876eSJim Ingham ranges.push_back(range); 417801237a2SJason Molenda } 41837023b06SJim Ingham } 41937023b06SJim Ingham } 4202f2c876eSJim Ingham } 4212f2c876eSJim Ingham } 4222f2c876eSJim Ingham } 4232f2c876eSJim Ingham } 424b9c1b51eSKate Stone } else 4252f2c876eSJim Ingham ranges.push_back(range); 4262f2c876eSJim Ingham 427b9c1b51eSKate Stone if (m_options.num_instructions != 0) { 428b9c1b51eSKate Stone if (ranges.empty()) { 42937023b06SJim Ingham // The default action is to disassemble the current frame function. 430b9c1b51eSKate Stone if (frame) { 431b9c1b51eSKate Stone SymbolContext sc(frame->GetSymbolContext(eSymbolContextFunction | 432b9c1b51eSKate Stone eSymbolContextSymbol)); 43337023b06SJim Ingham if (sc.function) 434b9c1b51eSKate Stone range.GetBaseAddress() = 435b9c1b51eSKate Stone sc.function->GetAddressRange().GetBaseAddress(); 436e7612134SGreg Clayton else if (sc.symbol && sc.symbol->ValueIsAddress()) 437e7612134SGreg Clayton range.GetBaseAddress() = sc.symbol->GetAddress(); 43837023b06SJim Ingham else 439c14ee32dSGreg Clayton range.GetBaseAddress() = frame->GetFrameCodeAddress(); 44037023b06SJim Ingham } 44137023b06SJim Ingham 442b9c1b51eSKate Stone if (!range.GetBaseAddress().IsValid()) { 44337023b06SJim Ingham result.AppendError("invalid frame"); 44437023b06SJim Ingham result.SetStatus(eReturnStatusFailed); 44537023b06SJim Ingham return false; 44637023b06SJim Ingham } 44737023b06SJim Ingham } 44837023b06SJim Ingham 4492f2c876eSJim Ingham bool print_sc_header = ranges.size() > 1; 450b9c1b51eSKate Stone for (AddressRange cur_range : ranges) { 451b9c1b51eSKate Stone if (Disassembler::Disassemble( 45257179860SJonas Devlieghere GetDebugger(), m_options.arch, plugin_name, flavor_string, 45357179860SJonas Devlieghere m_exe_ctx, cur_range.GetBaseAddress(), 4540b4c26b2SJason Molenda m_options.num_instructions, m_options.show_mixed, 455b9c1b51eSKate Stone m_options.show_mixed ? m_options.num_lines_context : 0, options, 456b9c1b51eSKate Stone result.GetOutputStream())) { 45737023b06SJim Ingham result.SetStatus(eReturnStatusSuccessFinishResult); 458b9c1b51eSKate Stone } else { 4592f2c876eSJim Ingham if (m_options.start_addr != LLDB_INVALID_ADDRESS) 460b9c1b51eSKate Stone result.AppendErrorWithFormat( 461b9c1b51eSKate Stone "Failed to disassemble memory at 0x%8.8" PRIx64 ".\n", 462b9c1b51eSKate Stone m_options.start_addr); 4632f2c876eSJim Ingham else if (m_options.symbol_containing_addr != LLDB_INVALID_ADDRESS) 464b9c1b51eSKate Stone result.AppendErrorWithFormat( 465b9c1b51eSKate Stone "Failed to disassemble memory in function at 0x%8.8" PRIx64 466b9c1b51eSKate Stone ".\n", 467b9c1b51eSKate Stone m_options.symbol_containing_addr); 46837023b06SJim Ingham result.SetStatus(eReturnStatusFailed); 46937023b06SJim Ingham } 47037023b06SJim Ingham } 4712f2c876eSJim Ingham if (print_sc_header) 4722f2c876eSJim Ingham result.AppendMessage("\n"); 473b9c1b51eSKate Stone } else { 474b9c1b51eSKate Stone if (ranges.empty()) { 4758ceb8ba2SJohnny Chen // The default action is to disassemble the current frame function. 476b9c1b51eSKate Stone if (frame) { 477b9c1b51eSKate Stone SymbolContext sc(frame->GetSymbolContext(eSymbolContextFunction | 478b9c1b51eSKate Stone eSymbolContextSymbol)); 47930fdc8d8SChris Lattner if (sc.function) 480dda4f7b5SGreg Clayton range = sc.function->GetAddressRange(); 481b9c1b51eSKate Stone else if (sc.symbol && sc.symbol->ValueIsAddress()) { 482e7612134SGreg Clayton range.GetBaseAddress() = sc.symbol->GetAddress(); 483e7612134SGreg Clayton range.SetByteSize(sc.symbol->GetByteSize()); 484b9c1b51eSKate Stone } else 485c14ee32dSGreg Clayton range.GetBaseAddress() = frame->GetFrameCodeAddress(); 486b9c1b51eSKate Stone } else { 48730fdc8d8SChris Lattner result.AppendError("invalid frame"); 48830fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 48930fdc8d8SChris Lattner return false; 49030fdc8d8SChris Lattner } 4912f2c876eSJim Ingham ranges.push_back(range); 49230fdc8d8SChris Lattner } 4932f2c876eSJim Ingham 4942f2c876eSJim Ingham bool print_sc_header = ranges.size() > 1; 495b9c1b51eSKate Stone for (AddressRange cur_range : ranges) { 4962f2c876eSJim Ingham if (cur_range.GetByteSize() == 0) 4972f2c876eSJim Ingham cur_range.SetByteSize(DEFAULT_DISASM_BYTE_SIZE); 49830fdc8d8SChris Lattner 499b9c1b51eSKate Stone if (Disassembler::Disassemble( 50057179860SJonas Devlieghere GetDebugger(), m_options.arch, plugin_name, flavor_string, 50157179860SJonas Devlieghere m_exe_ctx, cur_range, m_options.num_instructions, 5020b4c26b2SJason Molenda m_options.show_mixed, 503b9c1b51eSKate Stone m_options.show_mixed ? m_options.num_lines_context : 0, options, 504b9c1b51eSKate Stone result.GetOutputStream())) { 505dda4f7b5SGreg Clayton result.SetStatus(eReturnStatusSuccessFinishResult); 506b9c1b51eSKate Stone } else { 507b9c1b51eSKate Stone result.AppendErrorWithFormat( 508b9c1b51eSKate Stone "Failed to disassemble memory at 0x%8.8" PRIx64 ".\n", 50905229752SPavel Labath cur_range.GetBaseAddress().GetLoadAddress(target)); 51030fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 51130fdc8d8SChris Lattner } 5122f2c876eSJim Ingham if (print_sc_header) 5132f2c876eSJim Ingham result.AppendMessage("\n"); 5142f2c876eSJim Ingham } 51530fdc8d8SChris Lattner } 51637023b06SJim Ingham } 51730fdc8d8SChris Lattner 51830fdc8d8SChris Lattner return result.Succeeded(); 51930fdc8d8SChris Lattner } 520