174c93956SWalter Erquinigo //===-- CommandObjectTrace.cpp --------------------------------------------===//
274c93956SWalter Erquinigo //
374c93956SWalter Erquinigo // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
474c93956SWalter Erquinigo // See https://llvm.org/LICENSE.txt for license information.
574c93956SWalter Erquinigo // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
674c93956SWalter Erquinigo //
774c93956SWalter Erquinigo //===----------------------------------------------------------------------===//
874c93956SWalter Erquinigo 
974c93956SWalter Erquinigo #include "CommandObjectTrace.h"
1074c93956SWalter Erquinigo 
1174c93956SWalter Erquinigo #include "llvm/Support/JSON.h"
1274c93956SWalter Erquinigo #include "llvm/Support/MemoryBuffer.h"
1374c93956SWalter Erquinigo 
1474c93956SWalter Erquinigo #include "lldb/Core/Debugger.h"
15ea1f4974SWalter Erquinigo #include "lldb/Core/PluginManager.h"
1674c93956SWalter Erquinigo #include "lldb/Host/OptionParser.h"
1774c93956SWalter Erquinigo #include "lldb/Interpreter/CommandInterpreter.h"
1874c93956SWalter Erquinigo #include "lldb/Interpreter/CommandObject.h"
19*7ced9fffSJonas Devlieghere #include "lldb/Interpreter/CommandOptionArgumentTable.h"
2074c93956SWalter Erquinigo #include "lldb/Interpreter/CommandReturnObject.h"
2174c93956SWalter Erquinigo #include "lldb/Interpreter/OptionArgParser.h"
2274c93956SWalter Erquinigo #include "lldb/Interpreter/OptionGroupFormat.h"
2374c93956SWalter Erquinigo #include "lldb/Interpreter/OptionValueBoolean.h"
2474c93956SWalter Erquinigo #include "lldb/Interpreter/OptionValueLanguage.h"
2574c93956SWalter Erquinigo #include "lldb/Interpreter/OptionValueString.h"
2674c93956SWalter Erquinigo #include "lldb/Interpreter/Options.h"
270b697561SWalter Erquinigo #include "lldb/Target/Process.h"
2874c93956SWalter Erquinigo #include "lldb/Target/Trace.h"
2974c93956SWalter Erquinigo 
3074c93956SWalter Erquinigo using namespace lldb;
3174c93956SWalter Erquinigo using namespace lldb_private;
3274c93956SWalter Erquinigo using namespace llvm;
3374c93956SWalter Erquinigo 
34b532dd54SWalter Erquinigo // CommandObjectTraceSave
35b532dd54SWalter Erquinigo #define LLDB_OPTIONS_trace_save
36b532dd54SWalter Erquinigo #include "CommandOptions.inc"
37b532dd54SWalter Erquinigo 
38b532dd54SWalter Erquinigo #pragma mark CommandObjectTraceSave
39b532dd54SWalter Erquinigo 
40b532dd54SWalter Erquinigo class CommandObjectTraceSave : public CommandObjectParsed {
41b532dd54SWalter Erquinigo public:
42b532dd54SWalter Erquinigo   class CommandOptions : public Options {
43b532dd54SWalter Erquinigo   public:
CommandOptions()44b532dd54SWalter Erquinigo     CommandOptions() { OptionParsingStarting(nullptr); }
45b532dd54SWalter Erquinigo 
SetOptionValue(uint32_t option_idx,llvm::StringRef option_arg,ExecutionContext * execution_context)46b532dd54SWalter Erquinigo     Status SetOptionValue(uint32_t option_idx, llvm::StringRef option_arg,
47b532dd54SWalter Erquinigo                           ExecutionContext *execution_context) override {
48b532dd54SWalter Erquinigo       Status error;
49b532dd54SWalter Erquinigo       const int short_option = m_getopt_table[option_idx].val;
50b532dd54SWalter Erquinigo 
51b532dd54SWalter Erquinigo       switch (short_option) {
52b532dd54SWalter Erquinigo       case 'c': {
53b532dd54SWalter Erquinigo         m_compact = true;
54b532dd54SWalter Erquinigo         break;
55b532dd54SWalter Erquinigo       }
56b532dd54SWalter Erquinigo       default:
57b532dd54SWalter Erquinigo         llvm_unreachable("Unimplemented option");
58b532dd54SWalter Erquinigo       }
59b532dd54SWalter Erquinigo       return error;
60b532dd54SWalter Erquinigo     }
61b532dd54SWalter Erquinigo 
OptionParsingStarting(ExecutionContext * execution_context)62b532dd54SWalter Erquinigo     void OptionParsingStarting(ExecutionContext *execution_context) override {
63b532dd54SWalter Erquinigo       m_compact = false;
64b532dd54SWalter Erquinigo     };
65b532dd54SWalter Erquinigo 
GetDefinitions()66b532dd54SWalter Erquinigo     llvm::ArrayRef<OptionDefinition> GetDefinitions() override {
67b532dd54SWalter Erquinigo       return llvm::makeArrayRef(g_trace_save_options);
68b532dd54SWalter Erquinigo     };
69b532dd54SWalter Erquinigo 
70b532dd54SWalter Erquinigo     bool m_compact;
71b532dd54SWalter Erquinigo   };
72b532dd54SWalter Erquinigo 
GetOptions()73b532dd54SWalter Erquinigo   Options *GetOptions() override { return &m_options; }
74b532dd54SWalter Erquinigo 
CommandObjectTraceSave(CommandInterpreter & interpreter)75b532dd54SWalter Erquinigo   CommandObjectTraceSave(CommandInterpreter &interpreter)
76b532dd54SWalter Erquinigo       : CommandObjectParsed(
77b532dd54SWalter Erquinigo             interpreter, "trace save",
78b532dd54SWalter Erquinigo             "Save the trace of the current target in the specified directory, "
79b532dd54SWalter Erquinigo             "which will be created if needed. "
80b532dd54SWalter Erquinigo             "This directory will contain a trace bundle, with all the "
81b532dd54SWalter Erquinigo             "necessary files the reconstruct the trace session even on a "
82b532dd54SWalter Erquinigo             "different computer. "
83b532dd54SWalter Erquinigo             "Part of this bundle is the bundle description file with the name "
84b532dd54SWalter Erquinigo             "trace.json. This file can be used by the \"trace load\" command "
85b532dd54SWalter Erquinigo             "to load this trace in LLDB."
86b532dd54SWalter Erquinigo             "Note: if the current target contains information of multiple "
87b532dd54SWalter Erquinigo             "processes or targets, they all will be included in the bundle.",
88b532dd54SWalter Erquinigo             "trace save [<cmd-options>] <bundle_directory>",
89b532dd54SWalter Erquinigo             eCommandRequiresProcess | eCommandTryTargetAPILock |
90b532dd54SWalter Erquinigo                 eCommandProcessMustBeLaunched | eCommandProcessMustBePaused |
91b532dd54SWalter Erquinigo                 eCommandProcessMustBeTraced) {
92b532dd54SWalter Erquinigo     CommandArgumentData bundle_dir{eArgTypeDirectoryName, eArgRepeatPlain};
93b532dd54SWalter Erquinigo     m_arguments.push_back({bundle_dir});
94b532dd54SWalter Erquinigo   }
95b532dd54SWalter Erquinigo 
96b532dd54SWalter Erquinigo   void
HandleArgumentCompletion(CompletionRequest & request,OptionElementVector & opt_element_vector)97b532dd54SWalter Erquinigo   HandleArgumentCompletion(CompletionRequest &request,
98b532dd54SWalter Erquinigo                            OptionElementVector &opt_element_vector) override {
99b532dd54SWalter Erquinigo     CommandCompletions::InvokeCommonCompletionCallbacks(
100b532dd54SWalter Erquinigo         GetCommandInterpreter(), CommandCompletions::eDiskFileCompletion,
101b532dd54SWalter Erquinigo         request, nullptr);
102b532dd54SWalter Erquinigo   }
103b532dd54SWalter Erquinigo 
104b532dd54SWalter Erquinigo   ~CommandObjectTraceSave() override = default;
105b532dd54SWalter Erquinigo 
106b532dd54SWalter Erquinigo protected:
DoExecute(Args & command,CommandReturnObject & result)107b532dd54SWalter Erquinigo   bool DoExecute(Args &command, CommandReturnObject &result) override {
108b532dd54SWalter Erquinigo     if (command.size() != 1) {
109b532dd54SWalter Erquinigo       result.AppendError("a single path to a directory where the trace bundle "
110b532dd54SWalter Erquinigo                          "will be created is required");
111b532dd54SWalter Erquinigo       return false;
112b532dd54SWalter Erquinigo     }
113b532dd54SWalter Erquinigo 
114b532dd54SWalter Erquinigo     FileSpec bundle_dir(command[0].ref());
115b532dd54SWalter Erquinigo     FileSystem::Instance().Resolve(bundle_dir);
116b532dd54SWalter Erquinigo 
117b532dd54SWalter Erquinigo     ProcessSP process_sp = m_exe_ctx.GetProcessSP();
118b532dd54SWalter Erquinigo 
119b532dd54SWalter Erquinigo     TraceSP trace_sp = process_sp->GetTarget().GetTrace();
120b532dd54SWalter Erquinigo 
121b532dd54SWalter Erquinigo     if (llvm::Expected<FileSpec> desc_file =
122b532dd54SWalter Erquinigo             trace_sp->SaveToDisk(bundle_dir, m_options.m_compact)) {
123b532dd54SWalter Erquinigo       result.AppendMessageWithFormatv(
124b532dd54SWalter Erquinigo           "Trace bundle description file written to: {0}", *desc_file);
125b532dd54SWalter Erquinigo       result.SetStatus(eReturnStatusSuccessFinishResult);
126b532dd54SWalter Erquinigo     } else {
127b532dd54SWalter Erquinigo       result.AppendError(toString(desc_file.takeError()));
128b532dd54SWalter Erquinigo     }
129b532dd54SWalter Erquinigo 
130b532dd54SWalter Erquinigo     return result.Succeeded();
131b532dd54SWalter Erquinigo   }
132b532dd54SWalter Erquinigo 
133b532dd54SWalter Erquinigo   CommandOptions m_options;
134b532dd54SWalter Erquinigo };
135b532dd54SWalter Erquinigo 
13674c93956SWalter Erquinigo // CommandObjectTraceLoad
13774c93956SWalter Erquinigo #define LLDB_OPTIONS_trace_load
13874c93956SWalter Erquinigo #include "CommandOptions.inc"
13974c93956SWalter Erquinigo 
14074c93956SWalter Erquinigo #pragma mark CommandObjectTraceLoad
14174c93956SWalter Erquinigo 
14274c93956SWalter Erquinigo class CommandObjectTraceLoad : public CommandObjectParsed {
14374c93956SWalter Erquinigo public:
14474c93956SWalter Erquinigo   class CommandOptions : public Options {
14574c93956SWalter Erquinigo   public:
CommandOptions()146abb0ed44SKazu Hirata     CommandOptions() { OptionParsingStarting(nullptr); }
14774c93956SWalter Erquinigo 
14874c93956SWalter Erquinigo     ~CommandOptions() override = default;
14974c93956SWalter Erquinigo 
SetOptionValue(uint32_t option_idx,StringRef option_arg,ExecutionContext * execution_context)15074c93956SWalter Erquinigo     Status SetOptionValue(uint32_t option_idx, StringRef option_arg,
15174c93956SWalter Erquinigo                           ExecutionContext *execution_context) override {
15274c93956SWalter Erquinigo       Status error;
15374c93956SWalter Erquinigo       const int short_option = m_getopt_table[option_idx].val;
15474c93956SWalter Erquinigo 
15574c93956SWalter Erquinigo       switch (short_option) {
15674c93956SWalter Erquinigo       case 'v': {
15774c93956SWalter Erquinigo         m_verbose = true;
15874c93956SWalter Erquinigo         break;
15974c93956SWalter Erquinigo       }
16074c93956SWalter Erquinigo       default:
16174c93956SWalter Erquinigo         llvm_unreachable("Unimplemented option");
16274c93956SWalter Erquinigo       }
16374c93956SWalter Erquinigo       return error;
16474c93956SWalter Erquinigo     }
16574c93956SWalter Erquinigo 
OptionParsingStarting(ExecutionContext * execution_context)16674c93956SWalter Erquinigo     void OptionParsingStarting(ExecutionContext *execution_context) override {
16774c93956SWalter Erquinigo       m_verbose = false;
16874c93956SWalter Erquinigo     }
16974c93956SWalter Erquinigo 
GetDefinitions()17074c93956SWalter Erquinigo     ArrayRef<OptionDefinition> GetDefinitions() override {
17174c93956SWalter Erquinigo       return makeArrayRef(g_trace_load_options);
17274c93956SWalter Erquinigo     }
17374c93956SWalter Erquinigo 
17474c93956SWalter Erquinigo     bool m_verbose; // Enable verbose logging for debugging purposes.
17574c93956SWalter Erquinigo   };
17674c93956SWalter Erquinigo 
CommandObjectTraceLoad(CommandInterpreter & interpreter)17774c93956SWalter Erquinigo   CommandObjectTraceLoad(CommandInterpreter &interpreter)
178b8dcd0baSWalter Erquinigo       : CommandObjectParsed(
179b8dcd0baSWalter Erquinigo             interpreter, "trace load",
180b8dcd0baSWalter Erquinigo             "Load a post-mortem processor trace session from a trace bundle.",
181b532dd54SWalter Erquinigo             "trace load <trace_description_file>") {
182b532dd54SWalter Erquinigo     CommandArgumentData session_file_arg{eArgTypeFilename, eArgRepeatPlain};
183c1b07d61SJim Ingham     m_arguments.push_back({session_file_arg});
184c1b07d61SJim Ingham   }
18574c93956SWalter Erquinigo 
186b532dd54SWalter Erquinigo   void
HandleArgumentCompletion(CompletionRequest & request,OptionElementVector & opt_element_vector)187b532dd54SWalter Erquinigo   HandleArgumentCompletion(CompletionRequest &request,
188b532dd54SWalter Erquinigo                            OptionElementVector &opt_element_vector) override {
189b532dd54SWalter Erquinigo     CommandCompletions::InvokeCommonCompletionCallbacks(
190b532dd54SWalter Erquinigo         GetCommandInterpreter(), CommandCompletions::eDiskFileCompletion,
191b532dd54SWalter Erquinigo         request, nullptr);
192b532dd54SWalter Erquinigo   }
193b532dd54SWalter Erquinigo 
19474c93956SWalter Erquinigo   ~CommandObjectTraceLoad() override = default;
19574c93956SWalter Erquinigo 
GetOptions()19674c93956SWalter Erquinigo   Options *GetOptions() override { return &m_options; }
19774c93956SWalter Erquinigo 
19874c93956SWalter Erquinigo protected:
DoExecute(Args & command,CommandReturnObject & result)19974c93956SWalter Erquinigo   bool DoExecute(Args &command, CommandReturnObject &result) override {
20074c93956SWalter Erquinigo     if (command.size() != 1) {
201b8dcd0baSWalter Erquinigo       result.AppendError("a single path to a JSON file containing a the "
202b8dcd0baSWalter Erquinigo                          "description of the trace bundle is required");
20374c93956SWalter Erquinigo       return false;
20474c93956SWalter Erquinigo     }
20574c93956SWalter Erquinigo 
20650f93679SJakob Johnson     const FileSpec trace_description_file(command[0].ref());
20750f93679SJakob Johnson 
20850f93679SJakob Johnson     llvm::Expected<lldb::TraceSP> trace_or_err =
20950f93679SJakob Johnson         Trace::LoadPostMortemTraceFromFile(GetDebugger(),
21050f93679SJakob Johnson                                            trace_description_file);
21150f93679SJakob Johnson 
21250f93679SJakob Johnson     if (!trace_or_err) {
21350f93679SJakob Johnson       result.AppendErrorWithFormat(
21450f93679SJakob Johnson           "%s\n", llvm::toString(trace_or_err.takeError()).c_str());
21574c93956SWalter Erquinigo       return false;
21674c93956SWalter Erquinigo     }
21774c93956SWalter Erquinigo 
21850f93679SJakob Johnson     if (m_options.m_verbose) {
219a3939e15SPavel Labath       result.AppendMessageWithFormatv("loading trace with plugin {0}\n",
22050f93679SJakob Johnson                                       trace_or_err.get()->GetPluginName());
22150f93679SJakob Johnson     }
22274c93956SWalter Erquinigo 
22374c93956SWalter Erquinigo     result.SetStatus(eReturnStatusSuccessFinishResult);
22474c93956SWalter Erquinigo     return true;
22574c93956SWalter Erquinigo   }
22674c93956SWalter Erquinigo 
22774c93956SWalter Erquinigo   CommandOptions m_options;
22874c93956SWalter Erquinigo };
22974c93956SWalter Erquinigo 
23074c93956SWalter Erquinigo // CommandObjectTraceDump
23174c93956SWalter Erquinigo #define LLDB_OPTIONS_trace_dump
23274c93956SWalter Erquinigo #include "CommandOptions.inc"
23374c93956SWalter Erquinigo 
23474c93956SWalter Erquinigo #pragma mark CommandObjectTraceDump
23574c93956SWalter Erquinigo 
23674c93956SWalter Erquinigo class CommandObjectTraceDump : public CommandObjectParsed {
23774c93956SWalter Erquinigo public:
23874c93956SWalter Erquinigo   class CommandOptions : public Options {
23974c93956SWalter Erquinigo   public:
CommandOptions()240abb0ed44SKazu Hirata     CommandOptions() { OptionParsingStarting(nullptr); }
24174c93956SWalter Erquinigo 
24274c93956SWalter Erquinigo     ~CommandOptions() override = default;
24374c93956SWalter Erquinigo 
SetOptionValue(uint32_t option_idx,llvm::StringRef option_arg,ExecutionContext * execution_context)24474c93956SWalter Erquinigo     Status SetOptionValue(uint32_t option_idx, llvm::StringRef option_arg,
24574c93956SWalter Erquinigo                           ExecutionContext *execution_context) override {
24674c93956SWalter Erquinigo       Status error;
24774c93956SWalter Erquinigo       const int short_option = m_getopt_table[option_idx].val;
24874c93956SWalter Erquinigo 
24974c93956SWalter Erquinigo       switch (short_option) {
25074c93956SWalter Erquinigo       case 'v': {
25174c93956SWalter Erquinigo         m_verbose = true;
25274c93956SWalter Erquinigo         break;
25374c93956SWalter Erquinigo       }
25474c93956SWalter Erquinigo       default:
25574c93956SWalter Erquinigo         llvm_unreachable("Unimplemented option");
25674c93956SWalter Erquinigo       }
25774c93956SWalter Erquinigo       return error;
25874c93956SWalter Erquinigo     }
25974c93956SWalter Erquinigo 
OptionParsingStarting(ExecutionContext * execution_context)26074c93956SWalter Erquinigo     void OptionParsingStarting(ExecutionContext *execution_context) override {
26174c93956SWalter Erquinigo       m_verbose = false;
26274c93956SWalter Erquinigo     }
26374c93956SWalter Erquinigo 
GetDefinitions()26474c93956SWalter Erquinigo     llvm::ArrayRef<OptionDefinition> GetDefinitions() override {
26574c93956SWalter Erquinigo       return llvm::makeArrayRef(g_trace_dump_options);
26674c93956SWalter Erquinigo     }
26774c93956SWalter Erquinigo 
26874c93956SWalter Erquinigo     bool m_verbose; // Enable verbose logging for debugging purposes.
26974c93956SWalter Erquinigo   };
27074c93956SWalter Erquinigo 
CommandObjectTraceDump(CommandInterpreter & interpreter)27174c93956SWalter Erquinigo   CommandObjectTraceDump(CommandInterpreter &interpreter)
27274c93956SWalter Erquinigo       : CommandObjectParsed(interpreter, "trace dump",
27374c93956SWalter Erquinigo                             "Dump the loaded processor trace data.",
274abb0ed44SKazu Hirata                             "trace dump") {}
27574c93956SWalter Erquinigo 
27674c93956SWalter Erquinigo   ~CommandObjectTraceDump() override = default;
27774c93956SWalter Erquinigo 
GetOptions()27874c93956SWalter Erquinigo   Options *GetOptions() override { return &m_options; }
27974c93956SWalter Erquinigo 
28074c93956SWalter Erquinigo protected:
DoExecute(Args & command,CommandReturnObject & result)28174c93956SWalter Erquinigo   bool DoExecute(Args &command, CommandReturnObject &result) override {
28274c93956SWalter Erquinigo     Status error;
28374c93956SWalter Erquinigo     // TODO: fill in the dumping code here!
28474c93956SWalter Erquinigo     if (error.Success()) {
28574c93956SWalter Erquinigo       result.SetStatus(eReturnStatusSuccessFinishResult);
28674c93956SWalter Erquinigo     } else {
28774c93956SWalter Erquinigo       result.AppendErrorWithFormat("%s\n", error.AsCString());
28874c93956SWalter Erquinigo     }
28974c93956SWalter Erquinigo     return result.Succeeded();
29074c93956SWalter Erquinigo   }
29174c93956SWalter Erquinigo 
29274c93956SWalter Erquinigo   CommandOptions m_options;
29374c93956SWalter Erquinigo };
29474c93956SWalter Erquinigo 
29574c93956SWalter Erquinigo // CommandObjectTraceSchema
29674c93956SWalter Erquinigo #define LLDB_OPTIONS_trace_schema
29774c93956SWalter Erquinigo #include "CommandOptions.inc"
29874c93956SWalter Erquinigo 
29974c93956SWalter Erquinigo #pragma mark CommandObjectTraceSchema
30074c93956SWalter Erquinigo 
30174c93956SWalter Erquinigo class CommandObjectTraceSchema : public CommandObjectParsed {
30274c93956SWalter Erquinigo public:
30374c93956SWalter Erquinigo   class CommandOptions : public Options {
30474c93956SWalter Erquinigo   public:
CommandOptions()305abb0ed44SKazu Hirata     CommandOptions() { OptionParsingStarting(nullptr); }
30674c93956SWalter Erquinigo 
30774c93956SWalter Erquinigo     ~CommandOptions() override = default;
30874c93956SWalter Erquinigo 
SetOptionValue(uint32_t option_idx,llvm::StringRef option_arg,ExecutionContext * execution_context)30974c93956SWalter Erquinigo     Status SetOptionValue(uint32_t option_idx, llvm::StringRef option_arg,
31074c93956SWalter Erquinigo                           ExecutionContext *execution_context) override {
31174c93956SWalter Erquinigo       Status error;
31274c93956SWalter Erquinigo       const int short_option = m_getopt_table[option_idx].val;
31374c93956SWalter Erquinigo 
31474c93956SWalter Erquinigo       switch (short_option) {
31574c93956SWalter Erquinigo       case 'v': {
31674c93956SWalter Erquinigo         m_verbose = true;
31774c93956SWalter Erquinigo         break;
31874c93956SWalter Erquinigo       }
31974c93956SWalter Erquinigo       default:
32074c93956SWalter Erquinigo         llvm_unreachable("Unimplemented option");
32174c93956SWalter Erquinigo       }
32274c93956SWalter Erquinigo       return error;
32374c93956SWalter Erquinigo     }
32474c93956SWalter Erquinigo 
OptionParsingStarting(ExecutionContext * execution_context)32574c93956SWalter Erquinigo     void OptionParsingStarting(ExecutionContext *execution_context) override {
32674c93956SWalter Erquinigo       m_verbose = false;
32774c93956SWalter Erquinigo     }
32874c93956SWalter Erquinigo 
GetDefinitions()32974c93956SWalter Erquinigo     llvm::ArrayRef<OptionDefinition> GetDefinitions() override {
33074c93956SWalter Erquinigo       return llvm::makeArrayRef(g_trace_schema_options);
33174c93956SWalter Erquinigo     }
33274c93956SWalter Erquinigo 
33374c93956SWalter Erquinigo     bool m_verbose; // Enable verbose logging for debugging purposes.
33474c93956SWalter Erquinigo   };
33574c93956SWalter Erquinigo 
CommandObjectTraceSchema(CommandInterpreter & interpreter)33674c93956SWalter Erquinigo   CommandObjectTraceSchema(CommandInterpreter &interpreter)
33774c93956SWalter Erquinigo       : CommandObjectParsed(interpreter, "trace schema",
33874c93956SWalter Erquinigo                             "Show the schema of the given trace plugin.",
339ea1f4974SWalter Erquinigo                             "trace schema <plug-in>. Use the plug-in name "
340c1b07d61SJim Ingham                             "\"all\" to see all schemas.\n") {
341c1b07d61SJim Ingham     CommandArgumentData plugin_arg{eArgTypeNone, eArgRepeatPlain};
342c1b07d61SJim Ingham     m_arguments.push_back({plugin_arg});
343c1b07d61SJim Ingham   }
34474c93956SWalter Erquinigo 
34574c93956SWalter Erquinigo   ~CommandObjectTraceSchema() override = default;
34674c93956SWalter Erquinigo 
GetOptions()34774c93956SWalter Erquinigo   Options *GetOptions() override { return &m_options; }
34874c93956SWalter Erquinigo 
34974c93956SWalter Erquinigo protected:
DoExecute(Args & command,CommandReturnObject & result)35074c93956SWalter Erquinigo   bool DoExecute(Args &command, CommandReturnObject &result) override {
35174c93956SWalter Erquinigo     Status error;
35274c93956SWalter Erquinigo     if (command.empty()) {
3531b1c8e4aSDavid Spickett       result.AppendError(
35474c93956SWalter Erquinigo           "trace schema cannot be invoked without a plug-in as argument");
35574c93956SWalter Erquinigo       return false;
35674c93956SWalter Erquinigo     }
35774c93956SWalter Erquinigo 
35874c93956SWalter Erquinigo     StringRef plugin_name(command[0].c_str());
359ea1f4974SWalter Erquinigo     if (plugin_name == "all") {
360ea1f4974SWalter Erquinigo       size_t index = 0;
361ea1f4974SWalter Erquinigo       while (true) {
362ea1f4974SWalter Erquinigo         StringRef schema = PluginManager::GetTraceSchema(index++);
363ea1f4974SWalter Erquinigo         if (schema.empty())
364ea1f4974SWalter Erquinigo           break;
36574c93956SWalter Erquinigo 
366ea1f4974SWalter Erquinigo         result.AppendMessage(schema);
367ea1f4974SWalter Erquinigo       }
36874c93956SWalter Erquinigo     } else {
369ea1f4974SWalter Erquinigo       if (Expected<StringRef> schemaOrErr =
370ea1f4974SWalter Erquinigo               Trace::FindPluginSchema(plugin_name))
371ea1f4974SWalter Erquinigo         result.AppendMessage(*schemaOrErr);
372ea1f4974SWalter Erquinigo       else
373ea1f4974SWalter Erquinigo         error = schemaOrErr.takeError();
37474c93956SWalter Erquinigo     }
37574c93956SWalter Erquinigo 
37674c93956SWalter Erquinigo     if (error.Success()) {
37774c93956SWalter Erquinigo       result.SetStatus(eReturnStatusSuccessFinishResult);
37874c93956SWalter Erquinigo     } else {
37974c93956SWalter Erquinigo       result.AppendErrorWithFormat("%s\n", error.AsCString());
38074c93956SWalter Erquinigo     }
38174c93956SWalter Erquinigo     return result.Succeeded();
38274c93956SWalter Erquinigo   }
38374c93956SWalter Erquinigo 
38474c93956SWalter Erquinigo   CommandOptions m_options;
38574c93956SWalter Erquinigo };
38674c93956SWalter Erquinigo 
38774c93956SWalter Erquinigo // CommandObjectTrace
38874c93956SWalter Erquinigo 
CommandObjectTrace(CommandInterpreter & interpreter)38974c93956SWalter Erquinigo CommandObjectTrace::CommandObjectTrace(CommandInterpreter &interpreter)
39074c93956SWalter Erquinigo     : CommandObjectMultiword(interpreter, "trace",
39174c93956SWalter Erquinigo                              "Commands for loading and using processor "
39274c93956SWalter Erquinigo                              "trace information.",
39374c93956SWalter Erquinigo                              "trace [<sub-command-options>]") {
39474c93956SWalter Erquinigo   LoadSubCommand("load",
39574c93956SWalter Erquinigo                  CommandObjectSP(new CommandObjectTraceLoad(interpreter)));
39674c93956SWalter Erquinigo   LoadSubCommand("dump",
39774c93956SWalter Erquinigo                  CommandObjectSP(new CommandObjectTraceDump(interpreter)));
398b532dd54SWalter Erquinigo   LoadSubCommand("save",
399b532dd54SWalter Erquinigo                  CommandObjectSP(new CommandObjectTraceSave(interpreter)));
40074c93956SWalter Erquinigo   LoadSubCommand("schema",
40174c93956SWalter Erquinigo                  CommandObjectSP(new CommandObjectTraceSchema(interpreter)));
40274c93956SWalter Erquinigo }
40374c93956SWalter Erquinigo 
40474c93956SWalter Erquinigo CommandObjectTrace::~CommandObjectTrace() = default;
4050b697561SWalter Erquinigo 
DoGetProxyCommandObject()4060b697561SWalter Erquinigo Expected<CommandObjectSP> CommandObjectTraceProxy::DoGetProxyCommandObject() {
4070b697561SWalter Erquinigo   ProcessSP process_sp = m_interpreter.GetExecutionContext().GetProcessSP();
4080b697561SWalter Erquinigo 
4090b697561SWalter Erquinigo   if (!process_sp)
4100b697561SWalter Erquinigo     return createStringError(inconvertibleErrorCode(),
4110b697561SWalter Erquinigo                              "Process not available.");
4120b697561SWalter Erquinigo   if (m_live_debug_session_only && !process_sp->IsLiveDebugSession())
4130b697561SWalter Erquinigo     return createStringError(inconvertibleErrorCode(),
4140b697561SWalter Erquinigo                              "Process must be alive.");
4150b697561SWalter Erquinigo 
416bf9f21a2SWalter Erquinigo   if (Expected<TraceSP> trace_sp = process_sp->GetTarget().GetTraceOrCreate())
4170b697561SWalter Erquinigo     return GetDelegateCommand(**trace_sp);
4180b697561SWalter Erquinigo   else
4190b697561SWalter Erquinigo     return createStringError(inconvertibleErrorCode(),
4200b697561SWalter Erquinigo                              "Tracing is not supported. %s",
4210b697561SWalter Erquinigo                              toString(trace_sp.takeError()).c_str());
4220b697561SWalter Erquinigo }
4230b697561SWalter Erquinigo 
GetProxyCommandObject()4240b697561SWalter Erquinigo CommandObject *CommandObjectTraceProxy::GetProxyCommandObject() {
4250b697561SWalter Erquinigo   if (Expected<CommandObjectSP> delegate = DoGetProxyCommandObject()) {
4260b697561SWalter Erquinigo     m_delegate_sp = *delegate;
4270b697561SWalter Erquinigo     m_delegate_error.clear();
4280b697561SWalter Erquinigo     return m_delegate_sp.get();
4290b697561SWalter Erquinigo   } else {
4300b697561SWalter Erquinigo     m_delegate_sp.reset();
4310b697561SWalter Erquinigo     m_delegate_error = toString(delegate.takeError());
4320b697561SWalter Erquinigo     return nullptr;
4330b697561SWalter Erquinigo   }
4340b697561SWalter Erquinigo }
435