xref: /openbsd-src/gnu/llvm/lldb/source/Plugins/TraceExporter/ctf/CommandObjectThreadTraceExportCTF.cpp (revision f6aab3d83b51b91c24247ad2c2573574de475a82)
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