xref: /openbsd-src/gnu/llvm/lldb/source/Plugins/Trace/intel-pt/CommandObjectTraceStartIntelPT.cpp (revision f6aab3d83b51b91c24247ad2c2573574de475a82)
1be691f3bSpatrick //===-- CommandObjectTraceStartIntelPT.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 "CommandObjectTraceStartIntelPT.h"
10be691f3bSpatrick #include "TraceIntelPT.h"
11be691f3bSpatrick #include "TraceIntelPTConstants.h"
12be691f3bSpatrick #include "lldb/Host/OptionParser.h"
13*f6aab3d8Srobert #include "lldb/Interpreter/CommandOptionArgumentTable.h"
14be691f3bSpatrick #include "lldb/Target/Process.h"
15be691f3bSpatrick #include "lldb/Target/Trace.h"
16*f6aab3d8Srobert #include <optional>
17be691f3bSpatrick 
18be691f3bSpatrick using namespace lldb;
19be691f3bSpatrick using namespace lldb_private;
20be691f3bSpatrick using namespace lldb_private::trace_intel_pt;
21be691f3bSpatrick using namespace llvm;
22be691f3bSpatrick 
23be691f3bSpatrick // CommandObjectThreadTraceStartIntelPT
24be691f3bSpatrick 
25be691f3bSpatrick #define LLDB_OPTIONS_thread_trace_start_intel_pt
26be691f3bSpatrick #include "TraceIntelPTCommandOptions.inc"
27be691f3bSpatrick 
SetOptionValue(uint32_t option_idx,llvm::StringRef option_arg,ExecutionContext * execution_context)28be691f3bSpatrick Status CommandObjectThreadTraceStartIntelPT::CommandOptions::SetOptionValue(
29be691f3bSpatrick     uint32_t option_idx, llvm::StringRef option_arg,
30be691f3bSpatrick     ExecutionContext *execution_context) {
31be691f3bSpatrick   Status error;
32be691f3bSpatrick   const int short_option = m_getopt_table[option_idx].val;
33be691f3bSpatrick 
34be691f3bSpatrick   switch (short_option) {
35be691f3bSpatrick   case 's': {
36*f6aab3d8Srobert     if (std::optional<uint64_t> bytes =
37*f6aab3d8Srobert             ParsingUtils::ParseUserFriendlySizeExpression(option_arg))
38*f6aab3d8Srobert       m_ipt_trace_size = *bytes;
39be691f3bSpatrick     else
40*f6aab3d8Srobert       error.SetErrorStringWithFormat("invalid bytes expression for '%s'",
41*f6aab3d8Srobert                                      option_arg.str().c_str());
42be691f3bSpatrick     break;
43be691f3bSpatrick   }
44be691f3bSpatrick   case 't': {
45be691f3bSpatrick     m_enable_tsc = true;
46be691f3bSpatrick     break;
47be691f3bSpatrick   }
48be691f3bSpatrick   case 'p': {
49be691f3bSpatrick     int64_t psb_period;
50be691f3bSpatrick     if (option_arg.empty() || option_arg.getAsInteger(0, psb_period) ||
51be691f3bSpatrick         psb_period < 0)
52be691f3bSpatrick       error.SetErrorStringWithFormat("invalid integer value for option '%s'",
53be691f3bSpatrick                                      option_arg.str().c_str());
54be691f3bSpatrick     else
55be691f3bSpatrick       m_psb_period = psb_period;
56be691f3bSpatrick     break;
57be691f3bSpatrick   }
58be691f3bSpatrick   default:
59be691f3bSpatrick     llvm_unreachable("Unimplemented option");
60be691f3bSpatrick   }
61be691f3bSpatrick   return error;
62be691f3bSpatrick }
63be691f3bSpatrick 
64be691f3bSpatrick void CommandObjectThreadTraceStartIntelPT::CommandOptions::
OptionParsingStarting(ExecutionContext * execution_context)65be691f3bSpatrick     OptionParsingStarting(ExecutionContext *execution_context) {
66*f6aab3d8Srobert   m_ipt_trace_size = kDefaultIptTraceSize;
67be691f3bSpatrick   m_enable_tsc = kDefaultEnableTscValue;
68be691f3bSpatrick   m_psb_period = kDefaultPsbPeriod;
69be691f3bSpatrick }
70be691f3bSpatrick 
71be691f3bSpatrick llvm::ArrayRef<OptionDefinition>
GetDefinitions()72be691f3bSpatrick CommandObjectThreadTraceStartIntelPT::CommandOptions::GetDefinitions() {
73*f6aab3d8Srobert   return llvm::ArrayRef(g_thread_trace_start_intel_pt_options);
74be691f3bSpatrick }
75be691f3bSpatrick 
DoExecuteOnThreads(Args & command,CommandReturnObject & result,llvm::ArrayRef<lldb::tid_t> tids)76be691f3bSpatrick bool CommandObjectThreadTraceStartIntelPT::DoExecuteOnThreads(
77be691f3bSpatrick     Args &command, CommandReturnObject &result,
78be691f3bSpatrick     llvm::ArrayRef<lldb::tid_t> tids) {
79*f6aab3d8Srobert   if (Error err = m_trace.Start(tids, m_options.m_ipt_trace_size,
80be691f3bSpatrick                                 m_options.m_enable_tsc, m_options.m_psb_period))
81be691f3bSpatrick     result.SetError(Status(std::move(err)));
82be691f3bSpatrick   else
83be691f3bSpatrick     result.SetStatus(eReturnStatusSuccessFinishResult);
84be691f3bSpatrick 
85be691f3bSpatrick   return result.Succeeded();
86be691f3bSpatrick }
87be691f3bSpatrick 
88be691f3bSpatrick /// CommandObjectProcessTraceStartIntelPT
89be691f3bSpatrick 
90be691f3bSpatrick #define LLDB_OPTIONS_process_trace_start_intel_pt
91be691f3bSpatrick #include "TraceIntelPTCommandOptions.inc"
92be691f3bSpatrick 
SetOptionValue(uint32_t option_idx,llvm::StringRef option_arg,ExecutionContext * execution_context)93be691f3bSpatrick Status CommandObjectProcessTraceStartIntelPT::CommandOptions::SetOptionValue(
94be691f3bSpatrick     uint32_t option_idx, llvm::StringRef option_arg,
95be691f3bSpatrick     ExecutionContext *execution_context) {
96be691f3bSpatrick   Status error;
97be691f3bSpatrick   const int short_option = m_getopt_table[option_idx].val;
98be691f3bSpatrick 
99be691f3bSpatrick   switch (short_option) {
100be691f3bSpatrick   case 's': {
101*f6aab3d8Srobert     if (std::optional<uint64_t> bytes =
102*f6aab3d8Srobert             ParsingUtils::ParseUserFriendlySizeExpression(option_arg))
103*f6aab3d8Srobert       m_ipt_trace_size = *bytes;
104be691f3bSpatrick     else
105*f6aab3d8Srobert       error.SetErrorStringWithFormat("invalid bytes expression for '%s'",
106*f6aab3d8Srobert                                      option_arg.str().c_str());
107be691f3bSpatrick     break;
108be691f3bSpatrick   }
109be691f3bSpatrick   case 'l': {
110*f6aab3d8Srobert     if (std::optional<uint64_t> bytes =
111*f6aab3d8Srobert             ParsingUtils::ParseUserFriendlySizeExpression(option_arg))
112*f6aab3d8Srobert       m_process_buffer_size_limit = *bytes;
113be691f3bSpatrick     else
114*f6aab3d8Srobert       error.SetErrorStringWithFormat("invalid bytes expression for '%s'",
115*f6aab3d8Srobert                                      option_arg.str().c_str());
116be691f3bSpatrick     break;
117be691f3bSpatrick   }
118be691f3bSpatrick   case 't': {
119be691f3bSpatrick     m_enable_tsc = true;
120be691f3bSpatrick     break;
121be691f3bSpatrick   }
122*f6aab3d8Srobert   case 'c': {
123*f6aab3d8Srobert     m_per_cpu_tracing = true;
124*f6aab3d8Srobert     break;
125*f6aab3d8Srobert   }
126*f6aab3d8Srobert   case 'd': {
127*f6aab3d8Srobert     m_disable_cgroup_filtering = true;
128*f6aab3d8Srobert     break;
129*f6aab3d8Srobert   }
130be691f3bSpatrick   case 'p': {
131be691f3bSpatrick     int64_t psb_period;
132be691f3bSpatrick     if (option_arg.empty() || option_arg.getAsInteger(0, psb_period) ||
133be691f3bSpatrick         psb_period < 0)
134be691f3bSpatrick       error.SetErrorStringWithFormat("invalid integer value for option '%s'",
135be691f3bSpatrick                                      option_arg.str().c_str());
136be691f3bSpatrick     else
137be691f3bSpatrick       m_psb_period = psb_period;
138be691f3bSpatrick     break;
139be691f3bSpatrick   }
140be691f3bSpatrick   default:
141be691f3bSpatrick     llvm_unreachable("Unimplemented option");
142be691f3bSpatrick   }
143be691f3bSpatrick   return error;
144be691f3bSpatrick }
145be691f3bSpatrick 
146be691f3bSpatrick void CommandObjectProcessTraceStartIntelPT::CommandOptions::
OptionParsingStarting(ExecutionContext * execution_context)147be691f3bSpatrick     OptionParsingStarting(ExecutionContext *execution_context) {
148*f6aab3d8Srobert   m_ipt_trace_size = kDefaultIptTraceSize;
149be691f3bSpatrick   m_process_buffer_size_limit = kDefaultProcessBufferSizeLimit;
150be691f3bSpatrick   m_enable_tsc = kDefaultEnableTscValue;
151be691f3bSpatrick   m_psb_period = kDefaultPsbPeriod;
152*f6aab3d8Srobert   m_per_cpu_tracing = kDefaultPerCpuTracing;
153*f6aab3d8Srobert   m_disable_cgroup_filtering = kDefaultDisableCgroupFiltering;
154be691f3bSpatrick }
155be691f3bSpatrick 
156be691f3bSpatrick llvm::ArrayRef<OptionDefinition>
GetDefinitions()157be691f3bSpatrick CommandObjectProcessTraceStartIntelPT::CommandOptions::GetDefinitions() {
158*f6aab3d8Srobert   return llvm::ArrayRef(g_process_trace_start_intel_pt_options);
159be691f3bSpatrick }
160be691f3bSpatrick 
DoExecute(Args & command,CommandReturnObject & result)161be691f3bSpatrick bool CommandObjectProcessTraceStartIntelPT::DoExecute(
162be691f3bSpatrick     Args &command, CommandReturnObject &result) {
163*f6aab3d8Srobert   if (Error err = m_trace.Start(
164*f6aab3d8Srobert           m_options.m_ipt_trace_size, m_options.m_process_buffer_size_limit,
165*f6aab3d8Srobert           m_options.m_enable_tsc, m_options.m_psb_period,
166*f6aab3d8Srobert           m_options.m_per_cpu_tracing, m_options.m_disable_cgroup_filtering))
167be691f3bSpatrick     result.SetError(Status(std::move(err)));
168be691f3bSpatrick   else
169be691f3bSpatrick     result.SetStatus(eReturnStatusSuccessFinishResult);
170be691f3bSpatrick 
171be691f3bSpatrick   return result.Succeeded();
172be691f3bSpatrick }
173*f6aab3d8Srobert 
174*f6aab3d8Srobert std::optional<uint64_t>
ParseUserFriendlySizeExpression(llvm::StringRef size_expression)175*f6aab3d8Srobert ParsingUtils::ParseUserFriendlySizeExpression(llvm::StringRef size_expression) {
176*f6aab3d8Srobert   if (size_expression.empty()) {
177*f6aab3d8Srobert     return std::nullopt;
178*f6aab3d8Srobert   }
179*f6aab3d8Srobert   const uint64_t kBytesMultiplier = 1;
180*f6aab3d8Srobert   const uint64_t kKibiBytesMultiplier = 1024;
181*f6aab3d8Srobert   const uint64_t kMebiBytesMultiplier = 1024 * 1024;
182*f6aab3d8Srobert 
183*f6aab3d8Srobert   DenseMap<StringRef, uint64_t> multipliers = {
184*f6aab3d8Srobert       {"mib", kMebiBytesMultiplier}, {"mb", kMebiBytesMultiplier},
185*f6aab3d8Srobert       {"m", kMebiBytesMultiplier},   {"kib", kKibiBytesMultiplier},
186*f6aab3d8Srobert       {"kb", kKibiBytesMultiplier},  {"k", kKibiBytesMultiplier},
187*f6aab3d8Srobert       {"b", kBytesMultiplier},       {"", kBytesMultiplier}};
188*f6aab3d8Srobert 
189*f6aab3d8Srobert   const auto non_digit_index = size_expression.find_first_not_of("0123456789");
190*f6aab3d8Srobert   if (non_digit_index == 0) { // expression starts from from non-digit char.
191*f6aab3d8Srobert     return std::nullopt;
192*f6aab3d8Srobert   }
193*f6aab3d8Srobert 
194*f6aab3d8Srobert   const llvm::StringRef number_part =
195*f6aab3d8Srobert       non_digit_index == llvm::StringRef::npos
196*f6aab3d8Srobert           ? size_expression
197*f6aab3d8Srobert           : size_expression.substr(0, non_digit_index);
198*f6aab3d8Srobert   uint64_t parsed_number;
199*f6aab3d8Srobert   if (number_part.getAsInteger(10, parsed_number)) {
200*f6aab3d8Srobert     return std::nullopt;
201*f6aab3d8Srobert   }
202*f6aab3d8Srobert 
203*f6aab3d8Srobert   if (non_digit_index != llvm::StringRef::npos) { // if expression has units.
204*f6aab3d8Srobert     const auto multiplier = size_expression.substr(non_digit_index).lower();
205*f6aab3d8Srobert 
206*f6aab3d8Srobert     auto it = multipliers.find(multiplier);
207*f6aab3d8Srobert     if (it == multipliers.end())
208*f6aab3d8Srobert       return std::nullopt;
209*f6aab3d8Srobert 
210*f6aab3d8Srobert     return parsed_number * it->second;
211*f6aab3d8Srobert   } else {
212*f6aab3d8Srobert     return parsed_number;
213*f6aab3d8Srobert   }
214*f6aab3d8Srobert }
215