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