1be691f3bSpatrick //===-- CommandObjectThreadTraceExportCTF.cpp -----------------------------===//
2be691f3bSpatrick //
3be691f3bSpatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4be691f3bSpatrick // See https://llvm.org/LICENSE.txt for license information.
5be691f3bSpatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6be691f3bSpatrick //
7be691f3bSpatrick //===----------------------------------------------------------------------===//
8be691f3bSpatrick
9be691f3bSpatrick #include "CommandObjectThreadTraceExportCTF.h"
10be691f3bSpatrick
11*f6aab3d8Srobert #include "../common/TraceHTR.h"
12be691f3bSpatrick #include "lldb/Host/OptionParser.h"
13*f6aab3d8Srobert #include "lldb/Interpreter/CommandOptionArgumentTable.h"
14*f6aab3d8Srobert #include "lldb/Target/Process.h"
15*f6aab3d8Srobert #include "lldb/Target/Trace.h"
16be691f3bSpatrick
17be691f3bSpatrick using namespace lldb;
18be691f3bSpatrick using namespace lldb_private;
19be691f3bSpatrick using namespace lldb_private::ctf;
20be691f3bSpatrick using namespace llvm;
21be691f3bSpatrick
22be691f3bSpatrick // CommandObjectThreadTraceExportCTF
23be691f3bSpatrick
24be691f3bSpatrick #define LLDB_OPTIONS_thread_trace_export_ctf
25be691f3bSpatrick #include "TraceExporterCTFCommandOptions.inc"
26be691f3bSpatrick
SetOptionValue(uint32_t option_idx,llvm::StringRef option_arg,ExecutionContext * execution_context)27be691f3bSpatrick Status CommandObjectThreadTraceExportCTF::CommandOptions::SetOptionValue(
28be691f3bSpatrick uint32_t option_idx, llvm::StringRef option_arg,
29be691f3bSpatrick ExecutionContext *execution_context) {
30be691f3bSpatrick Status error;
31be691f3bSpatrick const int short_option = m_getopt_table[option_idx].val;
32be691f3bSpatrick
33be691f3bSpatrick switch (short_option) {
34*f6aab3d8Srobert case 'f': {
35*f6aab3d8Srobert m_file.assign(std::string(option_arg));
36*f6aab3d8Srobert break;
37*f6aab3d8Srobert }
38be691f3bSpatrick case 't': {
39be691f3bSpatrick int64_t thread_index;
40be691f3bSpatrick if (option_arg.empty() || option_arg.getAsInteger(0, thread_index) ||
41be691f3bSpatrick thread_index < 0)
42be691f3bSpatrick error.SetErrorStringWithFormat("invalid integer value for option '%s'",
43be691f3bSpatrick option_arg.str().c_str());
44be691f3bSpatrick else
45be691f3bSpatrick m_thread_index = thread_index;
46be691f3bSpatrick break;
47be691f3bSpatrick }
48be691f3bSpatrick default:
49be691f3bSpatrick llvm_unreachable("Unimplemented option");
50be691f3bSpatrick }
51be691f3bSpatrick return error;
52be691f3bSpatrick }
53be691f3bSpatrick
OptionParsingStarting(ExecutionContext * execution_context)54be691f3bSpatrick void CommandObjectThreadTraceExportCTF::CommandOptions::OptionParsingStarting(
55be691f3bSpatrick ExecutionContext *execution_context) {
56*f6aab3d8Srobert m_file.clear();
57*f6aab3d8Srobert m_thread_index = std::nullopt;
58be691f3bSpatrick }
59be691f3bSpatrick
60be691f3bSpatrick llvm::ArrayRef<OptionDefinition>
GetDefinitions()61be691f3bSpatrick CommandObjectThreadTraceExportCTF::CommandOptions::GetDefinitions() {
62*f6aab3d8Srobert return llvm::ArrayRef(g_thread_trace_export_ctf_options);
63be691f3bSpatrick }
64be691f3bSpatrick
DoExecute(Args & command,CommandReturnObject & result)65be691f3bSpatrick bool CommandObjectThreadTraceExportCTF::DoExecute(Args &command,
66be691f3bSpatrick CommandReturnObject &result) {
67*f6aab3d8Srobert const TraceSP &trace_sp = m_exe_ctx.GetTargetSP()->GetTrace();
68*f6aab3d8Srobert Process *process = m_exe_ctx.GetProcessPtr();
69*f6aab3d8Srobert Thread *thread = m_options.m_thread_index
70*f6aab3d8Srobert ? process->GetThreadList()
71*f6aab3d8Srobert .FindThreadByIndexID(*m_options.m_thread_index)
72*f6aab3d8Srobert .get()
73*f6aab3d8Srobert : GetDefaultThread();
74be691f3bSpatrick
75*f6aab3d8Srobert if (thread == nullptr) {
76*f6aab3d8Srobert const uint32_t num_threads = process->GetThreadList().GetSize();
77*f6aab3d8Srobert size_t tid = m_options.m_thread_index.value_or(LLDB_INVALID_THREAD_ID);
78*f6aab3d8Srobert result.AppendErrorWithFormatv(
79*f6aab3d8Srobert "Thread index {0} is out of range (valid values are 1 - {1}).\n", tid,
80*f6aab3d8Srobert num_threads);
81*f6aab3d8Srobert return false;
82*f6aab3d8Srobert } else {
83*f6aab3d8Srobert auto do_work = [&]() -> Error {
84*f6aab3d8Srobert Expected<TraceCursorSP> cursor = trace_sp->CreateNewCursor(*thread);
85*f6aab3d8Srobert if (!cursor)
86*f6aab3d8Srobert return cursor.takeError();
87*f6aab3d8Srobert TraceHTR htr(*thread, **cursor);
88*f6aab3d8Srobert htr.ExecutePasses();
89*f6aab3d8Srobert return htr.Export(m_options.m_file);
90*f6aab3d8Srobert };
91*f6aab3d8Srobert
92*f6aab3d8Srobert if (llvm::Error err = do_work()) {
93*f6aab3d8Srobert result.AppendErrorWithFormat("%s\n", toString(std::move(err)).c_str());
94*f6aab3d8Srobert return false;
95*f6aab3d8Srobert } else {
96*f6aab3d8Srobert return true;
97*f6aab3d8Srobert }
98*f6aab3d8Srobert }
99be691f3bSpatrick }
100