| 1 | //===-- TraceIntelPTMultiCpuDecoder.h ---------------------------*- C++ -*-===// |
| 2 | // |
| 3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| 4 | // See https://llvm.org/LICENSE.txt for license information. |
| 5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| 6 | // |
| 7 | //===----------------------------------------------------------------------===// |
| 8 | |
| 9 | #ifndef LLDB_SOURCE_PLUGINS_TRACE_INTEL_PT_TRACEINTELPTMULTICPUDECODER_H |
| 10 | #define LLDB_SOURCE_PLUGINS_TRACE_INTEL_PT_TRACEINTELPTMULTICPUDECODER_H |
| 11 | |
| 12 | #include "LibiptDecoder.h" |
| 13 | #include "PerfContextSwitchDecoder.h" |
| 14 | #include "ThreadDecoder.h" |
| 15 | #include "forward-declarations.h" |
| 16 | #include <optional> |
| 17 | |
| 18 | namespace lldb_private { |
| 19 | namespace trace_intel_pt { |
| 20 | |
| 21 | /// Class used to decode a multi-cpu Intel PT trace. It assumes that each |
| 22 | /// thread could have potentially been executed on different cpu cores. It uses |
| 23 | /// a context switch trace per CPU with timestamps to identify which thread owns |
| 24 | /// each Intel PT decoded instruction and in which order. It also assumes that |
| 25 | /// the Intel PT data and context switches might have gaps in their traces due |
| 26 | /// to contention or race conditions. Finally, it assumes that a tid is not |
| 27 | /// repeated twice for two different threads because of the shortness of the |
| 28 | /// intel pt trace. |
| 29 | /// |
| 30 | /// This object should be recreated after every stop in the case of live |
| 31 | /// processes. |
| 32 | class TraceIntelPTMultiCpuDecoder { |
| 33 | public: |
| 34 | /// \param[in] TraceIntelPT |
| 35 | /// The trace object to be decoded |
| 36 | TraceIntelPTMultiCpuDecoder(TraceIntelPTSP trace_sp); |
| 37 | |
| 38 | /// \return |
| 39 | /// A \a DecodedThread for the \p thread by decoding its instructions on all |
| 40 | /// CPUs, sorted by TSCs. An \a llvm::Error is returned if the decoder |
| 41 | /// couldn't be properly set up. |
| 42 | llvm::Expected<DecodedThreadSP> Decode(Thread &thread); |
| 43 | |
| 44 | /// \return |
| 45 | /// \b true if the given \p tid is managed by this decoder, regardless of |
| 46 | /// whether there's tracing data associated to it or not. |
| 47 | bool TracesThread(lldb::tid_t tid) const; |
| 48 | |
| 49 | /// \return |
| 50 | /// The number of continuous executions found for the given \p tid. |
| 51 | size_t GetNumContinuousExecutionsForThread(lldb::tid_t tid) const; |
| 52 | |
| 53 | /// \return |
| 54 | /// The number of PSB blocks for a given thread in all cores. |
| 55 | size_t GePSBBlocksCountForThread(lldb::tid_t tid) const; |
| 56 | |
| 57 | /// \return |
| 58 | /// The total number of continuous executions found across CPUs. |
| 59 | size_t GetTotalContinuousExecutionsCount() const; |
| 60 | |
| 61 | /// \return |
| 62 | /// The number of psb blocks in all cores that couldn't be matched with a |
| 63 | /// thread execution coming from context switch traces. |
| 64 | size_t GetUnattributedPSBBlocksCount() const; |
| 65 | |
| 66 | /// \return |
| 67 | /// The total number of PSB blocks in all cores. |
| 68 | size_t GetTotalPSBBlocksCount() const; |
| 69 | |
| 70 | /// \return |
| 71 | /// The lowest TSC value in this trace if available, \a std::nullopt if |
| 72 | /// the trace is empty or the trace contains no timing information, or an |
| 73 | /// \a llvm::Error if it was not possible to set up the decoder. |
| 74 | llvm::Expected<std::optional<uint64_t>> FindLowestTSC(); |
| 75 | |
| 76 | private: |
| 77 | /// Traverse the context switch traces and the basic intel pt continuous |
| 78 | /// subtraces and produce a list of continuous executions for each process and |
| 79 | /// thread. |
| 80 | /// |
| 81 | /// See \a DoCorrelateContextSwitchesAndIntelPtTraces. |
| 82 | /// |
| 83 | /// Any errors are stored in \a m_setup_error. |
| 84 | llvm::Error CorrelateContextSwitchesAndIntelPtTraces(); |
| 85 | |
| 86 | /// Produce a mapping from thread ids to the list of continuos executions with |
| 87 | /// their associated intel pt subtraces. |
| 88 | llvm::Expected< |
| 89 | llvm::DenseMap<lldb::tid_t, std::vector<IntelPTThreadContinousExecution>>> |
| 90 | DoCorrelateContextSwitchesAndIntelPtTraces(); |
| 91 | |
| 92 | TraceIntelPTSP GetTrace(); |
| 93 | |
| 94 | std::weak_ptr<TraceIntelPT> m_trace_wp; |
| 95 | std::set<lldb::tid_t> m_tids; |
| 96 | std::optional< |
| 97 | llvm::DenseMap<lldb::tid_t, std::vector<IntelPTThreadContinousExecution>>> |
| 98 | m_continuous_executions_per_thread; |
| 99 | llvm::DenseMap<lldb::tid_t, DecodedThreadSP> m_decoded_threads; |
| 100 | /// This variable will not be std::nullopt if a severe error happened during |
| 101 | /// the setup of the decoder and we don't want decoding to be reattempted. |
| 102 | std::optional<std::string> m_setup_error; |
| 103 | uint64_t m_unattributed_psb_blocks = 0; |
| 104 | uint64_t m_total_psb_blocks = 0; |
| 105 | }; |
| 106 | |
| 107 | } // namespace trace_intel_pt |
| 108 | } // namespace lldb_private |
| 109 | |
| 110 | #endif // LLDB_SOURCE_PLUGINS_TRACE_INTEL_PT_TRACEINTELPTMULTICPUDECODER_H |
| 111 | |