130fdc8d8SChris Lattner //===-- CommandObjectProcess.cpp --------------------------------*- C++ -*-===// 230fdc8d8SChris Lattner // 330fdc8d8SChris Lattner // The LLVM Compiler Infrastructure 430fdc8d8SChris Lattner // 530fdc8d8SChris Lattner // This file is distributed under the University of Illinois Open Source 630fdc8d8SChris Lattner // License. See LICENSE.TXT for details. 730fdc8d8SChris Lattner // 830fdc8d8SChris Lattner //===----------------------------------------------------------------------===// 930fdc8d8SChris Lattner 1093a64300SDaniel Malea #include "lldb/lldb-python.h" 1193a64300SDaniel Malea 1230fdc8d8SChris Lattner #include "CommandObjectProcess.h" 1330fdc8d8SChris Lattner 1430fdc8d8SChris Lattner // C Includes 1530fdc8d8SChris Lattner // C++ Includes 1630fdc8d8SChris Lattner // Other libraries and framework includes 1730fdc8d8SChris Lattner // Project includes 180e41084aSJim Ingham #include "lldb/Breakpoint/Breakpoint.h" 190e41084aSJim Ingham #include "lldb/Breakpoint/BreakpointLocation.h" 200e41084aSJim Ingham #include "lldb/Breakpoint/BreakpointSite.h" 2130fdc8d8SChris Lattner #include "lldb/Core/State.h" 221f746071SGreg Clayton #include "lldb/Core/Module.h" 237260f620SGreg Clayton #include "lldb/Host/Host.h" 240e41084aSJim Ingham #include "lldb/Interpreter/Args.h" 250e41084aSJim Ingham #include "lldb/Interpreter/Options.h" 2630fdc8d8SChris Lattner #include "lldb/Interpreter/CommandInterpreter.h" 2730fdc8d8SChris Lattner #include "lldb/Interpreter/CommandReturnObject.h" 28e996fd30SGreg Clayton #include "lldb/Target/Platform.h" 2930fdc8d8SChris Lattner #include "lldb/Target/Process.h" 300e41084aSJim Ingham #include "lldb/Target/StopInfo.h" 3130fdc8d8SChris Lattner #include "lldb/Target/Target.h" 3230fdc8d8SChris Lattner #include "lldb/Target/Thread.h" 3330fdc8d8SChris Lattner 3430fdc8d8SChris Lattner using namespace lldb; 3530fdc8d8SChris Lattner using namespace lldb_private; 3630fdc8d8SChris Lattner 3730fdc8d8SChris Lattner //------------------------------------------------------------------------- 3830fdc8d8SChris Lattner // CommandObjectProcessLaunch 3930fdc8d8SChris Lattner //------------------------------------------------------------------------- 404bddaeb5SJim Ingham #pragma mark CommandObjectProcessLaunch 415a988416SJim Ingham class CommandObjectProcessLaunch : public CommandObjectParsed 4230fdc8d8SChris Lattner { 4330fdc8d8SChris Lattner public: 4430fdc8d8SChris Lattner 45a7015092SGreg Clayton CommandObjectProcessLaunch (CommandInterpreter &interpreter) : 465a988416SJim Ingham CommandObjectParsed (interpreter, 47a7015092SGreg Clayton "process launch", 48e3d26315SCaroline Tice "Launch the executable in the debugger.", 49f9fc609fSGreg Clayton NULL, 50f9fc609fSGreg Clayton eFlagRequiresTarget), 51eb0103f2SGreg Clayton m_options (interpreter) 5230fdc8d8SChris Lattner { 53405fe67fSCaroline Tice CommandArgumentEntry arg; 54405fe67fSCaroline Tice CommandArgumentData run_args_arg; 55405fe67fSCaroline Tice 56405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 57405fe67fSCaroline Tice run_args_arg.arg_type = eArgTypeRunArgs; 58405fe67fSCaroline Tice run_args_arg.arg_repetition = eArgRepeatOptional; 59405fe67fSCaroline Tice 60405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 61405fe67fSCaroline Tice arg.push_back (run_args_arg); 62405fe67fSCaroline Tice 63405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 64405fe67fSCaroline Tice m_arguments.push_back (arg); 6530fdc8d8SChris Lattner } 6630fdc8d8SChris Lattner 6730fdc8d8SChris Lattner 6830fdc8d8SChris Lattner ~CommandObjectProcessLaunch () 6930fdc8d8SChris Lattner { 7030fdc8d8SChris Lattner } 7130fdc8d8SChris Lattner 72e9ce62b6SJim Ingham int 73e9ce62b6SJim Ingham HandleArgumentCompletion (Args &input, 74e9ce62b6SJim Ingham int &cursor_index, 75e9ce62b6SJim Ingham int &cursor_char_position, 76e9ce62b6SJim Ingham OptionElementVector &opt_element_vector, 77e9ce62b6SJim Ingham int match_start_point, 78e9ce62b6SJim Ingham int max_return_elements, 79e9ce62b6SJim Ingham bool &word_complete, 80e9ce62b6SJim Ingham StringList &matches) 81e9ce62b6SJim Ingham { 82e9ce62b6SJim Ingham std::string completion_str (input.GetArgumentAtIndex(cursor_index)); 83e9ce62b6SJim Ingham completion_str.erase (cursor_char_position); 84e9ce62b6SJim Ingham 85e9ce62b6SJim Ingham CommandCompletions::InvokeCommonCompletionCallbacks (m_interpreter, 86e9ce62b6SJim Ingham CommandCompletions::eDiskFileCompletion, 87e9ce62b6SJim Ingham completion_str.c_str(), 88e9ce62b6SJim Ingham match_start_point, 89e9ce62b6SJim Ingham max_return_elements, 90e9ce62b6SJim Ingham NULL, 91e9ce62b6SJim Ingham word_complete, 92e9ce62b6SJim Ingham matches); 93e9ce62b6SJim Ingham return matches.GetSize(); 94e9ce62b6SJim Ingham } 95e9ce62b6SJim Ingham 9630fdc8d8SChris Lattner Options * 9730fdc8d8SChris Lattner GetOptions () 9830fdc8d8SChris Lattner { 9930fdc8d8SChris Lattner return &m_options; 10030fdc8d8SChris Lattner } 10130fdc8d8SChris Lattner 1025a988416SJim Ingham virtual const char *GetRepeatCommand (Args ¤t_command_args, uint32_t index) 1035a988416SJim Ingham { 1045a988416SJim Ingham // No repeat for "process launch"... 1055a988416SJim Ingham return ""; 1065a988416SJim Ingham } 1075a988416SJim Ingham 1085a988416SJim Ingham protected: 10930fdc8d8SChris Lattner bool 1105a988416SJim Ingham DoExecute (Args& launch_args, CommandReturnObject &result) 11130fdc8d8SChris Lattner { 1121d885966SGreg Clayton Debugger &debugger = m_interpreter.GetDebugger(); 1131d885966SGreg Clayton Target *target = debugger.GetSelectedTarget().get(); 1141d885966SGreg Clayton Error error; 11530fdc8d8SChris Lattner // If our listener is NULL, users aren't allows to launch 11630fdc8d8SChris Lattner char filename[PATH_MAX]; 117aa149cbdSGreg Clayton const Module *exe_module = target->GetExecutableModulePointer(); 11871337622SGreg Clayton 11971337622SGreg Clayton if (exe_module == NULL) 12071337622SGreg Clayton { 121effe5c95SGreg Clayton result.AppendError ("no file in target, create a debug target using the 'target create' command"); 12271337622SGreg Clayton result.SetStatus (eReturnStatusFailed); 12371337622SGreg Clayton return false; 12471337622SGreg Clayton } 12571337622SGreg Clayton 12671337622SGreg Clayton StateType state = eStateInvalid; 127f9fc609fSGreg Clayton Process *process = m_exe_ctx.GetProcessPtr(); 12871337622SGreg Clayton if (process) 12971337622SGreg Clayton { 13071337622SGreg Clayton state = process->GetState(); 13171337622SGreg Clayton 13271337622SGreg Clayton if (process->IsAlive() && state != eStateConnected) 13330fdc8d8SChris Lattner { 134513c26ceSGreg Clayton char message[1024]; 135513c26ceSGreg Clayton if (process->GetState() == eStateAttaching) 136513c26ceSGreg Clayton ::strncpy (message, "There is a pending attach, abort it and launch a new process?", sizeof(message)); 137513c26ceSGreg Clayton else 138513c26ceSGreg Clayton ::strncpy (message, "There is a running process, kill it and restart?", sizeof(message)); 139513c26ceSGreg Clayton 140513c26ceSGreg Clayton if (!m_interpreter.Confirm (message, true)) 141bb9caf73SJim Ingham { 14230fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 14330fdc8d8SChris Lattner return false; 14430fdc8d8SChris Lattner } 145bb9caf73SJim Ingham else 146bb9caf73SJim Ingham { 1471d885966SGreg Clayton Error destroy_error (process->Destroy()); 1481d885966SGreg Clayton if (destroy_error.Success()) 149bb9caf73SJim Ingham { 150bb9caf73SJim Ingham result.SetStatus (eReturnStatusSuccessFinishResult); 151bb9caf73SJim Ingham } 152bb9caf73SJim Ingham else 153bb9caf73SJim Ingham { 1541d885966SGreg Clayton result.AppendErrorWithFormat ("Failed to kill process: %s\n", destroy_error.AsCString()); 155bb9caf73SJim Ingham result.SetStatus (eReturnStatusFailed); 156bb9caf73SJim Ingham } 157bb9caf73SJim Ingham } 158bb9caf73SJim Ingham } 15971337622SGreg Clayton } 16030fdc8d8SChris Lattner 16145392553SGreg Clayton const char *target_settings_argv0 = target->GetArg0(); 16245392553SGreg Clayton 16345392553SGreg Clayton exe_module->GetFileSpec().GetPath (filename, sizeof(filename)); 16445392553SGreg Clayton 16545392553SGreg Clayton if (target_settings_argv0) 16645392553SGreg Clayton { 16745392553SGreg Clayton m_options.launch_info.GetArguments().AppendArgument (target_settings_argv0); 16845392553SGreg Clayton m_options.launch_info.SetExecutableFile(exe_module->GetPlatformFileSpec(), false); 16945392553SGreg Clayton } 17045392553SGreg Clayton else 17145392553SGreg Clayton { 17245392553SGreg Clayton m_options.launch_info.SetExecutableFile(exe_module->GetPlatformFileSpec(), true); 17345392553SGreg Clayton } 17445392553SGreg Clayton 175144f3a9cSGreg Clayton if (launch_args.GetArgumentCount() == 0) 176144f3a9cSGreg Clayton { 17767cc0636SGreg Clayton Args target_setting_args; 17845392553SGreg Clayton if (target->GetRunArguments(target_setting_args)) 17967cc0636SGreg Clayton m_options.launch_info.GetArguments().AppendArguments (target_setting_args); 180144f3a9cSGreg Clayton } 181144f3a9cSGreg Clayton else 18230fdc8d8SChris Lattner { 18345392553SGreg Clayton m_options.launch_info.GetArguments().AppendArguments (launch_args); 18445392553SGreg Clayton 185162b597cSGreg Clayton // Save the arguments for subsequent runs in the current target. 186162b597cSGreg Clayton target->SetRunArguments (launch_args); 187982c9762SGreg Clayton } 1881d885966SGreg Clayton 189144f3a9cSGreg Clayton if (target->GetDisableASLR()) 190144f3a9cSGreg Clayton m_options.launch_info.GetFlags().Set (eLaunchFlagDisableASLR); 191144f3a9cSGreg Clayton 192144f3a9cSGreg Clayton if (target->GetDisableSTDIO()) 193144f3a9cSGreg Clayton m_options.launch_info.GetFlags().Set (eLaunchFlagDisableSTDIO); 194144f3a9cSGreg Clayton 195144f3a9cSGreg Clayton m_options.launch_info.GetFlags().Set (eLaunchFlagDebug); 196144f3a9cSGreg Clayton 197144f3a9cSGreg Clayton Args environment; 198144f3a9cSGreg Clayton target->GetEnvironmentAsArgs (environment); 199144f3a9cSGreg Clayton if (environment.GetArgumentCount() > 0) 200144f3a9cSGreg Clayton m_options.launch_info.GetEnvironmentEntries ().AppendArguments (environment); 201144f3a9cSGreg Clayton 202ee95ed50SGreg Clayton // Finalize the file actions, and if none were given, default to opening 203ee95ed50SGreg Clayton // up a pseudo terminal 204ee95ed50SGreg Clayton const bool default_to_use_pty = true; 205ee95ed50SGreg Clayton m_options.launch_info.FinalizeFileActions (target, default_to_use_pty); 2061d885966SGreg Clayton 2071d885966SGreg Clayton if (state == eStateConnected) 2081d885966SGreg Clayton { 2091d885966SGreg Clayton if (m_options.launch_info.GetFlags().Test (eLaunchFlagLaunchInTTY)) 2101d885966SGreg Clayton { 2111d885966SGreg Clayton result.AppendWarning("can't launch in tty when launching through a remote connection"); 2121d885966SGreg Clayton m_options.launch_info.GetFlags().Clear (eLaunchFlagLaunchInTTY); 2131d885966SGreg Clayton } 2141d885966SGreg Clayton } 215982c9762SGreg Clayton else 216982c9762SGreg Clayton { 217144f3a9cSGreg Clayton if (!m_options.launch_info.GetArchitecture().IsValid()) 218c9ed478aSGreg Clayton m_options.launch_info.GetArchitecture() = target->GetArchitecture(); 219c9ed478aSGreg Clayton 220c982b3d6SGreg Clayton PlatformSP platform_sp (target->GetPlatform()); 221c982b3d6SGreg Clayton 222c982b3d6SGreg Clayton if (platform_sp && platform_sp->CanDebugProcess ()) 223c982b3d6SGreg Clayton { 2241d885966SGreg Clayton process = target->GetPlatform()->DebugProcess (m_options.launch_info, 2251d885966SGreg Clayton debugger, 2261d885966SGreg Clayton target, 2271d885966SGreg Clayton debugger.GetListener(), 2281d885966SGreg Clayton error).get(); 229c982b3d6SGreg Clayton } 230c982b3d6SGreg Clayton else 231c982b3d6SGreg Clayton { 232c982b3d6SGreg Clayton const char *plugin_name = m_options.launch_info.GetProcessPluginName(); 233c3776bf2SGreg Clayton process = target->CreateProcess (debugger.GetListener(), plugin_name, NULL).get(); 234c982b3d6SGreg Clayton if (process) 235c982b3d6SGreg Clayton error = process->Launch (m_options.launch_info); 236c982b3d6SGreg Clayton } 2371d885966SGreg Clayton 2381d885966SGreg Clayton if (process == NULL) 2391d885966SGreg Clayton { 240144f3a9cSGreg Clayton result.SetError (error, "failed to launch or debug process"); 2411d885966SGreg Clayton return false; 2421d885966SGreg Clayton } 2431d885966SGreg Clayton } 24430fdc8d8SChris Lattner 24530fdc8d8SChris Lattner if (error.Success()) 24630fdc8d8SChris Lattner { 24764195a2cSGreg Clayton const char *archname = exe_module->GetArchitecture().GetArchitectureName(); 24819388cfcSGreg Clayton 249d01b2953SDaniel Malea result.AppendMessageWithFormat ("Process %" PRIu64 " launched: '%s' (%s)\n", process->GetID(), filename, archname); 25005faeb71SGreg Clayton result.SetDidChangeProcessState (true); 251982c9762SGreg Clayton if (m_options.launch_info.GetFlags().Test(eLaunchFlagStopAtEntry) == false) 25230fdc8d8SChris Lattner { 25305faeb71SGreg Clayton result.SetStatus (eReturnStatusSuccessContinuingNoResult); 25430fdc8d8SChris Lattner StateType state = process->WaitForProcessToStop (NULL); 25530fdc8d8SChris Lattner 25630fdc8d8SChris Lattner if (state == eStateStopped) 25730fdc8d8SChris Lattner { 25805faeb71SGreg Clayton error = process->Resume(); 25905faeb71SGreg Clayton if (error.Success()) 26005faeb71SGreg Clayton { 26105faeb71SGreg Clayton bool synchronous_execution = m_interpreter.GetSynchronous (); 26230fdc8d8SChris Lattner if (synchronous_execution) 26330fdc8d8SChris Lattner { 26405faeb71SGreg Clayton state = process->WaitForProcessToStop (NULL); 2652637f825SGreg Clayton const bool must_be_alive = true; 2662637f825SGreg Clayton if (!StateIsStoppedState(state, must_be_alive)) 267514487e8SGreg Clayton { 268144f3a9cSGreg Clayton result.AppendErrorWithFormat ("process isn't stopped: %s", StateAsCString(state)); 269514487e8SGreg Clayton } 27030fdc8d8SChris Lattner result.SetDidChangeProcessState (true); 27105faeb71SGreg Clayton result.SetStatus (eReturnStatusSuccessFinishResult); 27205faeb71SGreg Clayton } 27305faeb71SGreg Clayton else 27405faeb71SGreg Clayton { 27505faeb71SGreg Clayton result.SetStatus (eReturnStatusSuccessContinuingNoResult); 27605faeb71SGreg Clayton } 27705faeb71SGreg Clayton } 278514487e8SGreg Clayton else 279514487e8SGreg Clayton { 280144f3a9cSGreg Clayton result.AppendErrorWithFormat ("process resume at entry point failed: %s", error.AsCString()); 281514487e8SGreg Clayton result.SetStatus (eReturnStatusFailed); 28230fdc8d8SChris Lattner } 28330fdc8d8SChris Lattner } 284514487e8SGreg Clayton else 285514487e8SGreg Clayton { 286144f3a9cSGreg Clayton result.AppendErrorWithFormat ("initial process state wasn't stopped: %s", StateAsCString(state)); 287514487e8SGreg Clayton result.SetStatus (eReturnStatusFailed); 288514487e8SGreg Clayton } 289514487e8SGreg Clayton } 290514487e8SGreg Clayton } 291514487e8SGreg Clayton else 292514487e8SGreg Clayton { 293197bacffSGreg Clayton result.AppendErrorWithFormat ("process launch failed: %s", error.AsCString()); 294514487e8SGreg Clayton result.SetStatus (eReturnStatusFailed); 29530fdc8d8SChris Lattner } 29630fdc8d8SChris Lattner 29730fdc8d8SChris Lattner return result.Succeeded(); 29830fdc8d8SChris Lattner } 29930fdc8d8SChris Lattner 30030fdc8d8SChris Lattner protected: 301982c9762SGreg Clayton ProcessLaunchCommandOptions m_options; 30230fdc8d8SChris Lattner }; 30330fdc8d8SChris Lattner 30430fdc8d8SChris Lattner 305982c9762SGreg Clayton //#define SET1 LLDB_OPT_SET_1 306982c9762SGreg Clayton //#define SET2 LLDB_OPT_SET_2 307982c9762SGreg Clayton //#define SET3 LLDB_OPT_SET_3 308982c9762SGreg Clayton // 309982c9762SGreg Clayton //OptionDefinition 310982c9762SGreg Clayton //CommandObjectProcessLaunch::CommandOptions::g_option_table[] = 311982c9762SGreg Clayton //{ 312982c9762SGreg Clayton //{ SET1 | SET2 | SET3, false, "stop-at-entry", 's', no_argument, NULL, 0, eArgTypeNone, "Stop at the entry point of the program when launching a process."}, 3133154255fSSean Callanan //{ SET1 , false, "stdin", 'i', required_argument, NULL, 0, eArgTypeDirectoryName, "Redirect stdin for the process to <path>."}, 3143154255fSSean Callanan //{ SET1 , false, "stdout", 'o', required_argument, NULL, 0, eArgTypeDirectoryName, "Redirect stdout for the process to <path>."}, 3153154255fSSean Callanan //{ SET1 , false, "stderr", 'e', required_argument, NULL, 0, eArgTypeDirectoryName, "Redirect stderr for the process to <path>."}, 316982c9762SGreg Clayton //{ SET1 | SET2 | SET3, false, "plugin", 'p', required_argument, NULL, 0, eArgTypePlugin, "Name of the process plugin you want to use."}, 3173154255fSSean Callanan //{ SET2 , false, "tty", 't', optional_argument, NULL, 0, eArgTypeDirectoryName, "Start the process in a terminal. If <path> is specified, look for a terminal whose name contains <path>, else start the process in a new terminal."}, 318982c9762SGreg Clayton //{ SET3, false, "no-stdio", 'n', no_argument, NULL, 0, eArgTypeNone, "Do not set up for terminal I/O to go to running process."}, 3193154255fSSean Callanan //{ SET1 | SET2 | SET3, false, "working-dir", 'w', required_argument, NULL, 0, eArgTypeDirectoryName, "Set the current working directory to <path> when running the inferior."}, 320982c9762SGreg Clayton //{ 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL } 321982c9762SGreg Clayton //}; 322982c9762SGreg Clayton // 323982c9762SGreg Clayton //#undef SET1 324982c9762SGreg Clayton //#undef SET2 325982c9762SGreg Clayton //#undef SET3 32630fdc8d8SChris Lattner 32730fdc8d8SChris Lattner //------------------------------------------------------------------------- 32830fdc8d8SChris Lattner // CommandObjectProcessAttach 32930fdc8d8SChris Lattner //------------------------------------------------------------------------- 330bb9caf73SJim Ingham #pragma mark CommandObjectProcessAttach 3315a988416SJim Ingham class CommandObjectProcessAttach : public CommandObjectParsed 33230fdc8d8SChris Lattner { 33330fdc8d8SChris Lattner public: 33430fdc8d8SChris Lattner 33530fdc8d8SChris Lattner class CommandOptions : public Options 33630fdc8d8SChris Lattner { 33730fdc8d8SChris Lattner public: 33830fdc8d8SChris Lattner 339eb0103f2SGreg Clayton CommandOptions (CommandInterpreter &interpreter) : 340eb0103f2SGreg Clayton Options(interpreter) 34130fdc8d8SChris Lattner { 342f6b8b581SGreg Clayton // Keep default values of all options in one place: OptionParsingStarting () 343f6b8b581SGreg Clayton OptionParsingStarting (); 34430fdc8d8SChris Lattner } 34530fdc8d8SChris Lattner 34630fdc8d8SChris Lattner ~CommandOptions () 34730fdc8d8SChris Lattner { 34830fdc8d8SChris Lattner } 34930fdc8d8SChris Lattner 35030fdc8d8SChris Lattner Error 351f6b8b581SGreg Clayton SetOptionValue (uint32_t option_idx, const char *option_arg) 35230fdc8d8SChris Lattner { 35330fdc8d8SChris Lattner Error error; 3543bcdfc0eSGreg Clayton const int short_option = m_getopt_table[option_idx].val; 35530fdc8d8SChris Lattner bool success = false; 35630fdc8d8SChris Lattner switch (short_option) 35730fdc8d8SChris Lattner { 358a95ce623SJohnny Chen case 'c': 359a95ce623SJohnny Chen attach_info.SetContinueOnceAttached(true); 360a95ce623SJohnny Chen break; 361a95ce623SJohnny Chen 36230fdc8d8SChris Lattner case 'p': 363144f3a9cSGreg Clayton { 364144f3a9cSGreg Clayton lldb::pid_t pid = Args::StringToUInt32 (option_arg, LLDB_INVALID_PROCESS_ID, 0, &success); 36530fdc8d8SChris Lattner if (!success || pid == LLDB_INVALID_PROCESS_ID) 36630fdc8d8SChris Lattner { 36786edbf41SGreg Clayton error.SetErrorStringWithFormat("invalid process ID '%s'", option_arg); 36830fdc8d8SChris Lattner } 369144f3a9cSGreg Clayton else 370144f3a9cSGreg Clayton { 371144f3a9cSGreg Clayton attach_info.SetProcessID (pid); 372144f3a9cSGreg Clayton } 373144f3a9cSGreg Clayton } 37430fdc8d8SChris Lattner break; 37530fdc8d8SChris Lattner 37630fdc8d8SChris Lattner case 'P': 377144f3a9cSGreg Clayton attach_info.SetProcessPluginName (option_arg); 37830fdc8d8SChris Lattner break; 37930fdc8d8SChris Lattner 38030fdc8d8SChris Lattner case 'n': 381144f3a9cSGreg Clayton attach_info.GetExecutableFile().SetFile(option_arg, false); 38230fdc8d8SChris Lattner break; 38330fdc8d8SChris Lattner 38430fdc8d8SChris Lattner case 'w': 385144f3a9cSGreg Clayton attach_info.SetWaitForLaunch(true); 38630fdc8d8SChris Lattner break; 38730fdc8d8SChris Lattner 388cd16df91SJim Ingham case 'i': 389cd16df91SJim Ingham attach_info.SetIgnoreExisting(false); 390cd16df91SJim Ingham break; 391cd16df91SJim Ingham 39230fdc8d8SChris Lattner default: 39386edbf41SGreg Clayton error.SetErrorStringWithFormat("invalid short option character '%c'", short_option); 39430fdc8d8SChris Lattner break; 39530fdc8d8SChris Lattner } 39630fdc8d8SChris Lattner return error; 39730fdc8d8SChris Lattner } 39830fdc8d8SChris Lattner 39930fdc8d8SChris Lattner void 400f6b8b581SGreg Clayton OptionParsingStarting () 40130fdc8d8SChris Lattner { 402144f3a9cSGreg Clayton attach_info.Clear(); 40330fdc8d8SChris Lattner } 40430fdc8d8SChris Lattner 405e0d378b3SGreg Clayton const OptionDefinition* 40630fdc8d8SChris Lattner GetDefinitions () 40730fdc8d8SChris Lattner { 40830fdc8d8SChris Lattner return g_option_table; 40930fdc8d8SChris Lattner } 41030fdc8d8SChris Lattner 4115aee162fSJim Ingham virtual bool 412eb0103f2SGreg Clayton HandleOptionArgumentCompletion (Args &input, 4135aee162fSJim Ingham int cursor_index, 4145aee162fSJim Ingham int char_pos, 4155aee162fSJim Ingham OptionElementVector &opt_element_vector, 4165aee162fSJim Ingham int opt_element_index, 4175aee162fSJim Ingham int match_start_point, 4185aee162fSJim Ingham int max_return_elements, 4195aee162fSJim Ingham bool &word_complete, 4205aee162fSJim Ingham StringList &matches) 4215aee162fSJim Ingham { 4225aee162fSJim Ingham int opt_arg_pos = opt_element_vector[opt_element_index].opt_arg_pos; 4235aee162fSJim Ingham int opt_defs_index = opt_element_vector[opt_element_index].opt_defs_index; 4245aee162fSJim Ingham 4255aee162fSJim Ingham // We are only completing the name option for now... 4265aee162fSJim Ingham 427e0d378b3SGreg Clayton const OptionDefinition *opt_defs = GetDefinitions(); 4285aee162fSJim Ingham if (opt_defs[opt_defs_index].short_option == 'n') 4295aee162fSJim Ingham { 4305aee162fSJim Ingham // Are we in the name? 4315aee162fSJim Ingham 4325aee162fSJim Ingham // Look to see if there is a -P argument provided, and if so use that plugin, otherwise 4335aee162fSJim Ingham // use the default plugin. 4345aee162fSJim Ingham 4355aee162fSJim Ingham const char *partial_name = NULL; 4365aee162fSJim Ingham partial_name = input.GetArgumentAtIndex(opt_arg_pos); 4375aee162fSJim Ingham 4388b82f087SGreg Clayton PlatformSP platform_sp (m_interpreter.GetPlatform (true)); 439e996fd30SGreg Clayton if (platform_sp) 4405aee162fSJim Ingham { 4418b82f087SGreg Clayton ProcessInstanceInfoList process_infos; 4428b82f087SGreg Clayton ProcessInstanceInfoMatch match_info; 44332e0a750SGreg Clayton if (partial_name) 44432e0a750SGreg Clayton { 445144f3a9cSGreg Clayton match_info.GetProcessInfo().GetExecutableFile().SetFile(partial_name, false); 44632e0a750SGreg Clayton match_info.SetNameMatchType(eNameMatchStartsWith); 44732e0a750SGreg Clayton } 44832e0a750SGreg Clayton platform_sp->FindProcesses (match_info, process_infos); 449e996fd30SGreg Clayton const uint32_t num_matches = process_infos.GetSize(); 450e996fd30SGreg Clayton if (num_matches > 0) 451e996fd30SGreg Clayton { 452e996fd30SGreg Clayton for (uint32_t i=0; i<num_matches; ++i) 453e996fd30SGreg Clayton { 454e996fd30SGreg Clayton matches.AppendString (process_infos.GetProcessNameAtIndex(i), 455e996fd30SGreg Clayton process_infos.GetProcessNameLengthAtIndex(i)); 4565aee162fSJim Ingham } 4575aee162fSJim Ingham } 4585aee162fSJim Ingham } 4595aee162fSJim Ingham } 4605aee162fSJim Ingham 4615aee162fSJim Ingham return false; 4625aee162fSJim Ingham } 4635aee162fSJim Ingham 46430fdc8d8SChris Lattner // Options table: Required for subclasses of Options. 46530fdc8d8SChris Lattner 466e0d378b3SGreg Clayton static OptionDefinition g_option_table[]; 46730fdc8d8SChris Lattner 46830fdc8d8SChris Lattner // Instance variables to hold the values for command options. 46930fdc8d8SChris Lattner 470144f3a9cSGreg Clayton ProcessAttachInfo attach_info; 47130fdc8d8SChris Lattner }; 47230fdc8d8SChris Lattner 473a7015092SGreg Clayton CommandObjectProcessAttach (CommandInterpreter &interpreter) : 4745a988416SJim Ingham CommandObjectParsed (interpreter, 475a7015092SGreg Clayton "process attach", 476e3d26315SCaroline Tice "Attach to a process.", 477eb0103f2SGreg Clayton "process attach <cmd-options>"), 478eb0103f2SGreg Clayton m_options (interpreter) 4795aee162fSJim Ingham { 4805aee162fSJim Ingham } 4815aee162fSJim Ingham 4825aee162fSJim Ingham ~CommandObjectProcessAttach () 4835aee162fSJim Ingham { 4845aee162fSJim Ingham } 4855aee162fSJim Ingham 4865a988416SJim Ingham Options * 4875a988416SJim Ingham GetOptions () 4885a988416SJim Ingham { 4895a988416SJim Ingham return &m_options; 4905a988416SJim Ingham } 4915a988416SJim Ingham 4925a988416SJim Ingham protected: 4935aee162fSJim Ingham bool 4945a988416SJim Ingham DoExecute (Args& command, 4955aee162fSJim Ingham CommandReturnObject &result) 4965aee162fSJim Ingham { 497a7015092SGreg Clayton Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get(); 49831412642SJim Ingham // N.B. The attach should be synchronous. It doesn't help much to get the prompt back between initiating the attach 49931412642SJim Ingham // and the target actually stopping. So even if the interpreter is set to be asynchronous, we wait for the stop 50031412642SJim Ingham // ourselves here. 5015aee162fSJim Ingham 502f9fc609fSGreg Clayton Process *process = m_exe_ctx.GetProcessPtr(); 50371337622SGreg Clayton StateType state = eStateInvalid; 5045aee162fSJim Ingham if (process) 5055aee162fSJim Ingham { 50671337622SGreg Clayton state = process->GetState(); 50771337622SGreg Clayton if (process->IsAlive() && state != eStateConnected) 5085aee162fSJim Ingham { 509d01b2953SDaniel Malea result.AppendErrorWithFormat ("Process %" PRIu64 " is currently being debugged, kill the process before attaching.\n", 5105aee162fSJim Ingham process->GetID()); 5115aee162fSJim Ingham result.SetStatus (eReturnStatusFailed); 5125aee162fSJim Ingham return false; 5135aee162fSJim Ingham } 5145aee162fSJim Ingham } 5155aee162fSJim Ingham 5165aee162fSJim Ingham if (target == NULL) 5175aee162fSJim Ingham { 5185aee162fSJim Ingham // If there isn't a current target create one. 5195aee162fSJim Ingham TargetSP new_target_sp; 5205aee162fSJim Ingham Error error; 5215aee162fSJim Ingham 522a7015092SGreg Clayton error = m_interpreter.GetDebugger().GetTargetList().CreateTarget (m_interpreter.GetDebugger(), 523a0ca6601SGreg Clayton NULL, 524cac9c5f9SGreg Clayton NULL, 5255aee162fSJim Ingham false, 526cac9c5f9SGreg Clayton NULL, // No platform options 5275aee162fSJim Ingham new_target_sp); 5285aee162fSJim Ingham target = new_target_sp.get(); 5295aee162fSJim Ingham if (target == NULL || error.Fail()) 5305aee162fSJim Ingham { 531b766a73dSGreg Clayton result.AppendError(error.AsCString("Error creating target")); 5325aee162fSJim Ingham return false; 5335aee162fSJim Ingham } 534a7015092SGreg Clayton m_interpreter.GetDebugger().GetTargetList().SetSelectedTarget(target); 5355aee162fSJim Ingham } 5365aee162fSJim Ingham 5375aee162fSJim Ingham // Record the old executable module, we want to issue a warning if the process of attaching changed the 5385aee162fSJim Ingham // current executable (like somebody said "file foo" then attached to a PID whose executable was bar.) 5395aee162fSJim Ingham 5405aee162fSJim Ingham ModuleSP old_exec_module_sp = target->GetExecutableModule(); 5415aee162fSJim Ingham ArchSpec old_arch_spec = target->GetArchitecture(); 5425aee162fSJim Ingham 5435aee162fSJim Ingham if (command.GetArgumentCount()) 5445aee162fSJim Ingham { 545fd54b368SJason Molenda result.AppendErrorWithFormat("Invalid arguments for '%s'.\nUsage: %s\n", m_cmd_name.c_str(), m_cmd_syntax.c_str()); 5465aee162fSJim Ingham result.SetStatus (eReturnStatusFailed); 5475aee162fSJim Ingham } 5485aee162fSJim Ingham else 5495aee162fSJim Ingham { 55071337622SGreg Clayton if (state != eStateConnected) 55171337622SGreg Clayton { 552144f3a9cSGreg Clayton const char *plugin_name = m_options.attach_info.GetProcessPluginName(); 553c3776bf2SGreg Clayton process = target->CreateProcess (m_interpreter.GetDebugger().GetListener(), plugin_name, NULL).get(); 55471337622SGreg Clayton } 5555aee162fSJim Ingham 5565aee162fSJim Ingham if (process) 5575aee162fSJim Ingham { 5585aee162fSJim Ingham Error error; 559144f3a9cSGreg Clayton // If no process info was specified, then use the target executable 560144f3a9cSGreg Clayton // name as the process to attach to by default 561144f3a9cSGreg Clayton if (!m_options.attach_info.ProcessInfoSpecified ()) 5625aee162fSJim Ingham { 5633a0b9cdfSJim Ingham if (old_exec_module_sp) 564ad9e828cSGreg Clayton m_options.attach_info.GetExecutableFile().GetFilename() = old_exec_module_sp->GetPlatformFileSpec().GetFilename(); 565144f3a9cSGreg Clayton 566144f3a9cSGreg Clayton if (!m_options.attach_info.ProcessInfoSpecified ()) 5673a0b9cdfSJim Ingham { 568144f3a9cSGreg Clayton error.SetErrorString ("no process specified, create a target with a file, or specify the --pid or --name command option"); 5693a0b9cdfSJim Ingham } 5705aee162fSJim Ingham } 5713a0b9cdfSJim Ingham 572144f3a9cSGreg Clayton if (error.Success()) 5733a0b9cdfSJim Ingham { 574144f3a9cSGreg Clayton error = process->Attach (m_options.attach_info); 5753a0b9cdfSJim Ingham 5765aee162fSJim Ingham if (error.Success()) 5775aee162fSJim Ingham { 5785aee162fSJim Ingham result.SetStatus (eReturnStatusSuccessContinuingNoResult); 5795aee162fSJim Ingham } 5805aee162fSJim Ingham else 5815aee162fSJim Ingham { 582144f3a9cSGreg Clayton result.AppendErrorWithFormat ("attach failed: %s\n", error.AsCString()); 5835aee162fSJim Ingham result.SetStatus (eReturnStatusFailed); 5845aee162fSJim Ingham return false; 5855aee162fSJim Ingham } 586bb3a283bSJim Ingham // If we're synchronous, wait for the stopped event and report that. 587bb3a283bSJim Ingham // Otherwise just return. 588bb3a283bSJim Ingham // FIXME: in the async case it will now be possible to get to the command 589bb3a283bSJim Ingham // interpreter with a state eStateAttaching. Make sure we handle that correctly. 590bb3a283bSJim Ingham StateType state = process->WaitForProcessToStop (NULL); 591bb3a283bSJim Ingham 592bb3a283bSJim Ingham result.SetDidChangeProcessState (true); 593aa739093SJohnny Chen 594aa739093SJohnny Chen if (state == eStateStopped) 595aa739093SJohnny Chen { 596d01b2953SDaniel Malea result.AppendMessageWithFormat ("Process %" PRIu64 " %s\n", process->GetID(), StateAsCString (state)); 597bb3a283bSJim Ingham result.SetStatus (eReturnStatusSuccessFinishNoResult); 598bb3a283bSJim Ingham } 599aa739093SJohnny Chen else 600aa739093SJohnny Chen { 601aa739093SJohnny Chen result.AppendError ("attach failed: process did not stop (no such process or permission problem?)"); 602cfc0935eSJim Ingham process->Destroy(); 603aa739093SJohnny Chen result.SetStatus (eReturnStatusFailed); 604aa739093SJohnny Chen return false; 605aa739093SJohnny Chen } 606aa739093SJohnny Chen } 6075aee162fSJim Ingham } 6085aee162fSJim Ingham } 6095aee162fSJim Ingham 6105aee162fSJim Ingham if (result.Succeeded()) 6115aee162fSJim Ingham { 6125aee162fSJim Ingham // Okay, we're done. Last step is to warn if the executable module has changed: 613513c26ceSGreg Clayton char new_path[PATH_MAX]; 614aa149cbdSGreg Clayton ModuleSP new_exec_module_sp (target->GetExecutableModule()); 6155aee162fSJim Ingham if (!old_exec_module_sp) 6165aee162fSJim Ingham { 617513c26ceSGreg Clayton // We might not have a module if we attached to a raw pid... 618aa149cbdSGreg Clayton if (new_exec_module_sp) 619513c26ceSGreg Clayton { 620aa149cbdSGreg Clayton new_exec_module_sp->GetFileSpec().GetPath(new_path, PATH_MAX); 621513c26ceSGreg Clayton result.AppendMessageWithFormat("Executable module set to \"%s\".\n", new_path); 622513c26ceSGreg Clayton } 6235aee162fSJim Ingham } 624aa149cbdSGreg Clayton else if (old_exec_module_sp->GetFileSpec() != new_exec_module_sp->GetFileSpec()) 6255aee162fSJim Ingham { 626513c26ceSGreg Clayton char old_path[PATH_MAX]; 6275aee162fSJim Ingham 6285aee162fSJim Ingham old_exec_module_sp->GetFileSpec().GetPath (old_path, PATH_MAX); 629aa149cbdSGreg Clayton new_exec_module_sp->GetFileSpec().GetPath (new_path, PATH_MAX); 6305aee162fSJim Ingham 6315aee162fSJim Ingham result.AppendWarningWithFormat("Executable module changed from \"%s\" to \"%s\".\n", 6325aee162fSJim Ingham old_path, new_path); 6335aee162fSJim Ingham } 6345aee162fSJim Ingham 6355aee162fSJim Ingham if (!old_arch_spec.IsValid()) 6365aee162fSJim Ingham { 637c1b1f1eaSGreg Clayton result.AppendMessageWithFormat ("Architecture set to: %s.\n", target->GetArchitecture().GetTriple().getTriple().c_str()); 6385aee162fSJim Ingham } 639bf4b7be6SSean Callanan else if (!old_arch_spec.IsExactMatch(target->GetArchitecture())) 6405aee162fSJim Ingham { 6415aee162fSJim Ingham result.AppendWarningWithFormat("Architecture changed from %s to %s.\n", 642c1b1f1eaSGreg Clayton old_arch_spec.GetTriple().getTriple().c_str(), 643c1b1f1eaSGreg Clayton target->GetArchitecture().GetTriple().getTriple().c_str()); 6445aee162fSJim Ingham } 645a95ce623SJohnny Chen 646a95ce623SJohnny Chen // This supports the use-case scenario of immediately continuing the process once attached. 647a95ce623SJohnny Chen if (m_options.attach_info.GetContinueOnceAttached()) 6485bcaf583SSean Callanan m_interpreter.HandleCommand("process continue", eLazyBoolNo, result); 6495aee162fSJim Ingham } 6505aee162fSJim Ingham return result.Succeeded(); 6515aee162fSJim Ingham } 6525aee162fSJim Ingham 65330fdc8d8SChris Lattner CommandOptions m_options; 65430fdc8d8SChris Lattner }; 65530fdc8d8SChris Lattner 65630fdc8d8SChris Lattner 657e0d378b3SGreg Clayton OptionDefinition 65830fdc8d8SChris Lattner CommandObjectProcessAttach::CommandOptions::g_option_table[] = 65930fdc8d8SChris Lattner { 660a95ce623SJohnny Chen { LLDB_OPT_SET_ALL, false, "continue",'c', no_argument, NULL, 0, eArgTypeNone, "Immediately continue the process once attached."}, 661deaab222SCaroline Tice { LLDB_OPT_SET_ALL, false, "plugin", 'P', required_argument, NULL, 0, eArgTypePlugin, "Name of the process plugin you want to use."}, 662deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "pid", 'p', required_argument, NULL, 0, eArgTypePid, "The process ID of an existing process to attach to."}, 663deaab222SCaroline Tice { LLDB_OPT_SET_2, false, "name", 'n', required_argument, NULL, 0, eArgTypeProcessName, "The name of the process to attach to."}, 664cd16df91SJim Ingham { LLDB_OPT_SET_2, false, "include-existing", 'i', no_argument, NULL, 0, eArgTypeNone, "Include existing processes when doing attach -w."}, 6654ce37abbSJohnny Chen { LLDB_OPT_SET_2, false, "waitfor", 'w', no_argument, NULL, 0, eArgTypeNone, "Wait for the process with <process-name> to launch."}, 666deaab222SCaroline Tice { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL } 66730fdc8d8SChris Lattner }; 66830fdc8d8SChris Lattner 66930fdc8d8SChris Lattner //------------------------------------------------------------------------- 67030fdc8d8SChris Lattner // CommandObjectProcessContinue 67130fdc8d8SChris Lattner //------------------------------------------------------------------------- 672bb9caf73SJim Ingham #pragma mark CommandObjectProcessContinue 67330fdc8d8SChris Lattner 6745a988416SJim Ingham class CommandObjectProcessContinue : public CommandObjectParsed 67530fdc8d8SChris Lattner { 67630fdc8d8SChris Lattner public: 67730fdc8d8SChris Lattner 678a7015092SGreg Clayton CommandObjectProcessContinue (CommandInterpreter &interpreter) : 6795a988416SJim Ingham CommandObjectParsed (interpreter, 680a7015092SGreg Clayton "process continue", 681e3d26315SCaroline Tice "Continue execution of all threads in the current process.", 68230fdc8d8SChris Lattner "process continue", 683f9fc609fSGreg Clayton eFlagRequiresProcess | 684f9fc609fSGreg Clayton eFlagTryTargetAPILock | 685f9fc609fSGreg Clayton eFlagProcessMustBeLaunched | 686f9fc609fSGreg Clayton eFlagProcessMustBePaused ), 6870e41084aSJim Ingham m_options(interpreter) 68830fdc8d8SChris Lattner { 68930fdc8d8SChris Lattner } 69030fdc8d8SChris Lattner 69130fdc8d8SChris Lattner 69230fdc8d8SChris Lattner ~CommandObjectProcessContinue () 69330fdc8d8SChris Lattner { 69430fdc8d8SChris Lattner } 69530fdc8d8SChris Lattner 6965a988416SJim Ingham protected: 6970e41084aSJim Ingham 6980e41084aSJim Ingham class CommandOptions : public Options 6990e41084aSJim Ingham { 7000e41084aSJim Ingham public: 7010e41084aSJim Ingham 7020e41084aSJim Ingham CommandOptions (CommandInterpreter &interpreter) : 7030e41084aSJim Ingham Options(interpreter) 7040e41084aSJim Ingham { 7050e41084aSJim Ingham // Keep default values of all options in one place: OptionParsingStarting () 7060e41084aSJim Ingham OptionParsingStarting (); 7070e41084aSJim Ingham } 7080e41084aSJim Ingham 7090e41084aSJim Ingham ~CommandOptions () 7100e41084aSJim Ingham { 7110e41084aSJim Ingham } 7120e41084aSJim Ingham 7130e41084aSJim Ingham Error 7140e41084aSJim Ingham SetOptionValue (uint32_t option_idx, const char *option_arg) 7150e41084aSJim Ingham { 7160e41084aSJim Ingham Error error; 7173bcdfc0eSGreg Clayton const int short_option = m_getopt_table[option_idx].val; 7180e41084aSJim Ingham bool success = false; 7190e41084aSJim Ingham switch (short_option) 7200e41084aSJim Ingham { 7210e41084aSJim Ingham case 'i': 7220e41084aSJim Ingham m_ignore = Args::StringToUInt32 (option_arg, 0, 0, &success); 7230e41084aSJim Ingham if (!success) 7240e41084aSJim Ingham error.SetErrorStringWithFormat ("invalid value for ignore option: \"%s\", should be a number.", option_arg); 7250e41084aSJim Ingham break; 7260e41084aSJim Ingham 7270e41084aSJim Ingham default: 7280e41084aSJim Ingham error.SetErrorStringWithFormat("invalid short option character '%c'", short_option); 7290e41084aSJim Ingham break; 7300e41084aSJim Ingham } 7310e41084aSJim Ingham return error; 7320e41084aSJim Ingham } 7330e41084aSJim Ingham 7340e41084aSJim Ingham void 7350e41084aSJim Ingham OptionParsingStarting () 7360e41084aSJim Ingham { 7370e41084aSJim Ingham m_ignore = 0; 7380e41084aSJim Ingham } 7390e41084aSJim Ingham 7400e41084aSJim Ingham const OptionDefinition* 7410e41084aSJim Ingham GetDefinitions () 7420e41084aSJim Ingham { 7430e41084aSJim Ingham return g_option_table; 7440e41084aSJim Ingham } 7450e41084aSJim Ingham 7460e41084aSJim Ingham // Options table: Required for subclasses of Options. 7470e41084aSJim Ingham 7480e41084aSJim Ingham static OptionDefinition g_option_table[]; 7490e41084aSJim Ingham 7500e41084aSJim Ingham uint32_t m_ignore; 7510e41084aSJim Ingham }; 7520e41084aSJim Ingham 75330fdc8d8SChris Lattner bool 754f9fc609fSGreg Clayton DoExecute (Args& command, CommandReturnObject &result) 75530fdc8d8SChris Lattner { 756f9fc609fSGreg Clayton Process *process = m_exe_ctx.GetProcessPtr(); 757a7015092SGreg Clayton bool synchronous_execution = m_interpreter.GetSynchronous (); 75830fdc8d8SChris Lattner StateType state = process->GetState(); 75930fdc8d8SChris Lattner if (state == eStateStopped) 76030fdc8d8SChris Lattner { 76130fdc8d8SChris Lattner if (command.GetArgumentCount() != 0) 76230fdc8d8SChris Lattner { 76330fdc8d8SChris Lattner result.AppendErrorWithFormat ("The '%s' command does not take any arguments.\n", m_cmd_name.c_str()); 76430fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 76530fdc8d8SChris Lattner return false; 76630fdc8d8SChris Lattner } 76730fdc8d8SChris Lattner 7680e41084aSJim Ingham if (m_options.m_ignore > 0) 7690e41084aSJim Ingham { 7700e41084aSJim Ingham ThreadSP sel_thread_sp(process->GetThreadList().GetSelectedThread()); 7710e41084aSJim Ingham if (sel_thread_sp) 7720e41084aSJim Ingham { 7730e41084aSJim Ingham StopInfoSP stop_info_sp = sel_thread_sp->GetStopInfo(); 7740e41084aSJim Ingham if (stop_info_sp && stop_info_sp->GetStopReason() == eStopReasonBreakpoint) 7750e41084aSJim Ingham { 7760e41084aSJim Ingham uint64_t bp_site_id = stop_info_sp->GetValue(); 7770e41084aSJim Ingham BreakpointSiteSP bp_site_sp(process->GetBreakpointSiteList().FindByID(bp_site_id)); 7780e41084aSJim Ingham if (bp_site_sp) 7790e41084aSJim Ingham { 7800e41084aSJim Ingham uint32_t num_owners = bp_site_sp->GetNumberOfOwners(); 7810e41084aSJim Ingham for (uint32_t i = 0; i < num_owners; i++) 7820e41084aSJim Ingham { 7830e41084aSJim Ingham Breakpoint &bp_ref = bp_site_sp->GetOwnerAtIndex(i)->GetBreakpoint(); 7840e41084aSJim Ingham if (!bp_ref.IsInternal()) 7850e41084aSJim Ingham { 7860e41084aSJim Ingham bp_ref.SetIgnoreCount(m_options.m_ignore); 7870e41084aSJim Ingham } 7880e41084aSJim Ingham } 7890e41084aSJim Ingham } 7900e41084aSJim Ingham } 7910e41084aSJim Ingham } 7920e41084aSJim Ingham } 7930e41084aSJim Ingham 79441f2b940SJim Ingham { // Scope for thread list mutex: 79541f2b940SJim Ingham Mutex::Locker locker (process->GetThreadList().GetMutex()); 79630fdc8d8SChris Lattner const uint32_t num_threads = process->GetThreadList().GetSize(); 79730fdc8d8SChris Lattner 79830fdc8d8SChris Lattner // Set the actions that the threads should each take when resuming 79930fdc8d8SChris Lattner for (uint32_t idx=0; idx<num_threads; ++idx) 80030fdc8d8SChris Lattner { 80130fdc8d8SChris Lattner process->GetThreadList().GetThreadAtIndex(idx)->SetResumeState (eStateRunning); 80230fdc8d8SChris Lattner } 80341f2b940SJim Ingham } 80430fdc8d8SChris Lattner 80530fdc8d8SChris Lattner Error error(process->Resume()); 80630fdc8d8SChris Lattner if (error.Success()) 80730fdc8d8SChris Lattner { 808d01b2953SDaniel Malea result.AppendMessageWithFormat ("Process %" PRIu64 " resuming\n", process->GetID()); 80930fdc8d8SChris Lattner if (synchronous_execution) 81030fdc8d8SChris Lattner { 811b132097bSGreg Clayton state = process->WaitForProcessToStop (NULL); 81230fdc8d8SChris Lattner 81330fdc8d8SChris Lattner result.SetDidChangeProcessState (true); 814d01b2953SDaniel Malea result.AppendMessageWithFormat ("Process %" PRIu64 " %s\n", process->GetID(), StateAsCString (state)); 81530fdc8d8SChris Lattner result.SetStatus (eReturnStatusSuccessFinishNoResult); 81630fdc8d8SChris Lattner } 81730fdc8d8SChris Lattner else 81830fdc8d8SChris Lattner { 81930fdc8d8SChris Lattner result.SetStatus (eReturnStatusSuccessContinuingNoResult); 82030fdc8d8SChris Lattner } 82130fdc8d8SChris Lattner } 82230fdc8d8SChris Lattner else 82330fdc8d8SChris Lattner { 82430fdc8d8SChris Lattner result.AppendErrorWithFormat("Failed to resume process: %s.\n", error.AsCString()); 82530fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 82630fdc8d8SChris Lattner } 82730fdc8d8SChris Lattner } 82830fdc8d8SChris Lattner else 82930fdc8d8SChris Lattner { 83030fdc8d8SChris Lattner result.AppendErrorWithFormat ("Process cannot be continued from its current state (%s).\n", 83130fdc8d8SChris Lattner StateAsCString(state)); 83230fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 83330fdc8d8SChris Lattner } 83430fdc8d8SChris Lattner return result.Succeeded(); 83530fdc8d8SChris Lattner } 8360e41084aSJim Ingham 8370e41084aSJim Ingham Options * 8380e41084aSJim Ingham GetOptions () 8390e41084aSJim Ingham { 8400e41084aSJim Ingham return &m_options; 8410e41084aSJim Ingham } 8420e41084aSJim Ingham 8430e41084aSJim Ingham CommandOptions m_options; 8440e41084aSJim Ingham 8450e41084aSJim Ingham }; 8460e41084aSJim Ingham 8470e41084aSJim Ingham OptionDefinition 8480e41084aSJim Ingham CommandObjectProcessContinue::CommandOptions::g_option_table[] = 8490e41084aSJim Ingham { 8500e41084aSJim Ingham { LLDB_OPT_SET_ALL, false, "ignore-count",'i', required_argument, NULL, 0, eArgTypeUnsignedInteger, 8510e41084aSJim Ingham "Ignore <N> crossings of the breakpoint (if it exists) for the currently selected thread."}, 8520e41084aSJim Ingham { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL } 85330fdc8d8SChris Lattner }; 85430fdc8d8SChris Lattner 85530fdc8d8SChris Lattner //------------------------------------------------------------------------- 85630fdc8d8SChris Lattner // CommandObjectProcessDetach 85730fdc8d8SChris Lattner //------------------------------------------------------------------------- 858bb9caf73SJim Ingham #pragma mark CommandObjectProcessDetach 85930fdc8d8SChris Lattner 8605a988416SJim Ingham class CommandObjectProcessDetach : public CommandObjectParsed 86130fdc8d8SChris Lattner { 86230fdc8d8SChris Lattner public: 86330fdc8d8SChris Lattner 864a7015092SGreg Clayton CommandObjectProcessDetach (CommandInterpreter &interpreter) : 8655a988416SJim Ingham CommandObjectParsed (interpreter, 866a7015092SGreg Clayton "process detach", 867e3d26315SCaroline Tice "Detach from the current process being debugged.", 86830fdc8d8SChris Lattner "process detach", 869f9fc609fSGreg Clayton eFlagRequiresProcess | 870f9fc609fSGreg Clayton eFlagTryTargetAPILock | 87130fdc8d8SChris Lattner eFlagProcessMustBeLaunched) 87230fdc8d8SChris Lattner { 87330fdc8d8SChris Lattner } 87430fdc8d8SChris Lattner 87530fdc8d8SChris Lattner ~CommandObjectProcessDetach () 87630fdc8d8SChris Lattner { 87730fdc8d8SChris Lattner } 87830fdc8d8SChris Lattner 8795a988416SJim Ingham protected: 88030fdc8d8SChris Lattner bool 881f9fc609fSGreg Clayton DoExecute (Args& command, CommandReturnObject &result) 88230fdc8d8SChris Lattner { 883f9fc609fSGreg Clayton Process *process = m_exe_ctx.GetProcessPtr(); 884d01b2953SDaniel Malea result.AppendMessageWithFormat ("Detaching from process %" PRIu64 "\n", process->GetID()); 88530fdc8d8SChris Lattner Error error (process->Detach()); 88630fdc8d8SChris Lattner if (error.Success()) 88730fdc8d8SChris Lattner { 88830fdc8d8SChris Lattner result.SetStatus (eReturnStatusSuccessFinishResult); 88930fdc8d8SChris Lattner } 89030fdc8d8SChris Lattner else 89130fdc8d8SChris Lattner { 89230fdc8d8SChris Lattner result.AppendErrorWithFormat ("Detach failed: %s\n", error.AsCString()); 89330fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 89430fdc8d8SChris Lattner return false; 89530fdc8d8SChris Lattner } 89630fdc8d8SChris Lattner return result.Succeeded(); 89730fdc8d8SChris Lattner } 89830fdc8d8SChris Lattner }; 89930fdc8d8SChris Lattner 90030fdc8d8SChris Lattner //------------------------------------------------------------------------- 901b766a73dSGreg Clayton // CommandObjectProcessConnect 902b766a73dSGreg Clayton //------------------------------------------------------------------------- 903b766a73dSGreg Clayton #pragma mark CommandObjectProcessConnect 904b766a73dSGreg Clayton 9055a988416SJim Ingham class CommandObjectProcessConnect : public CommandObjectParsed 906b766a73dSGreg Clayton { 907b766a73dSGreg Clayton public: 908b766a73dSGreg Clayton 909b766a73dSGreg Clayton class CommandOptions : public Options 910b766a73dSGreg Clayton { 911b766a73dSGreg Clayton public: 912b766a73dSGreg Clayton 913eb0103f2SGreg Clayton CommandOptions (CommandInterpreter &interpreter) : 914eb0103f2SGreg Clayton Options(interpreter) 915b766a73dSGreg Clayton { 916f6b8b581SGreg Clayton // Keep default values of all options in one place: OptionParsingStarting () 917f6b8b581SGreg Clayton OptionParsingStarting (); 918b766a73dSGreg Clayton } 919b766a73dSGreg Clayton 920b766a73dSGreg Clayton ~CommandOptions () 921b766a73dSGreg Clayton { 922b766a73dSGreg Clayton } 923b766a73dSGreg Clayton 924b766a73dSGreg Clayton Error 925f6b8b581SGreg Clayton SetOptionValue (uint32_t option_idx, const char *option_arg) 926b766a73dSGreg Clayton { 927b766a73dSGreg Clayton Error error; 9283bcdfc0eSGreg Clayton const int short_option = m_getopt_table[option_idx].val; 929b766a73dSGreg Clayton 930b766a73dSGreg Clayton switch (short_option) 931b766a73dSGreg Clayton { 932b766a73dSGreg Clayton case 'p': 933b766a73dSGreg Clayton plugin_name.assign (option_arg); 934b766a73dSGreg Clayton break; 935b766a73dSGreg Clayton 936b766a73dSGreg Clayton default: 93786edbf41SGreg Clayton error.SetErrorStringWithFormat("invalid short option character '%c'", short_option); 938b766a73dSGreg Clayton break; 939b766a73dSGreg Clayton } 940b766a73dSGreg Clayton return error; 941b766a73dSGreg Clayton } 942b766a73dSGreg Clayton 943b766a73dSGreg Clayton void 944f6b8b581SGreg Clayton OptionParsingStarting () 945b766a73dSGreg Clayton { 946b766a73dSGreg Clayton plugin_name.clear(); 947b766a73dSGreg Clayton } 948b766a73dSGreg Clayton 949e0d378b3SGreg Clayton const OptionDefinition* 950b766a73dSGreg Clayton GetDefinitions () 951b766a73dSGreg Clayton { 952b766a73dSGreg Clayton return g_option_table; 953b766a73dSGreg Clayton } 954b766a73dSGreg Clayton 955b766a73dSGreg Clayton // Options table: Required for subclasses of Options. 956b766a73dSGreg Clayton 957e0d378b3SGreg Clayton static OptionDefinition g_option_table[]; 958b766a73dSGreg Clayton 959b766a73dSGreg Clayton // Instance variables to hold the values for command options. 960b766a73dSGreg Clayton 961b766a73dSGreg Clayton std::string plugin_name; 962b766a73dSGreg Clayton }; 963b766a73dSGreg Clayton 964b766a73dSGreg Clayton CommandObjectProcessConnect (CommandInterpreter &interpreter) : 9655a988416SJim Ingham CommandObjectParsed (interpreter, 966b766a73dSGreg Clayton "process connect", 967b766a73dSGreg Clayton "Connect to a remote debug service.", 968b766a73dSGreg Clayton "process connect <remote-url>", 969eb0103f2SGreg Clayton 0), 970eb0103f2SGreg Clayton m_options (interpreter) 971b766a73dSGreg Clayton { 972b766a73dSGreg Clayton } 973b766a73dSGreg Clayton 974b766a73dSGreg Clayton ~CommandObjectProcessConnect () 975b766a73dSGreg Clayton { 976b766a73dSGreg Clayton } 977b766a73dSGreg Clayton 978b766a73dSGreg Clayton 9795a988416SJim Ingham Options * 9805a988416SJim Ingham GetOptions () 9815a988416SJim Ingham { 9825a988416SJim Ingham return &m_options; 9835a988416SJim Ingham } 9845a988416SJim Ingham 9855a988416SJim Ingham protected: 986b766a73dSGreg Clayton bool 9875a988416SJim Ingham DoExecute (Args& command, 988b766a73dSGreg Clayton CommandReturnObject &result) 989b766a73dSGreg Clayton { 990b766a73dSGreg Clayton 991b766a73dSGreg Clayton TargetSP target_sp (m_interpreter.GetDebugger().GetSelectedTarget()); 992b766a73dSGreg Clayton Error error; 993f9fc609fSGreg Clayton Process *process = m_exe_ctx.GetProcessPtr(); 994b766a73dSGreg Clayton if (process) 995b766a73dSGreg Clayton { 996b766a73dSGreg Clayton if (process->IsAlive()) 997b766a73dSGreg Clayton { 998d01b2953SDaniel Malea result.AppendErrorWithFormat ("Process %" PRIu64 " is currently being debugged, kill the process before connecting.\n", 999b766a73dSGreg Clayton process->GetID()); 1000b766a73dSGreg Clayton result.SetStatus (eReturnStatusFailed); 1001b766a73dSGreg Clayton return false; 1002b766a73dSGreg Clayton } 1003b766a73dSGreg Clayton } 1004b766a73dSGreg Clayton 1005b766a73dSGreg Clayton if (!target_sp) 1006b766a73dSGreg Clayton { 1007b766a73dSGreg Clayton // If there isn't a current target create one. 1008b766a73dSGreg Clayton 1009b766a73dSGreg Clayton error = m_interpreter.GetDebugger().GetTargetList().CreateTarget (m_interpreter.GetDebugger(), 1010a0ca6601SGreg Clayton NULL, 1011cac9c5f9SGreg Clayton NULL, 1012b766a73dSGreg Clayton false, 1013cac9c5f9SGreg Clayton NULL, // No platform options 1014b766a73dSGreg Clayton target_sp); 1015b766a73dSGreg Clayton if (!target_sp || error.Fail()) 1016b766a73dSGreg Clayton { 1017b766a73dSGreg Clayton result.AppendError(error.AsCString("Error creating target")); 1018b766a73dSGreg Clayton result.SetStatus (eReturnStatusFailed); 1019b766a73dSGreg Clayton return false; 1020b766a73dSGreg Clayton } 1021b766a73dSGreg Clayton m_interpreter.GetDebugger().GetTargetList().SetSelectedTarget(target_sp.get()); 1022b766a73dSGreg Clayton } 1023b766a73dSGreg Clayton 1024b766a73dSGreg Clayton if (command.GetArgumentCount() == 1) 1025b766a73dSGreg Clayton { 1026b766a73dSGreg Clayton const char *plugin_name = NULL; 1027b766a73dSGreg Clayton if (!m_options.plugin_name.empty()) 1028b766a73dSGreg Clayton plugin_name = m_options.plugin_name.c_str(); 1029b766a73dSGreg Clayton 1030b766a73dSGreg Clayton const char *remote_url = command.GetArgumentAtIndex(0); 1031c3776bf2SGreg Clayton process = target_sp->CreateProcess (m_interpreter.GetDebugger().GetListener(), plugin_name, NULL).get(); 1032b766a73dSGreg Clayton 1033b766a73dSGreg Clayton if (process) 1034b766a73dSGreg Clayton { 10354bd4e7e3SJason Molenda error = process->ConnectRemote (&process->GetTarget().GetDebugger().GetOutputStream(), remote_url); 1036b766a73dSGreg Clayton 1037b766a73dSGreg Clayton if (error.Fail()) 1038b766a73dSGreg Clayton { 1039b766a73dSGreg Clayton result.AppendError(error.AsCString("Remote connect failed")); 1040b766a73dSGreg Clayton result.SetStatus (eReturnStatusFailed); 10411517dd33SGreg Clayton target_sp->DeleteCurrentProcess(); 1042b766a73dSGreg Clayton return false; 1043b766a73dSGreg Clayton } 1044b766a73dSGreg Clayton } 1045b766a73dSGreg Clayton else 1046b766a73dSGreg Clayton { 1047fd54b368SJason Molenda result.AppendErrorWithFormat ("Unable to find process plug-in for remote URL '%s'.\nPlease specify a process plug-in name with the --plugin option, or specify an object file using the \"file\" command.\n", 1048f00b7511SDaniel Malea remote_url); 1049b766a73dSGreg Clayton result.SetStatus (eReturnStatusFailed); 1050b766a73dSGreg Clayton } 1051b766a73dSGreg Clayton } 1052b766a73dSGreg Clayton else 1053b766a73dSGreg Clayton { 1054fd54b368SJason Molenda result.AppendErrorWithFormat ("'%s' takes exactly one argument:\nUsage: %s\n", 1055b766a73dSGreg Clayton m_cmd_name.c_str(), 1056b766a73dSGreg Clayton m_cmd_syntax.c_str()); 1057b766a73dSGreg Clayton result.SetStatus (eReturnStatusFailed); 1058b766a73dSGreg Clayton } 1059b766a73dSGreg Clayton return result.Succeeded(); 1060b766a73dSGreg Clayton } 1061b766a73dSGreg Clayton 1062b766a73dSGreg Clayton CommandOptions m_options; 1063b766a73dSGreg Clayton }; 1064b766a73dSGreg Clayton 1065e0d378b3SGreg Clayton OptionDefinition 1066b766a73dSGreg Clayton CommandObjectProcessConnect::CommandOptions::g_option_table[] = 1067b766a73dSGreg Clayton { 1068b766a73dSGreg Clayton { LLDB_OPT_SET_ALL, false, "plugin", 'p', required_argument, NULL, 0, eArgTypePlugin, "Name of the process plugin you want to use."}, 1069b766a73dSGreg Clayton { 0, false, NULL, 0 , 0, NULL, 0, eArgTypeNone, NULL } 1070b766a73dSGreg Clayton }; 1071b766a73dSGreg Clayton 1072b766a73dSGreg Clayton //------------------------------------------------------------------------- 1073998255bfSGreg Clayton // CommandObjectProcessPlugin 1074998255bfSGreg Clayton //------------------------------------------------------------------------- 1075998255bfSGreg Clayton #pragma mark CommandObjectProcessPlugin 1076998255bfSGreg Clayton 1077998255bfSGreg Clayton class CommandObjectProcessPlugin : public CommandObjectProxy 1078998255bfSGreg Clayton { 1079998255bfSGreg Clayton public: 1080998255bfSGreg Clayton 1081998255bfSGreg Clayton CommandObjectProcessPlugin (CommandInterpreter &interpreter) : 1082998255bfSGreg Clayton CommandObjectProxy (interpreter, 1083998255bfSGreg Clayton "process plugin", 1084998255bfSGreg Clayton "Send a custom command to the current process plug-in.", 1085998255bfSGreg Clayton "process plugin <args>", 1086998255bfSGreg Clayton 0) 1087998255bfSGreg Clayton { 1088998255bfSGreg Clayton } 1089998255bfSGreg Clayton 1090998255bfSGreg Clayton ~CommandObjectProcessPlugin () 1091998255bfSGreg Clayton { 1092998255bfSGreg Clayton } 1093998255bfSGreg Clayton 1094998255bfSGreg Clayton virtual CommandObject * 1095998255bfSGreg Clayton GetProxyCommandObject() 1096998255bfSGreg Clayton { 1097*e05b2efeSGreg Clayton Process *process = m_interpreter.GetExecutionContext().GetProcessPtr(); 1098998255bfSGreg Clayton if (process) 1099998255bfSGreg Clayton return process->GetPluginCommandObject(); 1100998255bfSGreg Clayton return NULL; 1101998255bfSGreg Clayton } 1102998255bfSGreg Clayton }; 1103998255bfSGreg Clayton 1104998255bfSGreg Clayton 1105998255bfSGreg Clayton //------------------------------------------------------------------------- 11068f343b09SGreg Clayton // CommandObjectProcessLoad 11078f343b09SGreg Clayton //------------------------------------------------------------------------- 1108bb9caf73SJim Ingham #pragma mark CommandObjectProcessLoad 11098f343b09SGreg Clayton 11105a988416SJim Ingham class CommandObjectProcessLoad : public CommandObjectParsed 11118f343b09SGreg Clayton { 11128f343b09SGreg Clayton public: 11138f343b09SGreg Clayton 11148f343b09SGreg Clayton CommandObjectProcessLoad (CommandInterpreter &interpreter) : 11155a988416SJim Ingham CommandObjectParsed (interpreter, 11168f343b09SGreg Clayton "process load", 11178f343b09SGreg Clayton "Load a shared library into the current process.", 11188f343b09SGreg Clayton "process load <filename> [<filename> ...]", 1119f9fc609fSGreg Clayton eFlagRequiresProcess | 1120f9fc609fSGreg Clayton eFlagTryTargetAPILock | 1121f9fc609fSGreg Clayton eFlagProcessMustBeLaunched | 1122f9fc609fSGreg Clayton eFlagProcessMustBePaused ) 11238f343b09SGreg Clayton { 11248f343b09SGreg Clayton } 11258f343b09SGreg Clayton 11268f343b09SGreg Clayton ~CommandObjectProcessLoad () 11278f343b09SGreg Clayton { 11288f343b09SGreg Clayton } 11298f343b09SGreg Clayton 11305a988416SJim Ingham protected: 11318f343b09SGreg Clayton bool 11325a988416SJim Ingham DoExecute (Args& command, 11338f343b09SGreg Clayton CommandReturnObject &result) 11348f343b09SGreg Clayton { 1135f9fc609fSGreg Clayton Process *process = m_exe_ctx.GetProcessPtr(); 11368f343b09SGreg Clayton 11378f343b09SGreg Clayton const uint32_t argc = command.GetArgumentCount(); 11388f343b09SGreg Clayton 11398f343b09SGreg Clayton for (uint32_t i=0; i<argc; ++i) 11408f343b09SGreg Clayton { 11418f343b09SGreg Clayton Error error; 11428f343b09SGreg Clayton const char *image_path = command.GetArgumentAtIndex(i); 11438f343b09SGreg Clayton FileSpec image_spec (image_path, false); 1144aa516843SGreg Clayton process->GetTarget().GetPlatform()->ResolveRemotePath(image_spec, image_spec); 11458f343b09SGreg Clayton uint32_t image_token = process->LoadImage(image_spec, error); 11468f343b09SGreg Clayton if (image_token != LLDB_INVALID_IMAGE_TOKEN) 11478f343b09SGreg Clayton { 11488f343b09SGreg Clayton result.AppendMessageWithFormat ("Loading \"%s\"...ok\nImage %u loaded.\n", image_path, image_token); 11498f343b09SGreg Clayton result.SetStatus (eReturnStatusSuccessFinishResult); 11508f343b09SGreg Clayton } 11518f343b09SGreg Clayton else 11528f343b09SGreg Clayton { 11538f343b09SGreg Clayton result.AppendErrorWithFormat ("failed to load '%s': %s", image_path, error.AsCString()); 11548f343b09SGreg Clayton result.SetStatus (eReturnStatusFailed); 11558f343b09SGreg Clayton } 11568f343b09SGreg Clayton } 11578f343b09SGreg Clayton return result.Succeeded(); 11588f343b09SGreg Clayton } 11598f343b09SGreg Clayton }; 11608f343b09SGreg Clayton 11618f343b09SGreg Clayton 11628f343b09SGreg Clayton //------------------------------------------------------------------------- 11638f343b09SGreg Clayton // CommandObjectProcessUnload 11648f343b09SGreg Clayton //------------------------------------------------------------------------- 1165bb9caf73SJim Ingham #pragma mark CommandObjectProcessUnload 11668f343b09SGreg Clayton 11675a988416SJim Ingham class CommandObjectProcessUnload : public CommandObjectParsed 11688f343b09SGreg Clayton { 11698f343b09SGreg Clayton public: 11708f343b09SGreg Clayton 11718f343b09SGreg Clayton CommandObjectProcessUnload (CommandInterpreter &interpreter) : 11725a988416SJim Ingham CommandObjectParsed (interpreter, 11738f343b09SGreg Clayton "process unload", 11748f343b09SGreg Clayton "Unload a shared library from the current process using the index returned by a previous call to \"process load\".", 11758f343b09SGreg Clayton "process unload <index>", 1176f9fc609fSGreg Clayton eFlagRequiresProcess | 1177f9fc609fSGreg Clayton eFlagTryTargetAPILock | 1178f9fc609fSGreg Clayton eFlagProcessMustBeLaunched | 1179f9fc609fSGreg Clayton eFlagProcessMustBePaused ) 11808f343b09SGreg Clayton { 11818f343b09SGreg Clayton } 11828f343b09SGreg Clayton 11838f343b09SGreg Clayton ~CommandObjectProcessUnload () 11848f343b09SGreg Clayton { 11858f343b09SGreg Clayton } 11868f343b09SGreg Clayton 11875a988416SJim Ingham protected: 11888f343b09SGreg Clayton bool 11895a988416SJim Ingham DoExecute (Args& command, 11908f343b09SGreg Clayton CommandReturnObject &result) 11918f343b09SGreg Clayton { 1192f9fc609fSGreg Clayton Process *process = m_exe_ctx.GetProcessPtr(); 11938f343b09SGreg Clayton 11948f343b09SGreg Clayton const uint32_t argc = command.GetArgumentCount(); 11958f343b09SGreg Clayton 11968f343b09SGreg Clayton for (uint32_t i=0; i<argc; ++i) 11978f343b09SGreg Clayton { 11988f343b09SGreg Clayton const char *image_token_cstr = command.GetArgumentAtIndex(i); 11998f343b09SGreg Clayton uint32_t image_token = Args::StringToUInt32(image_token_cstr, LLDB_INVALID_IMAGE_TOKEN, 0); 12008f343b09SGreg Clayton if (image_token == LLDB_INVALID_IMAGE_TOKEN) 12018f343b09SGreg Clayton { 12028f343b09SGreg Clayton result.AppendErrorWithFormat ("invalid image index argument '%s'", image_token_cstr); 12038f343b09SGreg Clayton result.SetStatus (eReturnStatusFailed); 12048f343b09SGreg Clayton break; 12058f343b09SGreg Clayton } 12068f343b09SGreg Clayton else 12078f343b09SGreg Clayton { 12088f343b09SGreg Clayton Error error (process->UnloadImage(image_token)); 12098f343b09SGreg Clayton if (error.Success()) 12108f343b09SGreg Clayton { 12118f343b09SGreg Clayton result.AppendMessageWithFormat ("Unloading shared library with index %u...ok\n", image_token); 12128f343b09SGreg Clayton result.SetStatus (eReturnStatusSuccessFinishResult); 12138f343b09SGreg Clayton } 12148f343b09SGreg Clayton else 12158f343b09SGreg Clayton { 12168f343b09SGreg Clayton result.AppendErrorWithFormat ("failed to unload image: %s", error.AsCString()); 12178f343b09SGreg Clayton result.SetStatus (eReturnStatusFailed); 12188f343b09SGreg Clayton break; 12198f343b09SGreg Clayton } 12208f343b09SGreg Clayton } 12218f343b09SGreg Clayton } 12228f343b09SGreg Clayton return result.Succeeded(); 12238f343b09SGreg Clayton } 12248f343b09SGreg Clayton }; 12258f343b09SGreg Clayton 12268f343b09SGreg Clayton //------------------------------------------------------------------------- 122730fdc8d8SChris Lattner // CommandObjectProcessSignal 122830fdc8d8SChris Lattner //------------------------------------------------------------------------- 1229bb9caf73SJim Ingham #pragma mark CommandObjectProcessSignal 123030fdc8d8SChris Lattner 12315a988416SJim Ingham class CommandObjectProcessSignal : public CommandObjectParsed 123230fdc8d8SChris Lattner { 123330fdc8d8SChris Lattner public: 123430fdc8d8SChris Lattner 1235a7015092SGreg Clayton CommandObjectProcessSignal (CommandInterpreter &interpreter) : 12365a988416SJim Ingham CommandObjectParsed (interpreter, 1237a7015092SGreg Clayton "process signal", 1238e3d26315SCaroline Tice "Send a UNIX signal to the current process being debugged.", 1239f9fc609fSGreg Clayton NULL, 1240f9fc609fSGreg Clayton eFlagRequiresProcess | eFlagTryTargetAPILock) 124130fdc8d8SChris Lattner { 1242405fe67fSCaroline Tice CommandArgumentEntry arg; 1243405fe67fSCaroline Tice CommandArgumentData signal_arg; 1244405fe67fSCaroline Tice 1245405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 1246c0dbdfb6SCaroline Tice signal_arg.arg_type = eArgTypeUnixSignal; 1247405fe67fSCaroline Tice signal_arg.arg_repetition = eArgRepeatPlain; 1248405fe67fSCaroline Tice 1249405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 1250405fe67fSCaroline Tice arg.push_back (signal_arg); 1251405fe67fSCaroline Tice 1252405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 1253405fe67fSCaroline Tice m_arguments.push_back (arg); 125430fdc8d8SChris Lattner } 125530fdc8d8SChris Lattner 125630fdc8d8SChris Lattner ~CommandObjectProcessSignal () 125730fdc8d8SChris Lattner { 125830fdc8d8SChris Lattner } 125930fdc8d8SChris Lattner 12605a988416SJim Ingham protected: 126130fdc8d8SChris Lattner bool 12625a988416SJim Ingham DoExecute (Args& command, 126330fdc8d8SChris Lattner CommandReturnObject &result) 126430fdc8d8SChris Lattner { 1265f9fc609fSGreg Clayton Process *process = m_exe_ctx.GetProcessPtr(); 126630fdc8d8SChris Lattner 126730fdc8d8SChris Lattner if (command.GetArgumentCount() == 1) 126830fdc8d8SChris Lattner { 1269237cd906SGreg Clayton int signo = LLDB_INVALID_SIGNAL_NUMBER; 1270237cd906SGreg Clayton 1271237cd906SGreg Clayton const char *signal_name = command.GetArgumentAtIndex(0); 1272237cd906SGreg Clayton if (::isxdigit (signal_name[0])) 1273237cd906SGreg Clayton signo = Args::StringToSInt32(signal_name, LLDB_INVALID_SIGNAL_NUMBER, 0); 1274237cd906SGreg Clayton else 1275237cd906SGreg Clayton signo = process->GetUnixSignals().GetSignalNumberFromName (signal_name); 1276237cd906SGreg Clayton 1277237cd906SGreg Clayton if (signo == LLDB_INVALID_SIGNAL_NUMBER) 127830fdc8d8SChris Lattner { 127930fdc8d8SChris Lattner result.AppendErrorWithFormat ("Invalid signal argument '%s'.\n", command.GetArgumentAtIndex(0)); 128030fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 128130fdc8d8SChris Lattner } 128230fdc8d8SChris Lattner else 128330fdc8d8SChris Lattner { 128430fdc8d8SChris Lattner Error error (process->Signal (signo)); 128530fdc8d8SChris Lattner if (error.Success()) 128630fdc8d8SChris Lattner { 128730fdc8d8SChris Lattner result.SetStatus (eReturnStatusSuccessFinishResult); 128830fdc8d8SChris Lattner } 128930fdc8d8SChris Lattner else 129030fdc8d8SChris Lattner { 129130fdc8d8SChris Lattner result.AppendErrorWithFormat ("Failed to send signal %i: %s\n", signo, error.AsCString()); 129230fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 129330fdc8d8SChris Lattner } 129430fdc8d8SChris Lattner } 129530fdc8d8SChris Lattner } 129630fdc8d8SChris Lattner else 129730fdc8d8SChris Lattner { 1298fd54b368SJason Molenda result.AppendErrorWithFormat("'%s' takes exactly one signal number argument:\nUsage: %s\n", m_cmd_name.c_str(), 129930fdc8d8SChris Lattner m_cmd_syntax.c_str()); 130030fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 130130fdc8d8SChris Lattner } 130230fdc8d8SChris Lattner return result.Succeeded(); 130330fdc8d8SChris Lattner } 130430fdc8d8SChris Lattner }; 130530fdc8d8SChris Lattner 130630fdc8d8SChris Lattner 130730fdc8d8SChris Lattner //------------------------------------------------------------------------- 130830fdc8d8SChris Lattner // CommandObjectProcessInterrupt 130930fdc8d8SChris Lattner //------------------------------------------------------------------------- 1310bb9caf73SJim Ingham #pragma mark CommandObjectProcessInterrupt 131130fdc8d8SChris Lattner 13125a988416SJim Ingham class CommandObjectProcessInterrupt : public CommandObjectParsed 131330fdc8d8SChris Lattner { 131430fdc8d8SChris Lattner public: 131530fdc8d8SChris Lattner 131630fdc8d8SChris Lattner 1317a7015092SGreg Clayton CommandObjectProcessInterrupt (CommandInterpreter &interpreter) : 13185a988416SJim Ingham CommandObjectParsed (interpreter, 1319a7015092SGreg Clayton "process interrupt", 1320e3d26315SCaroline Tice "Interrupt the current process being debugged.", 132130fdc8d8SChris Lattner "process interrupt", 1322f9fc609fSGreg Clayton eFlagRequiresProcess | 1323f9fc609fSGreg Clayton eFlagTryTargetAPILock | 132430fdc8d8SChris Lattner eFlagProcessMustBeLaunched) 132530fdc8d8SChris Lattner { 132630fdc8d8SChris Lattner } 132730fdc8d8SChris Lattner 132830fdc8d8SChris Lattner ~CommandObjectProcessInterrupt () 132930fdc8d8SChris Lattner { 133030fdc8d8SChris Lattner } 133130fdc8d8SChris Lattner 13325a988416SJim Ingham protected: 133330fdc8d8SChris Lattner bool 13345a988416SJim Ingham DoExecute (Args& command, 133530fdc8d8SChris Lattner CommandReturnObject &result) 133630fdc8d8SChris Lattner { 1337f9fc609fSGreg Clayton Process *process = m_exe_ctx.GetProcessPtr(); 133830fdc8d8SChris Lattner if (process == NULL) 133930fdc8d8SChris Lattner { 134030fdc8d8SChris Lattner result.AppendError ("no process to halt"); 134130fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 134230fdc8d8SChris Lattner return false; 134330fdc8d8SChris Lattner } 134430fdc8d8SChris Lattner 134530fdc8d8SChris Lattner if (command.GetArgumentCount() == 0) 134630fdc8d8SChris Lattner { 134730fdc8d8SChris Lattner Error error(process->Halt ()); 134830fdc8d8SChris Lattner if (error.Success()) 134930fdc8d8SChris Lattner { 135030fdc8d8SChris Lattner result.SetStatus (eReturnStatusSuccessFinishResult); 135130fdc8d8SChris Lattner 135230fdc8d8SChris Lattner // Maybe we should add a "SuspendThreadPlans so we 135330fdc8d8SChris Lattner // can halt, and keep in place all the current thread plans. 135430fdc8d8SChris Lattner process->GetThreadList().DiscardThreadPlans(); 135530fdc8d8SChris Lattner } 135630fdc8d8SChris Lattner else 135730fdc8d8SChris Lattner { 135830fdc8d8SChris Lattner result.AppendErrorWithFormat ("Failed to halt process: %s\n", error.AsCString()); 135930fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 136030fdc8d8SChris Lattner } 136130fdc8d8SChris Lattner } 136230fdc8d8SChris Lattner else 136330fdc8d8SChris Lattner { 1364fd54b368SJason Molenda result.AppendErrorWithFormat("'%s' takes no arguments:\nUsage: %s\n", 136530fdc8d8SChris Lattner m_cmd_name.c_str(), 136630fdc8d8SChris Lattner m_cmd_syntax.c_str()); 136730fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 136830fdc8d8SChris Lattner } 136930fdc8d8SChris Lattner return result.Succeeded(); 137030fdc8d8SChris Lattner } 137130fdc8d8SChris Lattner }; 137230fdc8d8SChris Lattner 137330fdc8d8SChris Lattner //------------------------------------------------------------------------- 137430fdc8d8SChris Lattner // CommandObjectProcessKill 137530fdc8d8SChris Lattner //------------------------------------------------------------------------- 1376bb9caf73SJim Ingham #pragma mark CommandObjectProcessKill 137730fdc8d8SChris Lattner 13785a988416SJim Ingham class CommandObjectProcessKill : public CommandObjectParsed 137930fdc8d8SChris Lattner { 138030fdc8d8SChris Lattner public: 138130fdc8d8SChris Lattner 1382a7015092SGreg Clayton CommandObjectProcessKill (CommandInterpreter &interpreter) : 13835a988416SJim Ingham CommandObjectParsed (interpreter, 1384a7015092SGreg Clayton "process kill", 1385e3d26315SCaroline Tice "Terminate the current process being debugged.", 138630fdc8d8SChris Lattner "process kill", 1387f9fc609fSGreg Clayton eFlagRequiresProcess | 1388f9fc609fSGreg Clayton eFlagTryTargetAPILock | 138930fdc8d8SChris Lattner eFlagProcessMustBeLaunched) 139030fdc8d8SChris Lattner { 139130fdc8d8SChris Lattner } 139230fdc8d8SChris Lattner 139330fdc8d8SChris Lattner ~CommandObjectProcessKill () 139430fdc8d8SChris Lattner { 139530fdc8d8SChris Lattner } 139630fdc8d8SChris Lattner 13975a988416SJim Ingham protected: 139830fdc8d8SChris Lattner bool 13995a988416SJim Ingham DoExecute (Args& command, 140030fdc8d8SChris Lattner CommandReturnObject &result) 140130fdc8d8SChris Lattner { 1402f9fc609fSGreg Clayton Process *process = m_exe_ctx.GetProcessPtr(); 140330fdc8d8SChris Lattner if (process == NULL) 140430fdc8d8SChris Lattner { 140530fdc8d8SChris Lattner result.AppendError ("no process to kill"); 140630fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 140730fdc8d8SChris Lattner return false; 140830fdc8d8SChris Lattner } 140930fdc8d8SChris Lattner 141030fdc8d8SChris Lattner if (command.GetArgumentCount() == 0) 141130fdc8d8SChris Lattner { 141230fdc8d8SChris Lattner Error error (process->Destroy()); 141330fdc8d8SChris Lattner if (error.Success()) 141430fdc8d8SChris Lattner { 141530fdc8d8SChris Lattner result.SetStatus (eReturnStatusSuccessFinishResult); 141630fdc8d8SChris Lattner } 141730fdc8d8SChris Lattner else 141830fdc8d8SChris Lattner { 141930fdc8d8SChris Lattner result.AppendErrorWithFormat ("Failed to kill process: %s\n", error.AsCString()); 142030fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 142130fdc8d8SChris Lattner } 142230fdc8d8SChris Lattner } 142330fdc8d8SChris Lattner else 142430fdc8d8SChris Lattner { 1425fd54b368SJason Molenda result.AppendErrorWithFormat("'%s' takes no arguments:\nUsage: %s\n", 142630fdc8d8SChris Lattner m_cmd_name.c_str(), 142730fdc8d8SChris Lattner m_cmd_syntax.c_str()); 142830fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 142930fdc8d8SChris Lattner } 143030fdc8d8SChris Lattner return result.Succeeded(); 143130fdc8d8SChris Lattner } 143230fdc8d8SChris Lattner }; 143330fdc8d8SChris Lattner 143430fdc8d8SChris Lattner //------------------------------------------------------------------------- 14354b9bea87SJim Ingham // CommandObjectProcessStatus 14364b9bea87SJim Ingham //------------------------------------------------------------------------- 1437bb9caf73SJim Ingham #pragma mark CommandObjectProcessStatus 1438bb9caf73SJim Ingham 14395a988416SJim Ingham class CommandObjectProcessStatus : public CommandObjectParsed 14404b9bea87SJim Ingham { 14414b9bea87SJim Ingham public: 1442a7015092SGreg Clayton CommandObjectProcessStatus (CommandInterpreter &interpreter) : 14435a988416SJim Ingham CommandObjectParsed (interpreter, 1444a7015092SGreg Clayton "process status", 1445e3d26315SCaroline Tice "Show the current status and location of executing process.", 1446e3d26315SCaroline Tice "process status", 1447f9fc609fSGreg Clayton eFlagRequiresProcess | eFlagTryTargetAPILock) 14484b9bea87SJim Ingham { 14494b9bea87SJim Ingham } 14504b9bea87SJim Ingham 14514b9bea87SJim Ingham ~CommandObjectProcessStatus() 14524b9bea87SJim Ingham { 14534b9bea87SJim Ingham } 14544b9bea87SJim Ingham 14554b9bea87SJim Ingham 14564b9bea87SJim Ingham bool 14575a988416SJim Ingham DoExecute (Args& command, CommandReturnObject &result) 14584b9bea87SJim Ingham { 14597260f620SGreg Clayton Stream &strm = result.GetOutputStream(); 14604b9bea87SJim Ingham result.SetStatus (eReturnStatusSuccessFinishNoResult); 1461f9fc609fSGreg Clayton // No need to check "process" for validity as eFlagRequiresProcess ensures it is valid 1462f9fc609fSGreg Clayton Process *process = m_exe_ctx.GetProcessPtr(); 14637260f620SGreg Clayton const bool only_threads_with_stop_reason = true; 14647260f620SGreg Clayton const uint32_t start_frame = 0; 14657260f620SGreg Clayton const uint32_t num_frames = 1; 14667260f620SGreg Clayton const uint32_t num_frames_with_source = 1; 1467c14ee32dSGreg Clayton process->GetStatus(strm); 1468c14ee32dSGreg Clayton process->GetThreadStatus (strm, 14697260f620SGreg Clayton only_threads_with_stop_reason, 14707260f620SGreg Clayton start_frame, 14717260f620SGreg Clayton num_frames, 14727260f620SGreg Clayton num_frames_with_source); 14734b9bea87SJim Ingham return result.Succeeded(); 14744b9bea87SJim Ingham } 14754b9bea87SJim Ingham }; 14764b9bea87SJim Ingham 14774b9bea87SJim Ingham //------------------------------------------------------------------------- 147835731357SCaroline Tice // CommandObjectProcessHandle 147935731357SCaroline Tice //------------------------------------------------------------------------- 1480bb9caf73SJim Ingham #pragma mark CommandObjectProcessHandle 148135731357SCaroline Tice 14825a988416SJim Ingham class CommandObjectProcessHandle : public CommandObjectParsed 148335731357SCaroline Tice { 148435731357SCaroline Tice public: 148535731357SCaroline Tice 148635731357SCaroline Tice class CommandOptions : public Options 148735731357SCaroline Tice { 148835731357SCaroline Tice public: 148935731357SCaroline Tice 1490eb0103f2SGreg Clayton CommandOptions (CommandInterpreter &interpreter) : 1491eb0103f2SGreg Clayton Options (interpreter) 149235731357SCaroline Tice { 1493f6b8b581SGreg Clayton OptionParsingStarting (); 149435731357SCaroline Tice } 149535731357SCaroline Tice 149635731357SCaroline Tice ~CommandOptions () 149735731357SCaroline Tice { 149835731357SCaroline Tice } 149935731357SCaroline Tice 150035731357SCaroline Tice Error 1501f6b8b581SGreg Clayton SetOptionValue (uint32_t option_idx, const char *option_arg) 150235731357SCaroline Tice { 150335731357SCaroline Tice Error error; 15043bcdfc0eSGreg Clayton const int short_option = m_getopt_table[option_idx].val; 150535731357SCaroline Tice 150635731357SCaroline Tice switch (short_option) 150735731357SCaroline Tice { 150835731357SCaroline Tice case 's': 150935731357SCaroline Tice stop = option_arg; 151035731357SCaroline Tice break; 151135731357SCaroline Tice case 'n': 151235731357SCaroline Tice notify = option_arg; 151335731357SCaroline Tice break; 151435731357SCaroline Tice case 'p': 151535731357SCaroline Tice pass = option_arg; 151635731357SCaroline Tice break; 151735731357SCaroline Tice default: 151886edbf41SGreg Clayton error.SetErrorStringWithFormat("invalid short option character '%c'", short_option); 151935731357SCaroline Tice break; 152035731357SCaroline Tice } 152135731357SCaroline Tice return error; 152235731357SCaroline Tice } 152335731357SCaroline Tice 152435731357SCaroline Tice void 1525f6b8b581SGreg Clayton OptionParsingStarting () 152635731357SCaroline Tice { 152735731357SCaroline Tice stop.clear(); 152835731357SCaroline Tice notify.clear(); 152935731357SCaroline Tice pass.clear(); 153035731357SCaroline Tice } 153135731357SCaroline Tice 1532e0d378b3SGreg Clayton const OptionDefinition* 153335731357SCaroline Tice GetDefinitions () 153435731357SCaroline Tice { 153535731357SCaroline Tice return g_option_table; 153635731357SCaroline Tice } 153735731357SCaroline Tice 153835731357SCaroline Tice // Options table: Required for subclasses of Options. 153935731357SCaroline Tice 1540e0d378b3SGreg Clayton static OptionDefinition g_option_table[]; 154135731357SCaroline Tice 154235731357SCaroline Tice // Instance variables to hold the values for command options. 154335731357SCaroline Tice 154435731357SCaroline Tice std::string stop; 154535731357SCaroline Tice std::string notify; 154635731357SCaroline Tice std::string pass; 154735731357SCaroline Tice }; 154835731357SCaroline Tice 154935731357SCaroline Tice 155035731357SCaroline Tice CommandObjectProcessHandle (CommandInterpreter &interpreter) : 15515a988416SJim Ingham CommandObjectParsed (interpreter, 155235731357SCaroline Tice "process handle", 155310ad7993SCaroline Tice "Show or update what the process and debugger should do with various signals received from the OS.", 1554eb0103f2SGreg Clayton NULL), 1555eb0103f2SGreg Clayton m_options (interpreter) 155635731357SCaroline Tice { 155710ad7993SCaroline Tice SetHelpLong ("If no signals are specified, update them all. If no update option is specified, list the current values.\n"); 155835731357SCaroline Tice CommandArgumentEntry arg; 1559c0dbdfb6SCaroline Tice CommandArgumentData signal_arg; 156035731357SCaroline Tice 1561c0dbdfb6SCaroline Tice signal_arg.arg_type = eArgTypeUnixSignal; 1562c0dbdfb6SCaroline Tice signal_arg.arg_repetition = eArgRepeatStar; 156335731357SCaroline Tice 1564c0dbdfb6SCaroline Tice arg.push_back (signal_arg); 156535731357SCaroline Tice 156635731357SCaroline Tice m_arguments.push_back (arg); 156735731357SCaroline Tice } 156835731357SCaroline Tice 156935731357SCaroline Tice ~CommandObjectProcessHandle () 157035731357SCaroline Tice { 157135731357SCaroline Tice } 157235731357SCaroline Tice 157335731357SCaroline Tice Options * 157435731357SCaroline Tice GetOptions () 157535731357SCaroline Tice { 157635731357SCaroline Tice return &m_options; 157735731357SCaroline Tice } 157835731357SCaroline Tice 157935731357SCaroline Tice bool 158010ad7993SCaroline Tice VerifyCommandOptionValue (const std::string &option, int &real_value) 158135731357SCaroline Tice { 158235731357SCaroline Tice bool okay = true; 158335731357SCaroline Tice 158410ad7993SCaroline Tice bool success = false; 158510ad7993SCaroline Tice bool tmp_value = Args::StringToBoolean (option.c_str(), false, &success); 158610ad7993SCaroline Tice 158710ad7993SCaroline Tice if (success && tmp_value) 158810ad7993SCaroline Tice real_value = 1; 158910ad7993SCaroline Tice else if (success && !tmp_value) 159010ad7993SCaroline Tice real_value = 0; 159135731357SCaroline Tice else 159235731357SCaroline Tice { 159335731357SCaroline Tice // If the value isn't 'true' or 'false', it had better be 0 or 1. 159410ad7993SCaroline Tice real_value = Args::StringToUInt32 (option.c_str(), 3); 159510ad7993SCaroline Tice if (real_value != 0 && real_value != 1) 159635731357SCaroline Tice okay = false; 159735731357SCaroline Tice } 159835731357SCaroline Tice 159935731357SCaroline Tice return okay; 160035731357SCaroline Tice } 160135731357SCaroline Tice 160210ad7993SCaroline Tice void 160310ad7993SCaroline Tice PrintSignalHeader (Stream &str) 160410ad7993SCaroline Tice { 160510ad7993SCaroline Tice str.Printf ("NAME PASS STOP NOTIFY\n"); 160610ad7993SCaroline Tice str.Printf ("========== ===== ===== ======\n"); 160710ad7993SCaroline Tice } 160810ad7993SCaroline Tice 160910ad7993SCaroline Tice void 161010ad7993SCaroline Tice PrintSignal (Stream &str, int32_t signo, const char *sig_name, UnixSignals &signals) 161110ad7993SCaroline Tice { 161210ad7993SCaroline Tice bool stop; 161310ad7993SCaroline Tice bool suppress; 161410ad7993SCaroline Tice bool notify; 161510ad7993SCaroline Tice 161610ad7993SCaroline Tice str.Printf ("%-10s ", sig_name); 161710ad7993SCaroline Tice if (signals.GetSignalInfo (signo, suppress, stop, notify)) 161810ad7993SCaroline Tice { 161910ad7993SCaroline Tice bool pass = !suppress; 162010ad7993SCaroline Tice str.Printf ("%s %s %s", 162110ad7993SCaroline Tice (pass ? "true " : "false"), 162210ad7993SCaroline Tice (stop ? "true " : "false"), 162310ad7993SCaroline Tice (notify ? "true " : "false")); 162410ad7993SCaroline Tice } 162510ad7993SCaroline Tice str.Printf ("\n"); 162610ad7993SCaroline Tice } 162710ad7993SCaroline Tice 162810ad7993SCaroline Tice void 162910ad7993SCaroline Tice PrintSignalInformation (Stream &str, Args &signal_args, int num_valid_signals, UnixSignals &signals) 163010ad7993SCaroline Tice { 163110ad7993SCaroline Tice PrintSignalHeader (str); 163210ad7993SCaroline Tice 163310ad7993SCaroline Tice if (num_valid_signals > 0) 163410ad7993SCaroline Tice { 163510ad7993SCaroline Tice size_t num_args = signal_args.GetArgumentCount(); 163610ad7993SCaroline Tice for (size_t i = 0; i < num_args; ++i) 163710ad7993SCaroline Tice { 163810ad7993SCaroline Tice int32_t signo = signals.GetSignalNumberFromName (signal_args.GetArgumentAtIndex (i)); 163910ad7993SCaroline Tice if (signo != LLDB_INVALID_SIGNAL_NUMBER) 164010ad7993SCaroline Tice PrintSignal (str, signo, signal_args.GetArgumentAtIndex (i), signals); 164110ad7993SCaroline Tice } 164210ad7993SCaroline Tice } 164310ad7993SCaroline Tice else // Print info for ALL signals 164410ad7993SCaroline Tice { 164510ad7993SCaroline Tice int32_t signo = signals.GetFirstSignalNumber(); 164610ad7993SCaroline Tice while (signo != LLDB_INVALID_SIGNAL_NUMBER) 164710ad7993SCaroline Tice { 164810ad7993SCaroline Tice PrintSignal (str, signo, signals.GetSignalAsCString (signo), signals); 164910ad7993SCaroline Tice signo = signals.GetNextSignalNumber (signo); 165010ad7993SCaroline Tice } 165110ad7993SCaroline Tice } 165210ad7993SCaroline Tice } 165310ad7993SCaroline Tice 16545a988416SJim Ingham protected: 165535731357SCaroline Tice bool 16565a988416SJim Ingham DoExecute (Args &signal_args, CommandReturnObject &result) 165735731357SCaroline Tice { 165835731357SCaroline Tice TargetSP target_sp = m_interpreter.GetDebugger().GetSelectedTarget(); 165935731357SCaroline Tice 166035731357SCaroline Tice if (!target_sp) 166135731357SCaroline Tice { 166235731357SCaroline Tice result.AppendError ("No current target;" 166335731357SCaroline Tice " cannot handle signals until you have a valid target and process.\n"); 166435731357SCaroline Tice result.SetStatus (eReturnStatusFailed); 166535731357SCaroline Tice return false; 166635731357SCaroline Tice } 166735731357SCaroline Tice 166835731357SCaroline Tice ProcessSP process_sp = target_sp->GetProcessSP(); 166935731357SCaroline Tice 167035731357SCaroline Tice if (!process_sp) 167135731357SCaroline Tice { 167235731357SCaroline Tice result.AppendError ("No current process; cannot handle signals until you have a valid process.\n"); 167335731357SCaroline Tice result.SetStatus (eReturnStatusFailed); 167435731357SCaroline Tice return false; 167535731357SCaroline Tice } 167635731357SCaroline Tice 167735731357SCaroline Tice int stop_action = -1; // -1 means leave the current setting alone 167835731357SCaroline Tice int pass_action = -1; // -1 means leave the current setting alone 167935731357SCaroline Tice int notify_action = -1; // -1 means leave the current setting alone 168035731357SCaroline Tice 168135731357SCaroline Tice if (! m_options.stop.empty() 168210ad7993SCaroline Tice && ! VerifyCommandOptionValue (m_options.stop, stop_action)) 168335731357SCaroline Tice { 168435731357SCaroline Tice result.AppendError ("Invalid argument for command option --stop; must be true or false.\n"); 168535731357SCaroline Tice result.SetStatus (eReturnStatusFailed); 168635731357SCaroline Tice return false; 168735731357SCaroline Tice } 168835731357SCaroline Tice 168935731357SCaroline Tice if (! m_options.notify.empty() 169010ad7993SCaroline Tice && ! VerifyCommandOptionValue (m_options.notify, notify_action)) 169135731357SCaroline Tice { 169235731357SCaroline Tice result.AppendError ("Invalid argument for command option --notify; must be true or false.\n"); 169335731357SCaroline Tice result.SetStatus (eReturnStatusFailed); 169435731357SCaroline Tice return false; 169535731357SCaroline Tice } 169635731357SCaroline Tice 169735731357SCaroline Tice if (! m_options.pass.empty() 169810ad7993SCaroline Tice && ! VerifyCommandOptionValue (m_options.pass, pass_action)) 169935731357SCaroline Tice { 170035731357SCaroline Tice result.AppendError ("Invalid argument for command option --pass; must be true or false.\n"); 170135731357SCaroline Tice result.SetStatus (eReturnStatusFailed); 170235731357SCaroline Tice return false; 170335731357SCaroline Tice } 170435731357SCaroline Tice 170535731357SCaroline Tice size_t num_args = signal_args.GetArgumentCount(); 170635731357SCaroline Tice UnixSignals &signals = process_sp->GetUnixSignals(); 170735731357SCaroline Tice int num_signals_set = 0; 170835731357SCaroline Tice 170910ad7993SCaroline Tice if (num_args > 0) 171010ad7993SCaroline Tice { 171135731357SCaroline Tice for (size_t i = 0; i < num_args; ++i) 171235731357SCaroline Tice { 171335731357SCaroline Tice int32_t signo = signals.GetSignalNumberFromName (signal_args.GetArgumentAtIndex (i)); 171435731357SCaroline Tice if (signo != LLDB_INVALID_SIGNAL_NUMBER) 171535731357SCaroline Tice { 171610ad7993SCaroline Tice // Casting the actions as bools here should be okay, because VerifyCommandOptionValue guarantees 171735731357SCaroline Tice // the value is either 0 or 1. 171835731357SCaroline Tice if (stop_action != -1) 171935731357SCaroline Tice signals.SetShouldStop (signo, (bool) stop_action); 172035731357SCaroline Tice if (pass_action != -1) 172135731357SCaroline Tice { 172210ad7993SCaroline Tice bool suppress = ! ((bool) pass_action); 172310ad7993SCaroline Tice signals.SetShouldSuppress (signo, suppress); 172435731357SCaroline Tice } 172535731357SCaroline Tice if (notify_action != -1) 172635731357SCaroline Tice signals.SetShouldNotify (signo, (bool) notify_action); 172735731357SCaroline Tice ++num_signals_set; 172835731357SCaroline Tice } 172935731357SCaroline Tice else 173035731357SCaroline Tice { 173135731357SCaroline Tice result.AppendErrorWithFormat ("Invalid signal name '%s'\n", signal_args.GetArgumentAtIndex (i)); 173235731357SCaroline Tice } 173335731357SCaroline Tice } 173410ad7993SCaroline Tice } 173510ad7993SCaroline Tice else 173610ad7993SCaroline Tice { 173710ad7993SCaroline Tice // No signal specified, if any command options were specified, update ALL signals. 173810ad7993SCaroline Tice if ((notify_action != -1) || (stop_action != -1) || (pass_action != -1)) 173910ad7993SCaroline Tice { 174010ad7993SCaroline Tice if (m_interpreter.Confirm ("Do you really want to update all the signals?", false)) 174110ad7993SCaroline Tice { 174210ad7993SCaroline Tice int32_t signo = signals.GetFirstSignalNumber(); 174310ad7993SCaroline Tice while (signo != LLDB_INVALID_SIGNAL_NUMBER) 174410ad7993SCaroline Tice { 174510ad7993SCaroline Tice if (notify_action != -1) 174610ad7993SCaroline Tice signals.SetShouldNotify (signo, (bool) notify_action); 174710ad7993SCaroline Tice if (stop_action != -1) 174810ad7993SCaroline Tice signals.SetShouldStop (signo, (bool) stop_action); 174910ad7993SCaroline Tice if (pass_action != -1) 175010ad7993SCaroline Tice { 175110ad7993SCaroline Tice bool suppress = ! ((bool) pass_action); 175210ad7993SCaroline Tice signals.SetShouldSuppress (signo, suppress); 175310ad7993SCaroline Tice } 175410ad7993SCaroline Tice signo = signals.GetNextSignalNumber (signo); 175510ad7993SCaroline Tice } 175610ad7993SCaroline Tice } 175710ad7993SCaroline Tice } 175810ad7993SCaroline Tice } 175910ad7993SCaroline Tice 176010ad7993SCaroline Tice PrintSignalInformation (result.GetOutputStream(), signal_args, num_signals_set, signals); 176135731357SCaroline Tice 176235731357SCaroline Tice if (num_signals_set > 0) 176335731357SCaroline Tice result.SetStatus (eReturnStatusSuccessFinishNoResult); 176435731357SCaroline Tice else 176535731357SCaroline Tice result.SetStatus (eReturnStatusFailed); 176635731357SCaroline Tice 176735731357SCaroline Tice return result.Succeeded(); 176835731357SCaroline Tice } 176935731357SCaroline Tice 177035731357SCaroline Tice CommandOptions m_options; 177135731357SCaroline Tice }; 177235731357SCaroline Tice 1773e0d378b3SGreg Clayton OptionDefinition 177435731357SCaroline Tice CommandObjectProcessHandle::CommandOptions::g_option_table[] = 177535731357SCaroline Tice { 177635731357SCaroline Tice { LLDB_OPT_SET_1, false, "stop", 's', required_argument, NULL, 0, eArgTypeBoolean, "Whether or not the process should be stopped if the signal is received." }, 177735731357SCaroline Tice { LLDB_OPT_SET_1, false, "notify", 'n', required_argument, NULL, 0, eArgTypeBoolean, "Whether or not the debugger should notify the user if the signal is received." }, 177835731357SCaroline Tice { LLDB_OPT_SET_1, false, "pass", 'p', required_argument, NULL, 0, eArgTypeBoolean, "Whether or not the signal should be passed to the process." }, 177935731357SCaroline Tice { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL } 178035731357SCaroline Tice }; 178135731357SCaroline Tice 178235731357SCaroline Tice //------------------------------------------------------------------------- 178330fdc8d8SChris Lattner // CommandObjectMultiwordProcess 178430fdc8d8SChris Lattner //------------------------------------------------------------------------- 178530fdc8d8SChris Lattner 17866611103cSGreg Clayton CommandObjectMultiwordProcess::CommandObjectMultiwordProcess (CommandInterpreter &interpreter) : 1787a7015092SGreg Clayton CommandObjectMultiword (interpreter, 1788a7015092SGreg Clayton "process", 178930fdc8d8SChris Lattner "A set of commands for operating on a process.", 179030fdc8d8SChris Lattner "process <subcommand> [<subcommand-options>]") 179130fdc8d8SChris Lattner { 1792a7015092SGreg Clayton LoadSubCommand ("attach", CommandObjectSP (new CommandObjectProcessAttach (interpreter))); 1793a7015092SGreg Clayton LoadSubCommand ("launch", CommandObjectSP (new CommandObjectProcessLaunch (interpreter))); 1794a7015092SGreg Clayton LoadSubCommand ("continue", CommandObjectSP (new CommandObjectProcessContinue (interpreter))); 1795b766a73dSGreg Clayton LoadSubCommand ("connect", CommandObjectSP (new CommandObjectProcessConnect (interpreter))); 1796a7015092SGreg Clayton LoadSubCommand ("detach", CommandObjectSP (new CommandObjectProcessDetach (interpreter))); 17978f343b09SGreg Clayton LoadSubCommand ("load", CommandObjectSP (new CommandObjectProcessLoad (interpreter))); 17988f343b09SGreg Clayton LoadSubCommand ("unload", CommandObjectSP (new CommandObjectProcessUnload (interpreter))); 1799a7015092SGreg Clayton LoadSubCommand ("signal", CommandObjectSP (new CommandObjectProcessSignal (interpreter))); 180035731357SCaroline Tice LoadSubCommand ("handle", CommandObjectSP (new CommandObjectProcessHandle (interpreter))); 1801a7015092SGreg Clayton LoadSubCommand ("status", CommandObjectSP (new CommandObjectProcessStatus (interpreter))); 1802a7015092SGreg Clayton LoadSubCommand ("interrupt", CommandObjectSP (new CommandObjectProcessInterrupt (interpreter))); 1803a7015092SGreg Clayton LoadSubCommand ("kill", CommandObjectSP (new CommandObjectProcessKill (interpreter))); 1804998255bfSGreg Clayton LoadSubCommand ("plugin", CommandObjectSP (new CommandObjectProcessPlugin (interpreter))); 180530fdc8d8SChris Lattner } 180630fdc8d8SChris Lattner 180730fdc8d8SChris Lattner CommandObjectMultiwordProcess::~CommandObjectMultiwordProcess () 180830fdc8d8SChris Lattner { 180930fdc8d8SChris Lattner } 181030fdc8d8SChris Lattner 1811