//===-- IntelPTCollector.cpp ------------------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "IntelPTCollector.h" #include "Perf.h" #include "Procfs.h" #include "Plugins/Process/POSIX/ProcessPOSIXLog.h" #include "lldb/Host/linux/Support.h" #include "lldb/Utility/StreamString.h" #include "llvm/ADT/StringRef.h" #include "llvm/Support/Error.h" #include "llvm/Support/MathExtras.h" #include #include #include #include #include #include #include using namespace lldb; using namespace lldb_private; using namespace process_linux; using namespace llvm; IntelPTCollector::IntelPTCollector(NativeProcessProtocol &process) : m_process(process) { if (Expected tsc_conversion = LoadPerfTscConversionParameters()) m_tsc_conversion = std::make_unique(*tsc_conversion); else LLDB_LOG_ERROR(GetLog(POSIXLog::Trace), tsc_conversion.takeError(), "unable to load TSC to wall time conversion: {0}"); } Error IntelPTCollector::TraceStop(lldb::tid_t tid) { if (m_per_thread_process_trace_up && m_per_thread_process_trace_up->TracesThread(tid)) return m_per_thread_process_trace_up->TraceStop(tid); else if (m_per_core_process_trace_up) return createStringError(inconvertibleErrorCode(), "Can't stop tracing an individual thread when " "per-core process tracing is enabled."); return m_thread_traces.TraceStop(tid); } Error IntelPTCollector::TraceStop(const TraceStopRequest &request) { if (request.IsProcessTracing()) { Clear(); return Error::success(); } else { Error error = Error::success(); for (int64_t tid : *request.tids) error = joinErrors(std::move(error), TraceStop(static_cast(tid))); return error; } } Expected IntelPTPerThreadProcessTrace::Start(const TraceIntelPTStartRequest &request, ArrayRef current_tids) { IntelPTPerThreadProcessTraceUP trace( new IntelPTPerThreadProcessTrace(request)); Error error = Error::success(); for (lldb::tid_t tid : current_tids) error = joinErrors(std::move(error), trace->TraceStart(tid)); if (error) return std::move(error); return std::move(trace); } Error IntelPTCollector::TraceStart(const TraceIntelPTStartRequest &request) { if (request.IsProcessTracing()) { if (IsProcessTracingEnabled()) { return createStringError( inconvertibleErrorCode(), "Process currently traced. Stop process tracing first"); } if (request.IsPerCoreTracing()) { if (m_thread_traces.GetTracedThreadsCount() > 0) return createStringError( inconvertibleErrorCode(), "Threads currently traced. Stop tracing them first."); if (Expected trace = IntelPTMultiCoreTrace::StartOnAllCores(request)) { m_per_core_process_trace_up = std::move(*trace); return Error::success(); } else { return trace.takeError(); } } else { std::vector process_threads; for (size_t i = 0; m_process.GetThreadAtIndex(i); i++) process_threads.push_back(m_process.GetThreadAtIndex(i)->GetID()); // per-thread process tracing if (Expected trace = IntelPTPerThreadProcessTrace::Start(request, process_threads)) { m_per_thread_process_trace_up = std::move(trace.get()); return Error::success(); } else { return trace.takeError(); } } } else { // individual thread tracing if (m_per_core_process_trace_up) return createStringError(inconvertibleErrorCode(), "Process currently traced with per-core " "tracing. Stop process tracing first"); Error error = Error::success(); for (int64_t tid : *request.tids) error = joinErrors(std::move(error), m_thread_traces.TraceStart(tid, request)); return error; } } void IntelPTCollector::OnProcessStateChanged(lldb::StateType state) { if (m_per_core_process_trace_up) m_per_core_process_trace_up->OnProcessStateChanged(state); } Error IntelPTCollector::OnThreadCreated(lldb::tid_t tid) { if (m_per_thread_process_trace_up) return m_per_thread_process_trace_up->TraceStart(tid); return Error::success(); } Error IntelPTCollector::OnThreadDestroyed(lldb::tid_t tid) { if (m_per_thread_process_trace_up && m_per_thread_process_trace_up->TracesThread(tid)) return m_per_thread_process_trace_up->TraceStop(tid); else if (m_thread_traces.TracesThread(tid)) return m_thread_traces.TraceStop(tid); return Error::success(); } Expected IntelPTCollector::GetState() { Expected> cpu_info = GetProcfsCpuInfo(); if (!cpu_info) return cpu_info.takeError(); TraceGetStateResponse state; state.process_binary_data.push_back( {IntelPTDataKinds::kProcFsCpuInfo, cpu_info->size()}); m_thread_traces.ForEachThread( [&](lldb::tid_t tid, const IntelPTSingleBufferTrace &thread_trace) { state.traced_threads.push_back({tid, {{IntelPTDataKinds::kTraceBuffer, thread_trace.GetTraceBufferSize()}}}); }); if (m_per_thread_process_trace_up) { m_per_thread_process_trace_up->GetThreadTraces().ForEachThread( [&](lldb::tid_t tid, const IntelPTSingleBufferTrace &thread_trace) { state.traced_threads.push_back( {tid, {{IntelPTDataKinds::kTraceBuffer, thread_trace.GetTraceBufferSize()}}}); }); } if (m_per_core_process_trace_up) { for (size_t i = 0; m_process.GetThreadAtIndex(i); i++) state.traced_threads.push_back( TraceThreadState{m_process.GetThreadAtIndex(i)->GetID(), {}}); state.cores.emplace(); m_per_core_process_trace_up->ForEachCore( [&](lldb::core_id_t core_id, const IntelPTSingleBufferTrace &core_trace) { state.cores->push_back({core_id, {{IntelPTDataKinds::kTraceBuffer, core_trace.GetTraceBufferSize()}}}); }); } return toJSON(state); } Expected IntelPTCollector::GetTracedThread(lldb::tid_t tid) { if (m_per_thread_process_trace_up && m_per_thread_process_trace_up->TracesThread(tid)) return m_per_thread_process_trace_up->GetThreadTraces().GetTracedThread( tid); return m_thread_traces.GetTracedThread(tid); } Expected> IntelPTCollector::GetBinaryData(const TraceGetBinaryDataRequest &request) { if (request.kind == IntelPTDataKinds::kTraceBuffer) { if (Expected trace = GetTracedThread(*request.tid)) return trace->GetTraceBuffer(request.offset, request.size); else return trace.takeError(); } else if (request.kind == IntelPTDataKinds::kProcFsCpuInfo) { return GetProcfsCpuInfo(); } return createStringError(inconvertibleErrorCode(), "Unsuported trace binary data kind: %s", request.kind.c_str()); } void IntelPTCollector::ClearProcessTracing() { m_per_thread_process_trace_up.reset(); m_per_core_process_trace_up.reset(); } bool IntelPTCollector::IsSupported() { if (Expected intel_pt_type = GetIntelPTOSEventType()) { return true; } else { llvm::consumeError(intel_pt_type.takeError()); return false; } } bool IntelPTCollector::IsProcessTracingEnabled() const { return (bool)m_per_thread_process_trace_up || (bool)m_per_core_process_trace_up; } void IntelPTCollector::Clear() { ClearProcessTracing(); m_thread_traces.Clear(); }