xref: /openbsd-src/gnu/llvm/lldb/source/Plugins/Trace/intel-pt/ThreadDecoder.cpp (revision f6aab3d83b51b91c24247ad2c2573574de475a82)
1*f6aab3d8Srobert //===-- ThreadDecoder.cpp --======-----------------------------------------===//
2*f6aab3d8Srobert // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
3*f6aab3d8Srobert // See https://llvm.org/LICENSE.txt for license information.
4*f6aab3d8Srobert // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
5*f6aab3d8Srobert //
6*f6aab3d8Srobert //===----------------------------------------------------------------------===//
7*f6aab3d8Srobert 
8*f6aab3d8Srobert #include "ThreadDecoder.h"
9*f6aab3d8Srobert #include "../common/ThreadPostMortemTrace.h"
10*f6aab3d8Srobert #include "LibiptDecoder.h"
11*f6aab3d8Srobert #include "TraceIntelPT.h"
12*f6aab3d8Srobert #include "llvm/Support/MemoryBuffer.h"
13*f6aab3d8Srobert #include <optional>
14*f6aab3d8Srobert #include <utility>
15*f6aab3d8Srobert 
16*f6aab3d8Srobert using namespace lldb;
17*f6aab3d8Srobert using namespace lldb_private;
18*f6aab3d8Srobert using namespace lldb_private::trace_intel_pt;
19*f6aab3d8Srobert using namespace llvm;
20*f6aab3d8Srobert 
ThreadDecoder(const ThreadSP & thread_sp,TraceIntelPT & trace)21*f6aab3d8Srobert ThreadDecoder::ThreadDecoder(const ThreadSP &thread_sp, TraceIntelPT &trace)
22*f6aab3d8Srobert     : m_thread_sp(thread_sp), m_trace(trace) {}
23*f6aab3d8Srobert 
FindLowestTSC()24*f6aab3d8Srobert Expected<std::optional<uint64_t>> ThreadDecoder::FindLowestTSC() {
25*f6aab3d8Srobert   std::optional<uint64_t> lowest_tsc;
26*f6aab3d8Srobert   Error err = m_trace.OnThreadBufferRead(
27*f6aab3d8Srobert       m_thread_sp->GetID(), [&](llvm::ArrayRef<uint8_t> data) -> llvm::Error {
28*f6aab3d8Srobert         Expected<std::optional<uint64_t>> tsc =
29*f6aab3d8Srobert             FindLowestTSCInTrace(m_trace, data);
30*f6aab3d8Srobert         if (!tsc)
31*f6aab3d8Srobert           return tsc.takeError();
32*f6aab3d8Srobert         lowest_tsc = *tsc;
33*f6aab3d8Srobert         return Error::success();
34*f6aab3d8Srobert       });
35*f6aab3d8Srobert   if (err)
36*f6aab3d8Srobert     return std::move(err);
37*f6aab3d8Srobert   return lowest_tsc;
38*f6aab3d8Srobert }
39*f6aab3d8Srobert 
Decode()40*f6aab3d8Srobert Expected<DecodedThreadSP> ThreadDecoder::Decode() {
41*f6aab3d8Srobert   if (!m_decoded_thread.has_value()) {
42*f6aab3d8Srobert     if (Expected<DecodedThreadSP> decoded_thread = DoDecode()) {
43*f6aab3d8Srobert       m_decoded_thread = *decoded_thread;
44*f6aab3d8Srobert     } else {
45*f6aab3d8Srobert       return decoded_thread.takeError();
46*f6aab3d8Srobert     }
47*f6aab3d8Srobert   }
48*f6aab3d8Srobert   return *m_decoded_thread;
49*f6aab3d8Srobert }
50*f6aab3d8Srobert 
DoDecode()51*f6aab3d8Srobert llvm::Expected<DecodedThreadSP> ThreadDecoder::DoDecode() {
52*f6aab3d8Srobert   return m_trace.GetThreadTimer(m_thread_sp->GetID())
53*f6aab3d8Srobert       .TimeTask("Decoding instructions", [&]() -> Expected<DecodedThreadSP> {
54*f6aab3d8Srobert         DecodedThreadSP decoded_thread_sp = std::make_shared<DecodedThread>(
55*f6aab3d8Srobert             m_thread_sp, m_trace.GetPerfZeroTscConversion());
56*f6aab3d8Srobert 
57*f6aab3d8Srobert         Error err = m_trace.OnThreadBufferRead(
58*f6aab3d8Srobert             m_thread_sp->GetID(), [&](llvm::ArrayRef<uint8_t> data) {
59*f6aab3d8Srobert               return DecodeSingleTraceForThread(*decoded_thread_sp, m_trace,
60*f6aab3d8Srobert                                                 data);
61*f6aab3d8Srobert             });
62*f6aab3d8Srobert 
63*f6aab3d8Srobert         if (err)
64*f6aab3d8Srobert           return std::move(err);
65*f6aab3d8Srobert         return decoded_thread_sp;
66*f6aab3d8Srobert       });
67*f6aab3d8Srobert }
68