//===-- CommandObjectScripting.cpp ----------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "CommandObjectScripting.h" #include "lldb/Core/Debugger.h" #include "lldb/Core/PluginManager.h" #include "lldb/DataFormatters/DataVisualization.h" #include "lldb/Host/Config.h" #include "lldb/Host/OptionParser.h" #include "lldb/Interpreter/CommandInterpreter.h" #include "lldb/Interpreter/CommandOptionArgumentTable.h" #include "lldb/Interpreter/CommandReturnObject.h" #include "lldb/Interpreter/Interfaces/ScriptedInterfaceUsages.h" #include "lldb/Interpreter/OptionArgParser.h" #include "lldb/Interpreter/ScriptInterpreter.h" #include "lldb/Utility/Args.h" using namespace lldb; using namespace lldb_private; #define LLDB_OPTIONS_scripting_run #include "CommandOptions.inc" class CommandObjectScriptingRun : public CommandObjectRaw { public: CommandObjectScriptingRun(CommandInterpreter &interpreter) : CommandObjectRaw( interpreter, "scripting run", "Invoke the script interpreter with provided code and display any " "results. Start the interactive interpreter if no code is " "supplied.", "scripting run [--language --] " "[]") {} ~CommandObjectScriptingRun() override = default; Options *GetOptions() override { return &m_options; } class CommandOptions : public Options { public: CommandOptions() = default; ~CommandOptions() override = default; Status SetOptionValue(uint32_t option_idx, llvm::StringRef option_arg, ExecutionContext *execution_context) override { Status error; const int short_option = m_getopt_table[option_idx].val; switch (short_option) { case 'l': language = (lldb::ScriptLanguage)OptionArgParser::ToOptionEnum( option_arg, GetDefinitions()[option_idx].enum_values, eScriptLanguageNone, error); if (!error.Success()) error = Status::FromErrorStringWithFormat( "unrecognized value for language '%s'", option_arg.str().c_str()); break; default: llvm_unreachable("Unimplemented option"); } return error; } void OptionParsingStarting(ExecutionContext *execution_context) override { language = lldb::eScriptLanguageNone; } llvm::ArrayRef GetDefinitions() override { return llvm::ArrayRef(g_scripting_run_options); } lldb::ScriptLanguage language = lldb::eScriptLanguageNone; }; protected: void DoExecute(llvm::StringRef command, CommandReturnObject &result) override { // Try parsing the language option but when the command contains a raw part // separated by the -- delimiter. OptionsWithRaw raw_args(command); if (raw_args.HasArgs()) { if (!ParseOptions(raw_args.GetArgs(), result)) return; command = raw_args.GetRawPart(); } lldb::ScriptLanguage language = (m_options.language == lldb::eScriptLanguageNone) ? m_interpreter.GetDebugger().GetScriptLanguage() : m_options.language; if (language == lldb::eScriptLanguageNone) { result.AppendError( "the script-lang setting is set to none - scripting not available"); return; } ScriptInterpreter *script_interpreter = GetDebugger().GetScriptInterpreter(true, language); if (script_interpreter == nullptr) { result.AppendError("no script interpreter"); return; } // Script might change Python code we use for formatting. Make sure we keep // up to date with it. DataVisualization::ForceUpdate(); if (command.empty()) { script_interpreter->ExecuteInterpreterLoop(); result.SetStatus(eReturnStatusSuccessFinishNoResult); return; } // We can do better when reporting the status of one-liner script execution. if (script_interpreter->ExecuteOneLine(command, &result)) result.SetStatus(eReturnStatusSuccessFinishNoResult); else result.SetStatus(eReturnStatusFailed); } private: CommandOptions m_options; }; #define LLDB_OPTIONS_scripting_extension_list #include "CommandOptions.inc" class CommandObjectScriptingExtensionList : public CommandObjectParsed { public: CommandObjectScriptingExtensionList(CommandInterpreter &interpreter) : CommandObjectParsed( interpreter, "scripting extension list", "List all the available scripting extension templates. ", "scripting template list [--language --]") {} ~CommandObjectScriptingExtensionList() override = default; Options *GetOptions() override { return &m_options; } class CommandOptions : public Options { public: CommandOptions() = default; ~CommandOptions() override = default; Status SetOptionValue(uint32_t option_idx, llvm::StringRef option_arg, ExecutionContext *execution_context) override { Status error; const int short_option = m_getopt_table[option_idx].val; switch (short_option) { case 'l': m_language = (lldb::ScriptLanguage)OptionArgParser::ToOptionEnum( option_arg, GetDefinitions()[option_idx].enum_values, eScriptLanguageNone, error); if (!error.Success()) error = Status::FromErrorStringWithFormatv( "unrecognized value for language '{0}'", option_arg); break; default: llvm_unreachable("Unimplemented option"); } return error; } void OptionParsingStarting(ExecutionContext *execution_context) override { m_language = lldb::eScriptLanguageDefault; } llvm::ArrayRef GetDefinitions() override { return llvm::ArrayRef(g_scripting_extension_list_options); } lldb::ScriptLanguage m_language = lldb::eScriptLanguageDefault; }; protected: void DoExecute(Args &command, CommandReturnObject &result) override { Stream &s = result.GetOutputStream(); s.Printf("Available scripted extension templates:"); auto print_field = [&s](llvm::StringRef key, llvm::StringRef value) { if (!value.empty()) { s.IndentMore(); s.Indent(); s << key << ": " << value << '\n'; s.IndentLess(); } }; size_t num_listed_interface = 0; size_t num_extensions = PluginManager::GetNumScriptedInterfaces(); for (size_t i = 0; i < num_extensions; i++) { llvm::StringRef plugin_name = PluginManager::GetScriptedInterfaceNameAtIndex(i); if (plugin_name.empty()) break; lldb::ScriptLanguage lang = PluginManager::GetScriptedInterfaceLanguageAtIndex(i); if (lang != m_options.m_language) continue; if (!num_listed_interface) s.EOL(); num_listed_interface++; llvm::StringRef desc = PluginManager::GetScriptedInterfaceDescriptionAtIndex(i); ScriptedInterfaceUsages usages = PluginManager::GetScriptedInterfaceUsagesAtIndex(i); print_field("Name", plugin_name); print_field("Language", ScriptInterpreter::LanguageToString(lang)); print_field("Description", desc); usages.Dump(s, ScriptedInterfaceUsages::UsageKind::API); usages.Dump(s, ScriptedInterfaceUsages::UsageKind::CommandInterpreter); if (i != num_extensions - 1) s.EOL(); } if (!num_listed_interface) s << " None\n"; } private: CommandOptions m_options; }; class CommandObjectMultiwordScriptingExtension : public CommandObjectMultiword { public: CommandObjectMultiwordScriptingExtension(CommandInterpreter &interpreter) : CommandObjectMultiword( interpreter, "scripting extension", "Commands for operating on the scripting extensions.", "scripting extension []") { LoadSubCommand( "list", CommandObjectSP(new CommandObjectScriptingExtensionList(interpreter))); } ~CommandObjectMultiwordScriptingExtension() override = default; }; CommandObjectMultiwordScripting::CommandObjectMultiwordScripting( CommandInterpreter &interpreter) : CommandObjectMultiword( interpreter, "scripting", "Commands for operating on the scripting functionalities.", "scripting []") { LoadSubCommand("run", CommandObjectSP(new CommandObjectScriptingRun(interpreter))); LoadSubCommand("extension", CommandObjectSP(new CommandObjectMultiwordScriptingExtension( interpreter))); } CommandObjectMultiwordScripting::~CommandObjectMultiwordScripting() = default;