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