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': 86adcd0268SBenjamin 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': 104adcd0268SBenjamin 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)) { 114adcd0268SBenjamin 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 284c6a38957SPavel Labath std::vector<AddressRange> ranges; 285b9c1b51eSKate Stone if (!m_options.func_name.empty()) { 28632e0a750SGreg Clayton ConstString name(m_options.func_name.c_str()); 287c6a38957SPavel Labath const bool include_symbols = true; 288c6a38957SPavel Labath const bool include_inlines = true; 289dda4f7b5SGreg Clayton 290c6a38957SPavel Labath // Find functions matching the given name. 291c6a38957SPavel Labath SymbolContextList sc_list; 292c6a38957SPavel Labath target->GetImages().FindFunctions( 293c6a38957SPavel Labath name, eFunctionNameTypeAuto, include_symbols, include_inlines, sc_list); 294c6a38957SPavel Labath 295c6a38957SPavel Labath AddressRange range; 296c6a38957SPavel Labath const uint32_t scope = 297c6a38957SPavel Labath eSymbolContextBlock | eSymbolContextFunction | eSymbolContextSymbol; 298c6a38957SPavel Labath const bool use_inline_block_range = true; 299c6a38957SPavel Labath for (SymbolContext sc : sc_list.SymbolContexts()) { 300c6a38957SPavel Labath for (uint32_t range_idx = 0; 301c6a38957SPavel Labath sc.GetAddressRange(scope, range_idx, use_inline_block_range, range); 302c6a38957SPavel Labath ++range_idx) { 303c6a38957SPavel Labath ranges.push_back(range); 304c6a38957SPavel Labath } 305c6a38957SPavel Labath } 306c6a38957SPavel Labath if (ranges.empty()) { 307b9c1b51eSKate Stone result.AppendErrorWithFormat("Unable to find symbol with name '%s'.\n", 308b9c1b51eSKate Stone name.GetCString()); 309dda4f7b5SGreg Clayton result.SetStatus(eReturnStatusFailed); 310c6a38957SPavel Labath return result.Succeeded(); 311dda4f7b5SGreg Clayton } 312b9c1b51eSKate Stone } else { 31332e0a750SGreg Clayton AddressRange range; 314b57e4a1bSJason Molenda StackFrame *frame = m_exe_ctx.GetFramePtr(); 315b9c1b51eSKate Stone if (m_options.frame_line) { 316b9c1b51eSKate Stone if (frame == nullptr) { 317b9c1b51eSKate Stone result.AppendError("Cannot disassemble around the current line without " 318b9c1b51eSKate Stone "a selected frame.\n"); 3193555b5d7SJim Ingham result.SetStatus(eReturnStatusFailed); 3203555b5d7SJim Ingham return false; 3213555b5d7SJim Ingham } 322b9c1b51eSKate Stone LineEntry pc_line_entry( 323b9c1b51eSKate Stone frame->GetSymbolContext(eSymbolContextLineEntry).line_entry); 324b9c1b51eSKate Stone if (pc_line_entry.IsValid()) { 32532e0a750SGreg Clayton range = pc_line_entry.range; 326b9c1b51eSKate Stone } else { 327b9c1b51eSKate Stone m_options.at_pc = 328b9c1b51eSKate Stone true; // No line entry, so just disassemble around the current pc 32932e0a750SGreg Clayton m_options.show_mixed = false; 33032e0a750SGreg Clayton } 331b9c1b51eSKate Stone } else if (m_options.current_function) { 332b9c1b51eSKate Stone if (frame == nullptr) { 333b9c1b51eSKate Stone result.AppendError("Cannot disassemble around the current function " 334b9c1b51eSKate Stone "without a selected frame.\n"); 3353555b5d7SJim Ingham result.SetStatus(eReturnStatusFailed); 3363555b5d7SJim Ingham return false; 3373555b5d7SJim Ingham } 338c14ee32dSGreg Clayton Symbol *symbol = frame->GetSymbolContext(eSymbolContextSymbol).symbol; 339b9c1b51eSKate Stone if (symbol) { 340e7612134SGreg Clayton range.GetBaseAddress() = symbol->GetAddress(); 341e7612134SGreg Clayton range.SetByteSize(symbol->GetByteSize()); 342e7612134SGreg Clayton } 3433555b5d7SJim Ingham } 34437023b06SJim Ingham 34505097246SAdrian Prantl // Did the "m_options.frame_line" find a valid range already? If so skip 34605097246SAdrian Prantl // the rest... 347b9c1b51eSKate Stone if (range.GetByteSize() == 0) { 348b9c1b51eSKate Stone if (m_options.at_pc) { 349b9c1b51eSKate Stone if (frame == nullptr) { 350b9c1b51eSKate Stone result.AppendError("Cannot disassemble around the current PC without " 351b9c1b51eSKate Stone "a selected frame.\n"); 35237023b06SJim Ingham result.SetStatus(eReturnStatusFailed); 35337023b06SJim Ingham return false; 35437023b06SJim Ingham } 355c14ee32dSGreg Clayton range.GetBaseAddress() = frame->GetFrameCodeAddress(); 356b9c1b51eSKate Stone if (m_options.num_instructions == 0) { 357b9c1b51eSKate Stone // Disassembling at the PC always disassembles some number of 358b9c1b51eSKate Stone // instructions (not the whole function). 35937023b06SJim Ingham m_options.num_instructions = DEFAULT_DISASM_NUM_INS; 36037023b06SJim Ingham } 3612f2c876eSJim Ingham ranges.push_back(range); 362b9c1b51eSKate Stone } else { 36332e0a750SGreg Clayton range.GetBaseAddress().SetOffset(m_options.start_addr); 364b9c1b51eSKate Stone if (range.GetBaseAddress().IsValid()) { 365b9c1b51eSKate Stone if (m_options.end_addr != LLDB_INVALID_ADDRESS) { 366b9c1b51eSKate Stone if (m_options.end_addr <= m_options.start_addr) { 367b9c1b51eSKate Stone result.AppendErrorWithFormat( 368b9c1b51eSKate Stone "End address before start address.\n"); 3698651121cSJim Ingham result.SetStatus(eReturnStatusFailed); 3708651121cSJim Ingham return false; 3718651121cSJim Ingham } 37232e0a750SGreg Clayton range.SetByteSize(m_options.end_addr - m_options.start_addr); 37332e0a750SGreg Clayton } 3742f4693aaSJim Ingham ranges.push_back(range); 375b9c1b51eSKate Stone } else { 376b9c1b51eSKate Stone if (m_options.symbol_containing_addr != LLDB_INVALID_ADDRESS && 377b9c1b51eSKate Stone target) { 378b9c1b51eSKate Stone if (!target->GetSectionLoadList().IsEmpty()) { 379801237a2SJason Molenda bool failed = false; 380801237a2SJason Molenda Address symbol_containing_address; 381b9c1b51eSKate Stone if (target->GetSectionLoadList().ResolveLoadAddress( 382b9c1b51eSKate Stone m_options.symbol_containing_addr, 383b9c1b51eSKate Stone symbol_containing_address)) { 384801237a2SJason Molenda ModuleSP module_sp(symbol_containing_address.GetModule()); 385801237a2SJason Molenda SymbolContext sc; 386b9c1b51eSKate Stone bool resolve_tail_call_address = true; // PC can be one past the 387b9c1b51eSKate Stone // address range of the 388b9c1b51eSKate Stone // function. 389b9c1b51eSKate Stone module_sp->ResolveSymbolContextForAddress( 390b9c1b51eSKate Stone symbol_containing_address, eSymbolContextEverything, sc, 39135729bb1SAshok Thirumurthi resolve_tail_call_address); 392b9c1b51eSKate Stone if (sc.function || sc.symbol) { 393b9c1b51eSKate Stone sc.GetAddressRange(eSymbolContextFunction | 394b9c1b51eSKate Stone eSymbolContextSymbol, 395b9c1b51eSKate Stone 0, false, range); 396b9c1b51eSKate Stone } else { 397801237a2SJason Molenda failed = true; 398801237a2SJason Molenda } 399b9c1b51eSKate Stone } else { 400801237a2SJason Molenda failed = true; 401801237a2SJason Molenda } 402b9c1b51eSKate Stone if (failed) { 403b9c1b51eSKate Stone result.AppendErrorWithFormat( 404b9c1b51eSKate Stone "Could not find function bounds for address 0x%" PRIx64 405b9c1b51eSKate Stone "\n", 406b9c1b51eSKate Stone m_options.symbol_containing_addr); 407801237a2SJason Molenda result.SetStatus(eReturnStatusFailed); 408801237a2SJason Molenda return false; 409801237a2SJason Molenda } 4102f2c876eSJim Ingham ranges.push_back(range); 411b9c1b51eSKate Stone } else { 412b9c1b51eSKate Stone for (lldb::ModuleSP module_sp : target->GetImages().Modules()) { 4132f2c876eSJim Ingham lldb::addr_t file_addr = m_options.symbol_containing_addr; 4142f2c876eSJim Ingham Address file_address; 415b9c1b51eSKate Stone if (module_sp->ResolveFileAddress(file_addr, file_address)) { 4162f2c876eSJim Ingham SymbolContext sc; 417b9c1b51eSKate Stone bool resolve_tail_call_address = true; // PC can be one past 418b9c1b51eSKate Stone // the address range of 419b9c1b51eSKate Stone // the function. 420b9c1b51eSKate Stone module_sp->ResolveSymbolContextForAddress( 421b9c1b51eSKate Stone file_address, eSymbolContextEverything, sc, 422b9c1b51eSKate Stone resolve_tail_call_address); 423b9c1b51eSKate Stone if (sc.function || sc.symbol) { 424b9c1b51eSKate Stone sc.GetAddressRange(eSymbolContextFunction | 425b9c1b51eSKate Stone eSymbolContextSymbol, 426b9c1b51eSKate Stone 0, false, range); 4272f2c876eSJim Ingham ranges.push_back(range); 428801237a2SJason Molenda } 42937023b06SJim Ingham } 43037023b06SJim Ingham } 4312f2c876eSJim Ingham } 4322f2c876eSJim Ingham } 4332f2c876eSJim Ingham } 4342f2c876eSJim Ingham } 435b9c1b51eSKate Stone } else 4362f2c876eSJim Ingham ranges.push_back(range); 4372f2c876eSJim Ingham 438b9c1b51eSKate Stone if (ranges.empty()) { 4398ceb8ba2SJohnny Chen // The default action is to disassemble the current frame function. 440b9c1b51eSKate Stone if (frame) { 441b9c1b51eSKate Stone SymbolContext sc(frame->GetSymbolContext(eSymbolContextFunction | 442b9c1b51eSKate Stone eSymbolContextSymbol)); 44330fdc8d8SChris Lattner if (sc.function) 444dda4f7b5SGreg Clayton range = sc.function->GetAddressRange(); 445b9c1b51eSKate Stone else if (sc.symbol && sc.symbol->ValueIsAddress()) { 446e7612134SGreg Clayton range.GetBaseAddress() = sc.symbol->GetAddress(); 447e7612134SGreg Clayton range.SetByteSize(sc.symbol->GetByteSize()); 448b9c1b51eSKate Stone } else 449c14ee32dSGreg Clayton range.GetBaseAddress() = frame->GetFrameCodeAddress(); 4503245dd59SPavel Labath } 4513245dd59SPavel Labath if (!range.GetBaseAddress().IsValid()) { 45230fdc8d8SChris Lattner result.AppendError("invalid frame"); 45330fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 45430fdc8d8SChris Lattner return false; 45530fdc8d8SChris Lattner } 4562f2c876eSJim Ingham ranges.push_back(range); 45730fdc8d8SChris Lattner } 458c6a38957SPavel Labath } 4592f2c876eSJim Ingham 4602f2c876eSJim Ingham bool print_sc_header = ranges.size() > 1; 461b9c1b51eSKate Stone for (AddressRange cur_range : ranges) { 462*af3db4e9SPavel Labath Disassembler::Limit limit; 463*af3db4e9SPavel Labath if (m_options.num_instructions == 0) { 464*af3db4e9SPavel Labath limit = {Disassembler::Limit::Bytes, cur_range.GetByteSize()}; 465*af3db4e9SPavel Labath if (limit.value == 0) 466*af3db4e9SPavel Labath limit.value = DEFAULT_DISASM_BYTE_SIZE; 4673245dd59SPavel Labath } else { 468*af3db4e9SPavel Labath limit = {Disassembler::Limit::Instructions, m_options.num_instructions}; 4693245dd59SPavel Labath } 470*af3db4e9SPavel Labath if (Disassembler::Disassemble( 471*af3db4e9SPavel Labath GetDebugger(), m_options.arch, plugin_name, flavor_string, 472*af3db4e9SPavel Labath m_exe_ctx, cur_range.GetBaseAddress(), limit, m_options.show_mixed, 473*af3db4e9SPavel Labath m_options.show_mixed ? m_options.num_lines_context : 0, options, 474*af3db4e9SPavel Labath result.GetOutputStream())) { 475dda4f7b5SGreg Clayton result.SetStatus(eReturnStatusSuccessFinishResult); 476b9c1b51eSKate Stone } else { 4773245dd59SPavel Labath if (m_options.symbol_containing_addr != LLDB_INVALID_ADDRESS) { 4783245dd59SPavel Labath result.AppendErrorWithFormat( 4793245dd59SPavel Labath "Failed to disassemble memory in function at 0x%8.8" PRIx64 ".\n", 4803245dd59SPavel Labath m_options.symbol_containing_addr); 4813245dd59SPavel Labath } else { 482b9c1b51eSKate Stone result.AppendErrorWithFormat( 483b9c1b51eSKate Stone "Failed to disassemble memory at 0x%8.8" PRIx64 ".\n", 48405229752SPavel Labath cur_range.GetBaseAddress().GetLoadAddress(target)); 4853245dd59SPavel Labath } 48630fdc8d8SChris Lattner result.SetStatus(eReturnStatusFailed); 48730fdc8d8SChris Lattner } 4882f2c876eSJim Ingham if (print_sc_header) 4892f2c876eSJim Ingham result.AppendMessage("\n"); 4902f2c876eSJim Ingham } 49130fdc8d8SChris Lattner 49230fdc8d8SChris Lattner return result.Succeeded(); 49330fdc8d8SChris Lattner } 494