//===-- SBCommandInterpreter.cpp --------------------------------*- C++ -*-===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "lldb/lldb-types.h" #include "lldb/Core/SourceManager.h" #include "lldb/Core/Listener.h" #include "lldb/Interpreter/CommandInterpreter.h" #include "lldb/Interpreter/CommandObjectMultiword.h" #include "lldb/Interpreter/CommandReturnObject.h" #include "lldb/Target/Target.h" #include "lldb/API/SBBroadcaster.h" #include "lldb/API/SBCommandReturnObject.h" #include "lldb/API/SBCommandInterpreter.h" #include "lldb/API/SBEvent.h" #include "lldb/API/SBExecutionContext.h" #include "lldb/API/SBProcess.h" #include "lldb/API/SBTarget.h" #include "lldb/API/SBListener.h" #include "lldb/API/SBStream.h" #include "lldb/API/SBStringList.h" using namespace lldb; using namespace lldb_private; SBCommandInterpreterRunOptions::SBCommandInterpreterRunOptions() { m_opaque_up.reset(new CommandInterpreterRunOptions()); } SBCommandInterpreterRunOptions::~SBCommandInterpreterRunOptions() { } bool SBCommandInterpreterRunOptions::GetStopOnContinue () const { return m_opaque_up->GetStopOnContinue(); } void SBCommandInterpreterRunOptions::SetStopOnContinue (bool stop_on_continue) { m_opaque_up->SetStopOnContinue(stop_on_continue); } bool SBCommandInterpreterRunOptions::GetStopOnError () const { return m_opaque_up->GetStopOnError(); } void SBCommandInterpreterRunOptions::SetStopOnError (bool stop_on_error) { m_opaque_up->SetStopOnError(stop_on_error); } bool SBCommandInterpreterRunOptions::GetStopOnCrash () const { return m_opaque_up->GetStopOnCrash(); } void SBCommandInterpreterRunOptions::SetStopOnCrash (bool stop_on_crash) { m_opaque_up->SetStopOnCrash(stop_on_crash); } bool SBCommandInterpreterRunOptions::GetEchoCommands () const { return m_opaque_up->GetEchoCommands(); } void SBCommandInterpreterRunOptions::SetEchoCommands (bool echo_commands) { m_opaque_up->SetEchoCommands(echo_commands); } bool SBCommandInterpreterRunOptions::GetPrintResults () const { return m_opaque_up->GetPrintResults(); } void SBCommandInterpreterRunOptions::SetPrintResults (bool print_results) { m_opaque_up->SetPrintResults(print_results); } bool SBCommandInterpreterRunOptions::GetAddToHistory () const { return m_opaque_up->GetAddToHistory(); } void SBCommandInterpreterRunOptions::SetAddToHistory (bool add_to_history) { m_opaque_up->SetAddToHistory(add_to_history); } lldb_private::CommandInterpreterRunOptions * SBCommandInterpreterRunOptions::get () const { return m_opaque_up.get(); } lldb_private::CommandInterpreterRunOptions & SBCommandInterpreterRunOptions::ref () const { return *m_opaque_up.get(); } class CommandPluginInterfaceImplementation : public CommandObjectParsed { public: CommandPluginInterfaceImplementation (CommandInterpreter &interpreter, const char *name, lldb::SBCommandPluginInterface* backend, const char *help = NULL, const char *syntax = NULL, uint32_t flags = 0) : CommandObjectParsed (interpreter, name, help, syntax, flags), m_backend(backend) {} virtual bool IsRemovable() const { return true; } protected: virtual bool DoExecute (Args& command, CommandReturnObject &result) { SBCommandReturnObject sb_return(&result); SBCommandInterpreter sb_interpreter(&m_interpreter); SBDebugger debugger_sb(m_interpreter.GetDebugger().shared_from_this()); bool ret = m_backend->DoExecute (debugger_sb,(char**)command.GetArgumentVector(), sb_return); sb_return.Release(); return ret; } lldb::SBCommandPluginInterface* m_backend; }; SBCommandInterpreter::SBCommandInterpreter (CommandInterpreter *interpreter) : m_opaque_ptr (interpreter) { Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); if (log) log->Printf ("SBCommandInterpreter::SBCommandInterpreter (interpreter=%p)" " => SBCommandInterpreter(%p)", static_cast(interpreter), static_cast(m_opaque_ptr)); } SBCommandInterpreter::SBCommandInterpreter(const SBCommandInterpreter &rhs) : m_opaque_ptr (rhs.m_opaque_ptr) { } const SBCommandInterpreter & SBCommandInterpreter::operator = (const SBCommandInterpreter &rhs) { m_opaque_ptr = rhs.m_opaque_ptr; return *this; } SBCommandInterpreter::~SBCommandInterpreter () { } bool SBCommandInterpreter::IsValid() const { return m_opaque_ptr != NULL; } bool SBCommandInterpreter::CommandExists (const char *cmd) { if (cmd && m_opaque_ptr) return m_opaque_ptr->CommandExists (cmd); return false; } bool SBCommandInterpreter::AliasExists (const char *cmd) { if (cmd && m_opaque_ptr) return m_opaque_ptr->AliasExists (cmd); return false; } bool SBCommandInterpreter::IsActive () { if (m_opaque_ptr) return m_opaque_ptr->IsActive (); return false; } const char * SBCommandInterpreter::GetIOHandlerControlSequence(char ch) { if (m_opaque_ptr) return m_opaque_ptr->GetDebugger().GetTopIOHandlerControlSequence (ch).GetCString(); return NULL; } lldb::ReturnStatus SBCommandInterpreter::HandleCommand (const char *command_line, SBCommandReturnObject &result, bool add_to_history) { SBExecutionContext sb_exe_ctx; return HandleCommand (command_line, sb_exe_ctx, result, add_to_history); } lldb::ReturnStatus SBCommandInterpreter::HandleCommand (const char *command_line, SBExecutionContext &override_context, SBCommandReturnObject &result, bool add_to_history) { Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); if (log) log->Printf ("SBCommandInterpreter(%p)::HandleCommand (command=\"%s\", SBCommandReturnObject(%p), add_to_history=%i)", static_cast(m_opaque_ptr), command_line, static_cast(result.get()), add_to_history); ExecutionContext ctx, *ctx_ptr; if (override_context.get()) { ctx = override_context.get()->Lock(true); ctx_ptr = &ctx; } else ctx_ptr = nullptr; result.Clear(); if (command_line && m_opaque_ptr) { result.ref().SetInteractive(false); m_opaque_ptr->HandleCommand (command_line, add_to_history ? eLazyBoolYes : eLazyBoolNo, result.ref(), ctx_ptr); } else { result->AppendError ("SBCommandInterpreter or the command line is not valid"); result->SetStatus (eReturnStatusFailed); } // We need to get the value again, in case the command disabled the log! log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) { SBStream sstr; result.GetDescription (sstr); log->Printf ("SBCommandInterpreter(%p)::HandleCommand (command=\"%s\", SBCommandReturnObject(%p): %s, add_to_history=%i) => %i", static_cast(m_opaque_ptr), command_line, static_cast(result.get()), sstr.GetData(), add_to_history, result.GetStatus()); } return result.GetStatus(); } void SBCommandInterpreter::HandleCommandsFromFile (lldb::SBFileSpec &file, lldb::SBExecutionContext &override_context, lldb::SBCommandInterpreterRunOptions &options, lldb::SBCommandReturnObject result) { Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); if (log) { SBStream s; file.GetDescription (s); log->Printf ("SBCommandInterpreter(%p)::HandleCommandsFromFile (file=\"%s\", SBCommandReturnObject(%p))", static_cast(m_opaque_ptr), s.GetData(), static_cast(result.get())); } if (!m_opaque_ptr) { result->AppendError ("SBCommandInterpreter is not valid."); result->SetStatus (eReturnStatusFailed); return; } if (!file.IsValid()) { SBStream s; file.GetDescription (s); result->AppendErrorWithFormat ("File is not valid: %s.", s.GetData()); result->SetStatus (eReturnStatusFailed); } FileSpec tmp_spec = file.ref(); ExecutionContext ctx, *ctx_ptr; if (override_context.get()) { ctx = override_context.get()->Lock(true); ctx_ptr = &ctx; } else ctx_ptr = nullptr; m_opaque_ptr->HandleCommandsFromFile (tmp_spec, ctx_ptr, options.ref(), result.ref()); } int SBCommandInterpreter::HandleCompletion (const char *current_line, const char *cursor, const char *last_char, int match_start_point, int max_return_elements, SBStringList &matches) { Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); int num_completions = 0; // Sanity check the arguments that are passed in: // cursor & last_char have to be within the current_line. if (current_line == NULL || cursor == NULL || last_char == NULL) return 0; if (cursor < current_line || last_char < current_line) return 0; size_t current_line_size = strlen (current_line); if (cursor - current_line > static_cast(current_line_size) || last_char - current_line > static_cast(current_line_size)) return 0; if (log) log->Printf ("SBCommandInterpreter(%p)::HandleCompletion (current_line=\"%s\", cursor at: %" PRId64 ", last char at: %" PRId64 ", match_start_point: %d, max_return_elements: %d)", static_cast(m_opaque_ptr), current_line, static_cast(cursor - current_line), static_cast(last_char - current_line), match_start_point, max_return_elements); if (m_opaque_ptr) { lldb_private::StringList lldb_matches; num_completions = m_opaque_ptr->HandleCompletion (current_line, cursor, last_char, match_start_point, max_return_elements, lldb_matches); SBStringList temp_list (&lldb_matches); matches.AppendList (temp_list); } if (log) log->Printf ("SBCommandInterpreter(%p)::HandleCompletion - Found %d completions.", static_cast(m_opaque_ptr), num_completions); return num_completions; } int SBCommandInterpreter::HandleCompletion (const char *current_line, uint32_t cursor_pos, int match_start_point, int max_return_elements, lldb::SBStringList &matches) { const char *cursor = current_line + cursor_pos; const char *last_char = current_line + strlen (current_line); return HandleCompletion (current_line, cursor, last_char, match_start_point, max_return_elements, matches); } bool SBCommandInterpreter::HasCommands () { if (m_opaque_ptr) return m_opaque_ptr->HasCommands(); return false; } bool SBCommandInterpreter::HasAliases () { if (m_opaque_ptr) return m_opaque_ptr->HasAliases(); return false; } bool SBCommandInterpreter::HasAliasOptions () { if (m_opaque_ptr) return m_opaque_ptr->HasAliasOptions (); return false; } SBProcess SBCommandInterpreter::GetProcess () { SBProcess sb_process; ProcessSP process_sp; if (m_opaque_ptr) { TargetSP target_sp(m_opaque_ptr->GetDebugger().GetSelectedTarget()); if (target_sp) { Mutex::Locker api_locker(target_sp->GetAPIMutex()); process_sp = target_sp->GetProcessSP(); sb_process.SetSP(process_sp); } } Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); if (log) log->Printf ("SBCommandInterpreter(%p)::GetProcess () => SBProcess(%p)", static_cast(m_opaque_ptr), static_cast(process_sp.get())); return sb_process; } SBDebugger SBCommandInterpreter::GetDebugger () { SBDebugger sb_debugger; if (m_opaque_ptr) sb_debugger.reset(m_opaque_ptr->GetDebugger().shared_from_this()); Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); if (log) log->Printf ("SBCommandInterpreter(%p)::GetDebugger () => SBDebugger(%p)", static_cast(m_opaque_ptr), static_cast(sb_debugger.get())); return sb_debugger; } bool SBCommandInterpreter::GetPromptOnQuit() { if (m_opaque_ptr) return m_opaque_ptr->GetPromptOnQuit(); return false; } void SBCommandInterpreter::SetPromptOnQuit (bool b) { if (m_opaque_ptr) m_opaque_ptr->SetPromptOnQuit(b); } void SBCommandInterpreter::ResolveCommand(const char *command_line, SBCommandReturnObject &result) { result.Clear(); if (command_line && m_opaque_ptr) { m_opaque_ptr->ResolveCommand(command_line, result.ref()); } else { result->AppendError("SBCommandInterpreter or the command line is not valid"); result->SetStatus(eReturnStatusFailed); } } CommandInterpreter * SBCommandInterpreter::get () { return m_opaque_ptr; } CommandInterpreter & SBCommandInterpreter::ref () { assert (m_opaque_ptr); return *m_opaque_ptr; } void SBCommandInterpreter::reset (lldb_private::CommandInterpreter *interpreter) { m_opaque_ptr = interpreter; } void SBCommandInterpreter::SourceInitFileInHomeDirectory (SBCommandReturnObject &result) { result.Clear(); if (m_opaque_ptr) { TargetSP target_sp(m_opaque_ptr->GetDebugger().GetSelectedTarget()); Mutex::Locker api_locker; if (target_sp) api_locker.Lock(target_sp->GetAPIMutex()); m_opaque_ptr->SourceInitFile (false, result.ref()); } else { result->AppendError ("SBCommandInterpreter is not valid"); result->SetStatus (eReturnStatusFailed); } Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); if (log) log->Printf ("SBCommandInterpreter(%p)::SourceInitFileInHomeDirectory (&SBCommandReturnObject(%p))", static_cast(m_opaque_ptr), static_cast(result.get())); } void SBCommandInterpreter::SourceInitFileInCurrentWorkingDirectory (SBCommandReturnObject &result) { result.Clear(); if (m_opaque_ptr) { TargetSP target_sp(m_opaque_ptr->GetDebugger().GetSelectedTarget()); Mutex::Locker api_locker; if (target_sp) api_locker.Lock(target_sp->GetAPIMutex()); m_opaque_ptr->SourceInitFile (true, result.ref()); } else { result->AppendError ("SBCommandInterpreter is not valid"); result->SetStatus (eReturnStatusFailed); } Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); if (log) log->Printf ("SBCommandInterpreter(%p)::SourceInitFileInCurrentWorkingDirectory (&SBCommandReturnObject(%p))", static_cast(m_opaque_ptr), static_cast(result.get())); } SBBroadcaster SBCommandInterpreter::GetBroadcaster () { Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API)); SBBroadcaster broadcaster (m_opaque_ptr, false); if (log) log->Printf ("SBCommandInterpreter(%p)::GetBroadcaster() => SBBroadcaster(%p)", static_cast(m_opaque_ptr), static_cast(broadcaster.get())); return broadcaster; } const char * SBCommandInterpreter::GetBroadcasterClass () { return CommandInterpreter::GetStaticBroadcasterClass().AsCString(); } const char * SBCommandInterpreter::GetArgumentTypeAsCString (const lldb::CommandArgumentType arg_type) { return CommandObject::GetArgumentTypeAsCString (arg_type); } const char * SBCommandInterpreter::GetArgumentDescriptionAsCString (const lldb::CommandArgumentType arg_type) { return CommandObject::GetArgumentDescriptionAsCString (arg_type); } bool SBCommandInterpreter::EventIsCommandInterpreterEvent (const lldb::SBEvent &event) { return event.GetBroadcasterClass() == SBCommandInterpreter::GetBroadcasterClass(); } bool SBCommandInterpreter::SetCommandOverrideCallback (const char *command_name, lldb::CommandOverrideCallback callback, void *baton) { if (command_name && command_name[0] && m_opaque_ptr) { std::string command_name_str (command_name); CommandObject *cmd_obj = m_opaque_ptr->GetCommandObjectForCommand(command_name_str); if (cmd_obj) { assert(command_name_str.empty()); cmd_obj->SetOverrideCallback (callback, baton); return true; } } return false; } lldb::SBCommand SBCommandInterpreter::AddMultiwordCommand (const char* name, const char* help) { CommandObjectMultiword *new_command = new CommandObjectMultiword(*m_opaque_ptr,name,help); new_command->SetRemovable (true); lldb::CommandObjectSP new_command_sp(new_command); if (new_command_sp && m_opaque_ptr->AddUserCommand(name, new_command_sp, true)) return lldb::SBCommand(new_command_sp); return lldb::SBCommand(); } lldb::SBCommand SBCommandInterpreter::AddCommand (const char* name, lldb::SBCommandPluginInterface* impl, const char* help) { lldb::CommandObjectSP new_command_sp; new_command_sp.reset(new CommandPluginInterfaceImplementation(*m_opaque_ptr,name,impl,help)); if (new_command_sp && m_opaque_ptr->AddUserCommand(name, new_command_sp, true)) return lldb::SBCommand(new_command_sp); return lldb::SBCommand(); } SBCommand::SBCommand () {} SBCommand::SBCommand (lldb::CommandObjectSP cmd_sp) : m_opaque_sp (cmd_sp) {} bool SBCommand::IsValid () { return (bool)m_opaque_sp; } const char* SBCommand::GetName () { if (IsValid ()) return m_opaque_sp->GetCommandName (); return NULL; } const char* SBCommand::GetHelp () { if (IsValid ()) return m_opaque_sp->GetHelp (); return NULL; } const char* SBCommand::GetHelpLong () { if (IsValid ()) return m_opaque_sp->GetHelpLong (); return NULL; } void SBCommand::SetHelp (const char* help) { if (IsValid()) m_opaque_sp->SetHelp(help); } void SBCommand::SetHelpLong (const char* help) { if (IsValid()) m_opaque_sp->SetHelpLong(help); } lldb::SBCommand SBCommand::AddMultiwordCommand (const char* name, const char* help) { if (!IsValid ()) return lldb::SBCommand(); if (m_opaque_sp->IsMultiwordObject() == false) return lldb::SBCommand(); CommandObjectMultiword *new_command = new CommandObjectMultiword(m_opaque_sp->GetCommandInterpreter(),name,help); new_command->SetRemovable (true); lldb::CommandObjectSP new_command_sp(new_command); if (new_command_sp && m_opaque_sp->LoadSubCommand(name,new_command_sp)) return lldb::SBCommand(new_command_sp); return lldb::SBCommand(); } lldb::SBCommand SBCommand::AddCommand (const char* name, lldb::SBCommandPluginInterface *impl, const char* help) { if (!IsValid ()) return lldb::SBCommand(); if (m_opaque_sp->IsMultiwordObject() == false) return lldb::SBCommand(); lldb::CommandObjectSP new_command_sp; new_command_sp.reset(new CommandPluginInterfaceImplementation(m_opaque_sp->GetCommandInterpreter(),name,impl,help)); if (new_command_sp && m_opaque_sp->LoadSubCommand(name,new_command_sp)) return lldb::SBCommand(new_command_sp); return lldb::SBCommand(); } uint32_t SBCommand::GetFlags () { if (!IsValid()) return 0; return m_opaque_sp->GetFlags().Get(); } void SBCommand::SetFlags (uint32_t flags) { if (IsValid()) m_opaque_sp->GetFlags().Set(flags); }