diff --git a/lldb/include/lldb/Target/TraceInstructionDumper.h b/lldb/include/lldb/Target/TraceInstructionDumper.h --- a/lldb/include/lldb/Target/TraceInstructionDumper.h +++ b/lldb/include/lldb/Target/TraceInstructionDumper.h @@ -8,11 +8,24 @@ #include "lldb/Target/TraceCursor.h" +#include "lldb/Symbol/SymbolContext.h" + #ifndef LLDB_TARGET_TRACE_INSTRUCTION_DUMPER_H #define LLDB_TARGET_TRACE_INSTRUCTION_DUMPER_H namespace lldb_private { +/// Helper struct that holds symbol, disassembly and address information of an +/// instruction. +struct InstructionSymbolInfo { + SymbolContext sc; + Address address; + lldb::addr_t load_address; + lldb::DisassemblerSP disassembler; + lldb::InstructionSP instruction; + lldb_private::ExecutionContext exe_ctx; +}; + /// Class that holds the configuration used by \a TraceInstructionDumper for /// traversing and dumping instructions. struct TraceInstructionDumperOptions { @@ -83,6 +96,28 @@ void PrintEvents(); + void PrintMissingInstructionsMessage(); + + void PrintInstructionHeader(); + + void DumpInstructionDisassembly(const InstructionSymbolInfo &insn); + + /// Dump the symbol context of the given instruction address if it's different + /// from the symbol context of the previous instruction in the trace. + /// + /// \param[in] prev_sc + /// The symbol context of the previous instruction in the trace. + /// + /// \param[in] address + /// The address whose symbol information will be dumped. + /// + /// \return + /// The symbol context of the current address, which might differ from the + /// previous one. + void DumpInstructionSymbolContext( + const llvm::Optional<InstructionSymbolInfo> &prev_insn, + const InstructionSymbolInfo &insn); + lldb::TraceCursorUP m_cursor_up; TraceInstructionDumperOptions m_options; Stream &m_s; diff --git a/lldb/source/Target/TraceInstructionDumper.cpp b/lldb/source/Target/TraceInstructionDumper.cpp --- a/lldb/source/Target/TraceInstructionDumper.cpp +++ b/lldb/source/Target/TraceInstructionDumper.cpp @@ -47,8 +47,6 @@ } } -/// \return -/// Return \b true if the cursor could move one step. bool TraceInstructionDumper::TryMoveOneStep() { if (!m_cursor_up->Next()) { SetNoMoreData(); @@ -57,17 +55,6 @@ return true; } -/// Helper struct that holds symbol, disassembly and address information of an -/// instruction. -struct InstructionSymbolInfo { - SymbolContext sc; - Address address; - lldb::addr_t load_address; - lldb::DisassemblerSP disassembler; - lldb::InstructionSP instruction; - lldb_private::ExecutionContext exe_ctx; -}; - // This custom LineEntry validator is neded because some line_entries have // 0 as line, which is meaningless. Notice that LineEntry::IsValid only // checks that line is not LLDB_INVALID_LINE_NUMBER, i.e. UINT32_MAX. @@ -122,46 +109,35 @@ return curr_line_valid == prev_line_valid; } -/// Dump the symbol context of the given instruction address if it's different -/// from the symbol context of the previous instruction in the trace. -/// -/// \param[in] prev_sc -/// The symbol context of the previous instruction in the trace. -/// -/// \param[in] address -/// The address whose symbol information will be dumped. -/// -/// \return -/// The symbol context of the current address, which might differ from the -/// previous one. -static void -DumpInstructionSymbolContext(Stream &s, - Optional<InstructionSymbolInfo> prev_insn, - InstructionSymbolInfo &insn) { +void TraceInstructionDumper::DumpInstructionSymbolContext( + const Optional<InstructionSymbolInfo> &prev_insn, + const InstructionSymbolInfo &insn) { if (prev_insn && IsSameInstructionSymbolContext(*prev_insn, insn)) return; - s.Printf(" "); + m_s << " "; if (!insn.sc.module_sp) - s.Printf("(none)"); + m_s << "(none)"; else if (!insn.sc.function && !insn.sc.symbol) - s.Printf("%s`(none)", - insn.sc.module_sp->GetFileSpec().GetFilename().AsCString()); + m_s.Format("{0}`(none)", + insn.sc.module_sp->GetFileSpec().GetFilename().AsCString()); else - insn.sc.DumpStopContext(&s, insn.exe_ctx.GetTargetPtr(), insn.address, + insn.sc.DumpStopContext(&m_s, insn.exe_ctx.GetTargetPtr(), insn.address, /*show_fullpaths=*/false, /*show_module=*/true, /*show_inlined_frames=*/false, /*show_function_arguments=*/true, /*show_function_name=*/true); - s.Printf("\n"); + m_s << "\n"; } -static void DumpInstructionDisassembly(Stream &s, InstructionSymbolInfo &insn) { +void TraceInstructionDumper::DumpInstructionDisassembly( + const InstructionSymbolInfo &insn) { if (!insn.instruction) return; - s.Printf(" "); - insn.instruction->Dump(&s, /*max_opcode_byte_size=*/0, /*show_address=*/false, + m_s << " "; + insn.instruction->Dump(&m_s, /*max_opcode_byte_size=*/0, + /*show_address=*/false, /*show_bytes=*/false, &insn.exe_ctx, &insn.sc, /*prev_sym_ctx=*/nullptr, /*disassembly_addr_format=*/nullptr, @@ -172,6 +148,26 @@ bool TraceInstructionDumper::HasMoreData() { return !m_no_more_data; } +void TraceInstructionDumper::PrintMissingInstructionsMessage() { + m_s << " ...missing instructions\n"; +} + +void TraceInstructionDumper::PrintInstructionHeader() { + m_s.Format(" {0}: ", m_cursor_up->GetId()); + + if (m_options.show_tsc) { + m_s << "[tsc="; + + if (Optional<uint64_t> timestamp = + m_cursor_up->GetCounter(lldb::eTraceCounterTSC)) + m_s.Format("{0:x+16}", *timestamp); + else + m_s << "unavailable"; + + m_s << "] "; + } +} + void TraceInstructionDumper::PrintEvents() { if (!m_options.show_events) return; @@ -182,90 +178,76 @@ }); } -Optional<lldb::tid_t> TraceInstructionDumper::DumpInstructions(size_t count) { +/// Find the symbol context for the given address reusing the previous +/// instruction's symbol context when possible. +static SymbolContext +CalculateSymbolContext(const Address &address, + const InstructionSymbolInfo &prev_insn_info) { + AddressRange range; + if (prev_insn_info.sc.GetAddressRange(eSymbolContextEverything, 0, + /*inline_block_range*/ false, range) && + range.Contains(address)) + return prev_insn_info.sc; + + SymbolContext sc; + address.CalculateSymbolContext(&sc, eSymbolContextEverything); + return sc; +} + +/// Find the disassembler for the given address reusing the previous +/// instruction's disassembler when possible. +static std::tuple<DisassemblerSP, InstructionSP> +CalculateDisass(const InstructionSymbolInfo &insn_info, + const InstructionSymbolInfo &prev_insn_info, + const ExecutionContext &exe_ctx) { + if (prev_insn_info.disassembler) { + if (InstructionSP instruction = + prev_insn_info.disassembler->GetInstructionList() + .GetInstructionAtAddress(insn_info.address)) + return std::make_tuple(prev_insn_info.disassembler, instruction); + } + + if (insn_info.sc.function) { + if (DisassemblerSP disassembler = + insn_info.sc.function->GetInstructions(exe_ctx, nullptr)) { + if (InstructionSP instruction = + disassembler->GetInstructionList().GetInstructionAtAddress( + insn_info.address)) + return std::make_tuple(disassembler, instruction); + } + } + // We fallback to a single instruction disassembler + Target &target = exe_ctx.GetTargetRef(); + const ArchSpec arch = target.GetArchitecture(); + AddressRange range(insn_info.address, arch.GetMaximumOpcodeByteSize()); + DisassemblerSP disassembler = + Disassembler::DisassembleRange(arch, /*plugin_name*/ nullptr, + /*flavor*/ nullptr, target, range); + return std::make_tuple( + disassembler, + disassembler ? disassembler->GetInstructionList().GetInstructionAtAddress( + insn_info.address) + : InstructionSP()); +} + +Optional<lldb::user_id_t> +TraceInstructionDumper::DumpInstructions(size_t count) { ThreadSP thread_sp = m_cursor_up->GetExecutionContextRef().GetThreadSP(); if (!thread_sp) { - m_s.Printf("invalid thread"); + m_s << "invalid thread"; return None; } bool was_prev_instruction_an_error = false; - - auto printMissingInstructionsMessage = [&]() { - m_s.Printf(" ...missing instructions\n"); - }; - - auto printInstructionHeader = [&](uint64_t id) { - m_s.Printf(" %" PRIu64 ": ", id); - - if (m_options.show_tsc) { - m_s.Printf("[tsc="); - - if (Optional<uint64_t> timestamp = m_cursor_up->GetCounter(lldb::eTraceCounterTSC)) - m_s.Printf("0x%016" PRIx64, *timestamp); - else - m_s.Printf("unavailable"); - - m_s.Printf("] "); - } - }; - InstructionSymbolInfo prev_insn_info; + Optional<lldb::user_id_t> last_id; - Target &target = thread_sp->GetProcess()->GetTarget(); ExecutionContext exe_ctx; - target.CalculateExecutionContext(exe_ctx); - const ArchSpec &arch = target.GetArchitecture(); - - // Find the symbol context for the given address reusing the previous - // instruction's symbol context when possible. - auto calculateSymbolContext = [&](const Address &address) { - AddressRange range; - if (prev_insn_info.sc.GetAddressRange(eSymbolContextEverything, 0, - /*inline_block_range*/ false, - range) && - range.Contains(address)) - return prev_insn_info.sc; - - SymbolContext sc; - address.CalculateSymbolContext(&sc, eSymbolContextEverything); - return sc; - }; - - // Find the disassembler for the given address reusing the previous - // instruction's disassembler when possible. - auto calculateDisass = [&](const Address &address, const SymbolContext &sc) { - if (prev_insn_info.disassembler) { - if (InstructionSP instruction = - prev_insn_info.disassembler->GetInstructionList() - .GetInstructionAtAddress(address)) - return std::make_tuple(prev_insn_info.disassembler, instruction); - } + thread_sp->GetProcess()->GetTarget().CalculateExecutionContext(exe_ctx); - if (sc.function) { - if (DisassemblerSP disassembler = - sc.function->GetInstructions(exe_ctx, nullptr)) { - if (InstructionSP instruction = - disassembler->GetInstructionList().GetInstructionAtAddress( - address)) - return std::make_tuple(disassembler, instruction); - } - } - // We fallback to a single instruction disassembler - AddressRange range(address, arch.GetMaximumOpcodeByteSize()); - DisassemblerSP disassembler = - Disassembler::DisassembleRange(arch, /*plugin_name*/ nullptr, - /*flavor*/ nullptr, target, range); - return std::make_tuple(disassembler, - disassembler ? disassembler->GetInstructionList() - .GetInstructionAtAddress(address) - : InstructionSP()); - }; - - Optional<lldb::user_id_t> last_id; for (size_t i = 0; i < count; i++) { if (!HasMoreData()) { - m_s.Printf(" no more data\n"); + m_s << " no more data\n"; break; } last_id = m_cursor_up->GetId(); @@ -277,15 +259,15 @@ if (const char *err = m_cursor_up->GetError()) { if (!m_cursor_up->IsForwards() && !was_prev_instruction_an_error) - printMissingInstructionsMessage(); + PrintMissingInstructionsMessage(); was_prev_instruction_an_error = true; - printInstructionHeader(m_cursor_up->GetId()); + PrintInstructionHeader(); m_s << err; } else { if (m_cursor_up->IsForwards() && was_prev_instruction_an_error) - printMissingInstructionsMessage(); + PrintMissingInstructionsMessage(); was_prev_instruction_an_error = false; @@ -294,24 +276,26 @@ if (!m_options.raw) { insn_info.load_address = m_cursor_up->GetLoadAddress(); insn_info.exe_ctx = exe_ctx; - insn_info.address.SetLoadAddress(insn_info.load_address, &target); - insn_info.sc = calculateSymbolContext(insn_info.address); + insn_info.address.SetLoadAddress(insn_info.load_address, + exe_ctx.GetTargetPtr()); + insn_info.sc = + CalculateSymbolContext(insn_info.address, prev_insn_info); std::tie(insn_info.disassembler, insn_info.instruction) = - calculateDisass(insn_info.address, insn_info.sc); + CalculateDisass(insn_info, prev_insn_info, exe_ctx); - DumpInstructionSymbolContext(m_s, prev_insn_info, insn_info); + DumpInstructionSymbolContext(prev_insn_info, insn_info); } - printInstructionHeader(m_cursor_up->GetId()); - m_s.Printf("0x%016" PRIx64, m_cursor_up->GetLoadAddress()); + PrintInstructionHeader(); + m_s.Format("{0:x+16}", m_cursor_up->GetLoadAddress()); if (!m_options.raw) - DumpInstructionDisassembly(m_s, insn_info); + DumpInstructionDisassembly(insn_info); prev_insn_info = insn_info; } - m_s.Printf("\n"); + m_s << "\n"; if (!m_options.forwards) { // If we move backwards, we print the events after printing