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 1030fdc8d8SChris Lattner #include "CommandObjectProcess.h" 1130fdc8d8SChris Lattner 1230fdc8d8SChris Lattner // C Includes 1330fdc8d8SChris Lattner // C++ Includes 1430fdc8d8SChris Lattner // Other libraries and framework includes 1530fdc8d8SChris Lattner // Project includes 160e41084aSJim Ingham #include "lldb/Breakpoint/Breakpoint.h" 170e41084aSJim Ingham #include "lldb/Breakpoint/BreakpointLocation.h" 180e41084aSJim Ingham #include "lldb/Breakpoint/BreakpointSite.h" 1930fdc8d8SChris Lattner #include "lldb/Core/State.h" 201f746071SGreg Clayton #include "lldb/Core/Module.h" 217260f620SGreg Clayton #include "lldb/Host/Host.h" 220e41084aSJim Ingham #include "lldb/Interpreter/Args.h" 230e41084aSJim Ingham #include "lldb/Interpreter/Options.h" 2430fdc8d8SChris Lattner #include "lldb/Interpreter/CommandInterpreter.h" 2530fdc8d8SChris Lattner #include "lldb/Interpreter/CommandReturnObject.h" 26e996fd30SGreg Clayton #include "lldb/Target/Platform.h" 2730fdc8d8SChris Lattner #include "lldb/Target/Process.h" 280e41084aSJim Ingham #include "lldb/Target/StopInfo.h" 2930fdc8d8SChris Lattner #include "lldb/Target/Target.h" 3030fdc8d8SChris Lattner #include "lldb/Target/Thread.h" 3130fdc8d8SChris Lattner 3230fdc8d8SChris Lattner using namespace lldb; 3330fdc8d8SChris Lattner using namespace lldb_private; 3430fdc8d8SChris Lattner 3530fdc8d8SChris Lattner //------------------------------------------------------------------------- 3630fdc8d8SChris Lattner // CommandObjectProcessLaunch 3730fdc8d8SChris Lattner //------------------------------------------------------------------------- 384bddaeb5SJim Ingham #pragma mark CommandObjectProcessLaunch 395a988416SJim Ingham class CommandObjectProcessLaunch : public CommandObjectParsed 4030fdc8d8SChris Lattner { 4130fdc8d8SChris Lattner public: 4230fdc8d8SChris Lattner 43a7015092SGreg Clayton CommandObjectProcessLaunch (CommandInterpreter &interpreter) : 445a988416SJim Ingham CommandObjectParsed (interpreter, 45a7015092SGreg Clayton "process launch", 46e3d26315SCaroline Tice "Launch the executable in the debugger.", 47eb0103f2SGreg Clayton NULL), 48eb0103f2SGreg Clayton m_options (interpreter) 4930fdc8d8SChris Lattner { 50405fe67fSCaroline Tice CommandArgumentEntry arg; 51405fe67fSCaroline Tice CommandArgumentData run_args_arg; 52405fe67fSCaroline Tice 53405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 54405fe67fSCaroline Tice run_args_arg.arg_type = eArgTypeRunArgs; 55405fe67fSCaroline Tice run_args_arg.arg_repetition = eArgRepeatOptional; 56405fe67fSCaroline Tice 57405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 58405fe67fSCaroline Tice arg.push_back (run_args_arg); 59405fe67fSCaroline Tice 60405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 61405fe67fSCaroline Tice m_arguments.push_back (arg); 6230fdc8d8SChris Lattner } 6330fdc8d8SChris Lattner 6430fdc8d8SChris Lattner 6530fdc8d8SChris Lattner ~CommandObjectProcessLaunch () 6630fdc8d8SChris Lattner { 6730fdc8d8SChris Lattner } 6830fdc8d8SChris Lattner 69e9ce62b6SJim Ingham int 70e9ce62b6SJim Ingham HandleArgumentCompletion (Args &input, 71e9ce62b6SJim Ingham int &cursor_index, 72e9ce62b6SJim Ingham int &cursor_char_position, 73e9ce62b6SJim Ingham OptionElementVector &opt_element_vector, 74e9ce62b6SJim Ingham int match_start_point, 75e9ce62b6SJim Ingham int max_return_elements, 76e9ce62b6SJim Ingham bool &word_complete, 77e9ce62b6SJim Ingham StringList &matches) 78e9ce62b6SJim Ingham { 79e9ce62b6SJim Ingham std::string completion_str (input.GetArgumentAtIndex(cursor_index)); 80e9ce62b6SJim Ingham completion_str.erase (cursor_char_position); 81e9ce62b6SJim Ingham 82e9ce62b6SJim Ingham CommandCompletions::InvokeCommonCompletionCallbacks (m_interpreter, 83e9ce62b6SJim Ingham CommandCompletions::eDiskFileCompletion, 84e9ce62b6SJim Ingham completion_str.c_str(), 85e9ce62b6SJim Ingham match_start_point, 86e9ce62b6SJim Ingham max_return_elements, 87e9ce62b6SJim Ingham NULL, 88e9ce62b6SJim Ingham word_complete, 89e9ce62b6SJim Ingham matches); 90e9ce62b6SJim Ingham return matches.GetSize(); 91e9ce62b6SJim Ingham } 92e9ce62b6SJim Ingham 9330fdc8d8SChris Lattner Options * 9430fdc8d8SChris Lattner GetOptions () 9530fdc8d8SChris Lattner { 9630fdc8d8SChris Lattner return &m_options; 9730fdc8d8SChris Lattner } 9830fdc8d8SChris Lattner 995a988416SJim Ingham virtual const char *GetRepeatCommand (Args ¤t_command_args, uint32_t index) 1005a988416SJim Ingham { 1015a988416SJim Ingham // No repeat for "process launch"... 1025a988416SJim Ingham return ""; 1035a988416SJim Ingham } 1045a988416SJim Ingham 1055a988416SJim Ingham protected: 10630fdc8d8SChris Lattner bool 1075a988416SJim Ingham DoExecute (Args& launch_args, CommandReturnObject &result) 10830fdc8d8SChris Lattner { 1091d885966SGreg Clayton Debugger &debugger = m_interpreter.GetDebugger(); 1101d885966SGreg Clayton Target *target = debugger.GetSelectedTarget().get(); 1111d885966SGreg Clayton Error error; 11230fdc8d8SChris Lattner 11330fdc8d8SChris Lattner if (target == NULL) 11430fdc8d8SChris Lattner { 115effe5c95SGreg Clayton result.AppendError ("invalid target, create a debug target using the 'target create' command"); 11630fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 11730fdc8d8SChris Lattner return false; 11830fdc8d8SChris Lattner } 11930fdc8d8SChris Lattner // If our listener is NULL, users aren't allows to launch 12030fdc8d8SChris Lattner char filename[PATH_MAX]; 121aa149cbdSGreg Clayton const Module *exe_module = target->GetExecutableModulePointer(); 12271337622SGreg Clayton 12371337622SGreg Clayton if (exe_module == NULL) 12471337622SGreg Clayton { 125effe5c95SGreg Clayton result.AppendError ("no file in target, create a debug target using the 'target create' command"); 12671337622SGreg Clayton result.SetStatus (eReturnStatusFailed); 12771337622SGreg Clayton return false; 12871337622SGreg Clayton } 12971337622SGreg Clayton 13030fdc8d8SChris Lattner exe_module->GetFileSpec().GetPath (filename, sizeof(filename)); 13130fdc8d8SChris Lattner 132982c9762SGreg Clayton const bool add_exe_file_as_first_arg = true; 133ad9e828cSGreg Clayton m_options.launch_info.SetExecutableFile(exe_module->GetPlatformFileSpec(), add_exe_file_as_first_arg); 134982c9762SGreg Clayton 13571337622SGreg Clayton StateType state = eStateInvalid; 136c14ee32dSGreg Clayton Process *process = m_interpreter.GetExecutionContext().GetProcessPtr(); 13771337622SGreg Clayton if (process) 13871337622SGreg Clayton { 13971337622SGreg Clayton state = process->GetState(); 14071337622SGreg Clayton 14171337622SGreg Clayton if (process->IsAlive() && state != eStateConnected) 14230fdc8d8SChris Lattner { 143513c26ceSGreg Clayton char message[1024]; 144513c26ceSGreg Clayton if (process->GetState() == eStateAttaching) 145513c26ceSGreg Clayton ::strncpy (message, "There is a pending attach, abort it and launch a new process?", sizeof(message)); 146513c26ceSGreg Clayton else 147513c26ceSGreg Clayton ::strncpy (message, "There is a running process, kill it and restart?", sizeof(message)); 148513c26ceSGreg Clayton 149513c26ceSGreg Clayton if (!m_interpreter.Confirm (message, true)) 150bb9caf73SJim Ingham { 15130fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 15230fdc8d8SChris Lattner return false; 15330fdc8d8SChris Lattner } 154bb9caf73SJim Ingham else 155bb9caf73SJim Ingham { 1561d885966SGreg Clayton Error destroy_error (process->Destroy()); 1571d885966SGreg Clayton if (destroy_error.Success()) 158bb9caf73SJim Ingham { 159bb9caf73SJim Ingham result.SetStatus (eReturnStatusSuccessFinishResult); 160bb9caf73SJim Ingham } 161bb9caf73SJim Ingham else 162bb9caf73SJim Ingham { 1631d885966SGreg Clayton result.AppendErrorWithFormat ("Failed to kill process: %s\n", destroy_error.AsCString()); 164bb9caf73SJim Ingham result.SetStatus (eReturnStatusFailed); 165bb9caf73SJim Ingham } 166bb9caf73SJim Ingham } 167bb9caf73SJim Ingham } 16871337622SGreg Clayton } 16930fdc8d8SChris Lattner 170144f3a9cSGreg Clayton if (launch_args.GetArgumentCount() == 0) 171144f3a9cSGreg Clayton { 17267cc0636SGreg Clayton Args target_setting_args; 17367cc0636SGreg Clayton if (target->GetRunArguments(target_setting_args) > 0) 17467cc0636SGreg Clayton m_options.launch_info.GetArguments().AppendArguments (target_setting_args); 175144f3a9cSGreg Clayton } 176144f3a9cSGreg Clayton else 17730fdc8d8SChris Lattner { 178162b597cSGreg Clayton // Save the arguments for subsequent runs in the current target. 179162b597cSGreg Clayton target->SetRunArguments (launch_args); 180162b597cSGreg Clayton 181982c9762SGreg Clayton m_options.launch_info.GetArguments().AppendArguments (launch_args); 182982c9762SGreg Clayton } 1831d885966SGreg Clayton 184144f3a9cSGreg Clayton if (target->GetDisableASLR()) 185144f3a9cSGreg Clayton m_options.launch_info.GetFlags().Set (eLaunchFlagDisableASLR); 186144f3a9cSGreg Clayton 187144f3a9cSGreg Clayton if (target->GetDisableSTDIO()) 188144f3a9cSGreg Clayton m_options.launch_info.GetFlags().Set (eLaunchFlagDisableSTDIO); 189144f3a9cSGreg Clayton 190144f3a9cSGreg Clayton m_options.launch_info.GetFlags().Set (eLaunchFlagDebug); 191144f3a9cSGreg Clayton 192144f3a9cSGreg Clayton Args environment; 193144f3a9cSGreg Clayton target->GetEnvironmentAsArgs (environment); 194144f3a9cSGreg Clayton if (environment.GetArgumentCount() > 0) 195144f3a9cSGreg Clayton m_options.launch_info.GetEnvironmentEntries ().AppendArguments (environment); 196144f3a9cSGreg Clayton 197ee95ed50SGreg Clayton // Finalize the file actions, and if none were given, default to opening 198ee95ed50SGreg Clayton // up a pseudo terminal 199ee95ed50SGreg Clayton const bool default_to_use_pty = true; 200ee95ed50SGreg Clayton m_options.launch_info.FinalizeFileActions (target, default_to_use_pty); 2011d885966SGreg Clayton 2021d885966SGreg Clayton if (state == eStateConnected) 2031d885966SGreg Clayton { 2041d885966SGreg Clayton if (m_options.launch_info.GetFlags().Test (eLaunchFlagLaunchInTTY)) 2051d885966SGreg Clayton { 2061d885966SGreg Clayton result.AppendWarning("can't launch in tty when launching through a remote connection"); 2071d885966SGreg Clayton m_options.launch_info.GetFlags().Clear (eLaunchFlagLaunchInTTY); 2081d885966SGreg Clayton } 2091d885966SGreg Clayton } 210982c9762SGreg Clayton else 211982c9762SGreg Clayton { 212144f3a9cSGreg Clayton if (!m_options.launch_info.GetArchitecture().IsValid()) 213c9ed478aSGreg Clayton m_options.launch_info.GetArchitecture() = target->GetArchitecture(); 214c9ed478aSGreg Clayton 215c982b3d6SGreg Clayton PlatformSP platform_sp (target->GetPlatform()); 216c982b3d6SGreg Clayton 217c982b3d6SGreg Clayton if (platform_sp && platform_sp->CanDebugProcess ()) 218c982b3d6SGreg Clayton { 2191d885966SGreg Clayton process = target->GetPlatform()->DebugProcess (m_options.launch_info, 2201d885966SGreg Clayton debugger, 2211d885966SGreg Clayton target, 2221d885966SGreg Clayton debugger.GetListener(), 2231d885966SGreg Clayton error).get(); 224c982b3d6SGreg Clayton } 225c982b3d6SGreg Clayton else 226c982b3d6SGreg Clayton { 227c982b3d6SGreg Clayton const char *plugin_name = m_options.launch_info.GetProcessPluginName(); 228c3776bf2SGreg Clayton process = target->CreateProcess (debugger.GetListener(), plugin_name, NULL).get(); 229c982b3d6SGreg Clayton if (process) 230c982b3d6SGreg Clayton error = process->Launch (m_options.launch_info); 231c982b3d6SGreg Clayton } 2321d885966SGreg Clayton 2331d885966SGreg Clayton if (process == NULL) 2341d885966SGreg Clayton { 235144f3a9cSGreg Clayton result.SetError (error, "failed to launch or debug process"); 2361d885966SGreg Clayton return false; 2371d885966SGreg Clayton } 2381d885966SGreg Clayton } 23930fdc8d8SChris Lattner 24030fdc8d8SChris Lattner if (error.Success()) 24130fdc8d8SChris Lattner { 24264195a2cSGreg Clayton const char *archname = exe_module->GetArchitecture().GetArchitectureName(); 24319388cfcSGreg Clayton 24481c22f61SGreg Clayton result.AppendMessageWithFormat ("Process %llu launched: '%s' (%s)\n", process->GetID(), filename, archname); 24505faeb71SGreg Clayton result.SetDidChangeProcessState (true); 246982c9762SGreg Clayton if (m_options.launch_info.GetFlags().Test(eLaunchFlagStopAtEntry) == false) 24730fdc8d8SChris Lattner { 24805faeb71SGreg Clayton result.SetStatus (eReturnStatusSuccessContinuingNoResult); 24930fdc8d8SChris Lattner StateType state = process->WaitForProcessToStop (NULL); 25030fdc8d8SChris Lattner 25130fdc8d8SChris Lattner if (state == eStateStopped) 25230fdc8d8SChris Lattner { 25305faeb71SGreg Clayton error = process->Resume(); 25405faeb71SGreg Clayton if (error.Success()) 25505faeb71SGreg Clayton { 25605faeb71SGreg Clayton bool synchronous_execution = m_interpreter.GetSynchronous (); 25730fdc8d8SChris Lattner if (synchronous_execution) 25830fdc8d8SChris Lattner { 25905faeb71SGreg Clayton state = process->WaitForProcessToStop (NULL); 2602637f825SGreg Clayton const bool must_be_alive = true; 2612637f825SGreg Clayton if (!StateIsStoppedState(state, must_be_alive)) 262514487e8SGreg Clayton { 263144f3a9cSGreg Clayton result.AppendErrorWithFormat ("process isn't stopped: %s", StateAsCString(state)); 264514487e8SGreg Clayton } 26530fdc8d8SChris Lattner result.SetDidChangeProcessState (true); 26605faeb71SGreg Clayton result.SetStatus (eReturnStatusSuccessFinishResult); 26705faeb71SGreg Clayton } 26805faeb71SGreg Clayton else 26905faeb71SGreg Clayton { 27005faeb71SGreg Clayton result.SetStatus (eReturnStatusSuccessContinuingNoResult); 27105faeb71SGreg Clayton } 27205faeb71SGreg Clayton } 273514487e8SGreg Clayton else 274514487e8SGreg Clayton { 275144f3a9cSGreg Clayton result.AppendErrorWithFormat ("process resume at entry point failed: %s", error.AsCString()); 276514487e8SGreg Clayton result.SetStatus (eReturnStatusFailed); 27730fdc8d8SChris Lattner } 27830fdc8d8SChris Lattner } 279514487e8SGreg Clayton else 280514487e8SGreg Clayton { 281144f3a9cSGreg Clayton result.AppendErrorWithFormat ("initial process state wasn't stopped: %s", StateAsCString(state)); 282514487e8SGreg Clayton result.SetStatus (eReturnStatusFailed); 283514487e8SGreg Clayton } 284514487e8SGreg Clayton } 285514487e8SGreg Clayton } 286514487e8SGreg Clayton else 287514487e8SGreg Clayton { 288197bacffSGreg Clayton result.AppendErrorWithFormat ("process launch failed: %s", error.AsCString()); 289514487e8SGreg Clayton result.SetStatus (eReturnStatusFailed); 29030fdc8d8SChris Lattner } 29130fdc8d8SChris Lattner 29230fdc8d8SChris Lattner return result.Succeeded(); 29330fdc8d8SChris Lattner } 29430fdc8d8SChris Lattner 29530fdc8d8SChris Lattner protected: 296982c9762SGreg Clayton ProcessLaunchCommandOptions m_options; 29730fdc8d8SChris Lattner }; 29830fdc8d8SChris Lattner 29930fdc8d8SChris Lattner 300982c9762SGreg Clayton //#define SET1 LLDB_OPT_SET_1 301982c9762SGreg Clayton //#define SET2 LLDB_OPT_SET_2 302982c9762SGreg Clayton //#define SET3 LLDB_OPT_SET_3 303982c9762SGreg Clayton // 304982c9762SGreg Clayton //OptionDefinition 305982c9762SGreg Clayton //CommandObjectProcessLaunch::CommandOptions::g_option_table[] = 306982c9762SGreg Clayton //{ 307982c9762SGreg 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."}, 308982c9762SGreg Clayton //{ SET1 , false, "stdin", 'i', required_argument, NULL, 0, eArgTypePath, "Redirect stdin for the process to <path>."}, 309982c9762SGreg Clayton //{ SET1 , false, "stdout", 'o', required_argument, NULL, 0, eArgTypePath, "Redirect stdout for the process to <path>."}, 310982c9762SGreg Clayton //{ SET1 , false, "stderr", 'e', required_argument, NULL, 0, eArgTypePath, "Redirect stderr for the process to <path>."}, 311982c9762SGreg Clayton //{ SET1 | SET2 | SET3, false, "plugin", 'p', required_argument, NULL, 0, eArgTypePlugin, "Name of the process plugin you want to use."}, 312982c9762SGreg Clayton //{ SET2 , false, "tty", 't', optional_argument, NULL, 0, eArgTypePath, "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."}, 313982c9762SGreg Clayton //{ SET3, false, "no-stdio", 'n', no_argument, NULL, 0, eArgTypeNone, "Do not set up for terminal I/O to go to running process."}, 314982c9762SGreg Clayton //{ SET1 | SET2 | SET3, false, "working-dir", 'w', required_argument, NULL, 0, eArgTypePath, "Set the current working directory to <path> when running the inferior."}, 315982c9762SGreg Clayton //{ 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL } 316982c9762SGreg Clayton //}; 317982c9762SGreg Clayton // 318982c9762SGreg Clayton //#undef SET1 319982c9762SGreg Clayton //#undef SET2 320982c9762SGreg Clayton //#undef SET3 32130fdc8d8SChris Lattner 32230fdc8d8SChris Lattner //------------------------------------------------------------------------- 32330fdc8d8SChris Lattner // CommandObjectProcessAttach 32430fdc8d8SChris Lattner //------------------------------------------------------------------------- 325bb9caf73SJim Ingham #pragma mark CommandObjectProcessAttach 3265a988416SJim Ingham class CommandObjectProcessAttach : public CommandObjectParsed 32730fdc8d8SChris Lattner { 32830fdc8d8SChris Lattner public: 32930fdc8d8SChris Lattner 33030fdc8d8SChris Lattner class CommandOptions : public Options 33130fdc8d8SChris Lattner { 33230fdc8d8SChris Lattner public: 33330fdc8d8SChris Lattner 334eb0103f2SGreg Clayton CommandOptions (CommandInterpreter &interpreter) : 335eb0103f2SGreg Clayton Options(interpreter) 33630fdc8d8SChris Lattner { 337f6b8b581SGreg Clayton // Keep default values of all options in one place: OptionParsingStarting () 338f6b8b581SGreg Clayton OptionParsingStarting (); 33930fdc8d8SChris Lattner } 34030fdc8d8SChris Lattner 34130fdc8d8SChris Lattner ~CommandOptions () 34230fdc8d8SChris Lattner { 34330fdc8d8SChris Lattner } 34430fdc8d8SChris Lattner 34530fdc8d8SChris Lattner Error 346f6b8b581SGreg Clayton SetOptionValue (uint32_t option_idx, const char *option_arg) 34730fdc8d8SChris Lattner { 34830fdc8d8SChris Lattner Error error; 34930fdc8d8SChris Lattner char short_option = (char) m_getopt_table[option_idx].val; 35030fdc8d8SChris Lattner bool success = false; 35130fdc8d8SChris Lattner switch (short_option) 35230fdc8d8SChris Lattner { 353a95ce623SJohnny Chen case 'c': 354a95ce623SJohnny Chen attach_info.SetContinueOnceAttached(true); 355a95ce623SJohnny Chen break; 356a95ce623SJohnny Chen 35730fdc8d8SChris Lattner case 'p': 358144f3a9cSGreg Clayton { 359144f3a9cSGreg Clayton lldb::pid_t pid = Args::StringToUInt32 (option_arg, LLDB_INVALID_PROCESS_ID, 0, &success); 36030fdc8d8SChris Lattner if (!success || pid == LLDB_INVALID_PROCESS_ID) 36130fdc8d8SChris Lattner { 36286edbf41SGreg Clayton error.SetErrorStringWithFormat("invalid process ID '%s'", option_arg); 36330fdc8d8SChris Lattner } 364144f3a9cSGreg Clayton else 365144f3a9cSGreg Clayton { 366144f3a9cSGreg Clayton attach_info.SetProcessID (pid); 367144f3a9cSGreg Clayton } 368144f3a9cSGreg Clayton } 36930fdc8d8SChris Lattner break; 37030fdc8d8SChris Lattner 37130fdc8d8SChris Lattner case 'P': 372144f3a9cSGreg Clayton attach_info.SetProcessPluginName (option_arg); 37330fdc8d8SChris Lattner break; 37430fdc8d8SChris Lattner 37530fdc8d8SChris Lattner case 'n': 376144f3a9cSGreg Clayton attach_info.GetExecutableFile().SetFile(option_arg, false); 37730fdc8d8SChris Lattner break; 37830fdc8d8SChris Lattner 37930fdc8d8SChris Lattner case 'w': 380144f3a9cSGreg Clayton attach_info.SetWaitForLaunch(true); 38130fdc8d8SChris Lattner break; 38230fdc8d8SChris Lattner 383cd16df91SJim Ingham case 'i': 384cd16df91SJim Ingham attach_info.SetIgnoreExisting(false); 385cd16df91SJim Ingham break; 386cd16df91SJim Ingham 38730fdc8d8SChris Lattner default: 38886edbf41SGreg Clayton error.SetErrorStringWithFormat("invalid short option character '%c'", short_option); 38930fdc8d8SChris Lattner break; 39030fdc8d8SChris Lattner } 39130fdc8d8SChris Lattner return error; 39230fdc8d8SChris Lattner } 39330fdc8d8SChris Lattner 39430fdc8d8SChris Lattner void 395f6b8b581SGreg Clayton OptionParsingStarting () 39630fdc8d8SChris Lattner { 397144f3a9cSGreg Clayton attach_info.Clear(); 39830fdc8d8SChris Lattner } 39930fdc8d8SChris Lattner 400e0d378b3SGreg Clayton const OptionDefinition* 40130fdc8d8SChris Lattner GetDefinitions () 40230fdc8d8SChris Lattner { 40330fdc8d8SChris Lattner return g_option_table; 40430fdc8d8SChris Lattner } 40530fdc8d8SChris Lattner 4065aee162fSJim Ingham virtual bool 407eb0103f2SGreg Clayton HandleOptionArgumentCompletion (Args &input, 4085aee162fSJim Ingham int cursor_index, 4095aee162fSJim Ingham int char_pos, 4105aee162fSJim Ingham OptionElementVector &opt_element_vector, 4115aee162fSJim Ingham int opt_element_index, 4125aee162fSJim Ingham int match_start_point, 4135aee162fSJim Ingham int max_return_elements, 4145aee162fSJim Ingham bool &word_complete, 4155aee162fSJim Ingham StringList &matches) 4165aee162fSJim Ingham { 4175aee162fSJim Ingham int opt_arg_pos = opt_element_vector[opt_element_index].opt_arg_pos; 4185aee162fSJim Ingham int opt_defs_index = opt_element_vector[opt_element_index].opt_defs_index; 4195aee162fSJim Ingham 4205aee162fSJim Ingham // We are only completing the name option for now... 4215aee162fSJim Ingham 422e0d378b3SGreg Clayton const OptionDefinition *opt_defs = GetDefinitions(); 4235aee162fSJim Ingham if (opt_defs[opt_defs_index].short_option == 'n') 4245aee162fSJim Ingham { 4255aee162fSJim Ingham // Are we in the name? 4265aee162fSJim Ingham 4275aee162fSJim Ingham // Look to see if there is a -P argument provided, and if so use that plugin, otherwise 4285aee162fSJim Ingham // use the default plugin. 4295aee162fSJim Ingham 4305aee162fSJim Ingham const char *partial_name = NULL; 4315aee162fSJim Ingham partial_name = input.GetArgumentAtIndex(opt_arg_pos); 4325aee162fSJim Ingham 4338b82f087SGreg Clayton PlatformSP platform_sp (m_interpreter.GetPlatform (true)); 434e996fd30SGreg Clayton if (platform_sp) 4355aee162fSJim Ingham { 4368b82f087SGreg Clayton ProcessInstanceInfoList process_infos; 4378b82f087SGreg Clayton ProcessInstanceInfoMatch match_info; 43832e0a750SGreg Clayton if (partial_name) 43932e0a750SGreg Clayton { 440144f3a9cSGreg Clayton match_info.GetProcessInfo().GetExecutableFile().SetFile(partial_name, false); 44132e0a750SGreg Clayton match_info.SetNameMatchType(eNameMatchStartsWith); 44232e0a750SGreg Clayton } 44332e0a750SGreg Clayton platform_sp->FindProcesses (match_info, process_infos); 444e996fd30SGreg Clayton const uint32_t num_matches = process_infos.GetSize(); 445e996fd30SGreg Clayton if (num_matches > 0) 446e996fd30SGreg Clayton { 447e996fd30SGreg Clayton for (uint32_t i=0; i<num_matches; ++i) 448e996fd30SGreg Clayton { 449e996fd30SGreg Clayton matches.AppendString (process_infos.GetProcessNameAtIndex(i), 450e996fd30SGreg Clayton process_infos.GetProcessNameLengthAtIndex(i)); 4515aee162fSJim Ingham } 4525aee162fSJim Ingham } 4535aee162fSJim Ingham } 4545aee162fSJim Ingham } 4555aee162fSJim Ingham 4565aee162fSJim Ingham return false; 4575aee162fSJim Ingham } 4585aee162fSJim Ingham 45930fdc8d8SChris Lattner // Options table: Required for subclasses of Options. 46030fdc8d8SChris Lattner 461e0d378b3SGreg Clayton static OptionDefinition g_option_table[]; 46230fdc8d8SChris Lattner 46330fdc8d8SChris Lattner // Instance variables to hold the values for command options. 46430fdc8d8SChris Lattner 465144f3a9cSGreg Clayton ProcessAttachInfo attach_info; 46630fdc8d8SChris Lattner }; 46730fdc8d8SChris Lattner 468a7015092SGreg Clayton CommandObjectProcessAttach (CommandInterpreter &interpreter) : 4695a988416SJim Ingham CommandObjectParsed (interpreter, 470a7015092SGreg Clayton "process attach", 471e3d26315SCaroline Tice "Attach to a process.", 472eb0103f2SGreg Clayton "process attach <cmd-options>"), 473eb0103f2SGreg Clayton m_options (interpreter) 4745aee162fSJim Ingham { 4755aee162fSJim Ingham } 4765aee162fSJim Ingham 4775aee162fSJim Ingham ~CommandObjectProcessAttach () 4785aee162fSJim Ingham { 4795aee162fSJim Ingham } 4805aee162fSJim Ingham 4815a988416SJim Ingham Options * 4825a988416SJim Ingham GetOptions () 4835a988416SJim Ingham { 4845a988416SJim Ingham return &m_options; 4855a988416SJim Ingham } 4865a988416SJim Ingham 4875a988416SJim Ingham protected: 4885aee162fSJim Ingham bool 4895a988416SJim Ingham DoExecute (Args& command, 4905aee162fSJim Ingham CommandReturnObject &result) 4915aee162fSJim Ingham { 492a7015092SGreg Clayton Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get(); 49331412642SJim Ingham // N.B. The attach should be synchronous. It doesn't help much to get the prompt back between initiating the attach 49431412642SJim Ingham // and the target actually stopping. So even if the interpreter is set to be asynchronous, we wait for the stop 49531412642SJim Ingham // ourselves here. 4965aee162fSJim Ingham 497c14ee32dSGreg Clayton Process *process = m_interpreter.GetExecutionContext().GetProcessPtr(); 49871337622SGreg Clayton StateType state = eStateInvalid; 4995aee162fSJim Ingham if (process) 5005aee162fSJim Ingham { 50171337622SGreg Clayton state = process->GetState(); 50271337622SGreg Clayton if (process->IsAlive() && state != eStateConnected) 5035aee162fSJim Ingham { 50481c22f61SGreg Clayton result.AppendErrorWithFormat ("Process %llu is currently being debugged, kill the process before attaching.\n", 5055aee162fSJim Ingham process->GetID()); 5065aee162fSJim Ingham result.SetStatus (eReturnStatusFailed); 5075aee162fSJim Ingham return false; 5085aee162fSJim Ingham } 5095aee162fSJim Ingham } 5105aee162fSJim Ingham 5115aee162fSJim Ingham if (target == NULL) 5125aee162fSJim Ingham { 5135aee162fSJim Ingham // If there isn't a current target create one. 5145aee162fSJim Ingham TargetSP new_target_sp; 5155aee162fSJim Ingham FileSpec emptyFileSpec; 5165aee162fSJim Ingham Error error; 5175aee162fSJim Ingham 518a7015092SGreg Clayton error = m_interpreter.GetDebugger().GetTargetList().CreateTarget (m_interpreter.GetDebugger(), 5195aee162fSJim Ingham emptyFileSpec, 520cac9c5f9SGreg Clayton NULL, 5215aee162fSJim Ingham false, 522cac9c5f9SGreg Clayton NULL, // No platform options 5235aee162fSJim Ingham new_target_sp); 5245aee162fSJim Ingham target = new_target_sp.get(); 5255aee162fSJim Ingham if (target == NULL || error.Fail()) 5265aee162fSJim Ingham { 527b766a73dSGreg Clayton result.AppendError(error.AsCString("Error creating target")); 5285aee162fSJim Ingham return false; 5295aee162fSJim Ingham } 530a7015092SGreg Clayton m_interpreter.GetDebugger().GetTargetList().SetSelectedTarget(target); 5315aee162fSJim Ingham } 5325aee162fSJim Ingham 5335aee162fSJim Ingham // Record the old executable module, we want to issue a warning if the process of attaching changed the 5345aee162fSJim Ingham // current executable (like somebody said "file foo" then attached to a PID whose executable was bar.) 5355aee162fSJim Ingham 5365aee162fSJim Ingham ModuleSP old_exec_module_sp = target->GetExecutableModule(); 5375aee162fSJim Ingham ArchSpec old_arch_spec = target->GetArchitecture(); 5385aee162fSJim Ingham 5395aee162fSJim Ingham if (command.GetArgumentCount()) 5405aee162fSJim Ingham { 541fd54b368SJason Molenda result.AppendErrorWithFormat("Invalid arguments for '%s'.\nUsage: %s\n", m_cmd_name.c_str(), m_cmd_syntax.c_str()); 5425aee162fSJim Ingham result.SetStatus (eReturnStatusFailed); 5435aee162fSJim Ingham } 5445aee162fSJim Ingham else 5455aee162fSJim Ingham { 54671337622SGreg Clayton if (state != eStateConnected) 54771337622SGreg Clayton { 548144f3a9cSGreg Clayton const char *plugin_name = m_options.attach_info.GetProcessPluginName(); 549c3776bf2SGreg Clayton process = target->CreateProcess (m_interpreter.GetDebugger().GetListener(), plugin_name, NULL).get(); 55071337622SGreg Clayton } 5515aee162fSJim Ingham 5525aee162fSJim Ingham if (process) 5535aee162fSJim Ingham { 5545aee162fSJim Ingham Error error; 555144f3a9cSGreg Clayton // If no process info was specified, then use the target executable 556144f3a9cSGreg Clayton // name as the process to attach to by default 557144f3a9cSGreg Clayton if (!m_options.attach_info.ProcessInfoSpecified ()) 5585aee162fSJim Ingham { 5593a0b9cdfSJim Ingham if (old_exec_module_sp) 560ad9e828cSGreg Clayton m_options.attach_info.GetExecutableFile().GetFilename() = old_exec_module_sp->GetPlatformFileSpec().GetFilename(); 561144f3a9cSGreg Clayton 562144f3a9cSGreg Clayton if (!m_options.attach_info.ProcessInfoSpecified ()) 5633a0b9cdfSJim Ingham { 564144f3a9cSGreg Clayton error.SetErrorString ("no process specified, create a target with a file, or specify the --pid or --name command option"); 5653a0b9cdfSJim Ingham } 5665aee162fSJim Ingham } 5673a0b9cdfSJim Ingham 568144f3a9cSGreg Clayton if (error.Success()) 5693a0b9cdfSJim Ingham { 570144f3a9cSGreg Clayton error = process->Attach (m_options.attach_info); 5713a0b9cdfSJim Ingham 5725aee162fSJim Ingham if (error.Success()) 5735aee162fSJim Ingham { 5745aee162fSJim Ingham result.SetStatus (eReturnStatusSuccessContinuingNoResult); 5755aee162fSJim Ingham } 5765aee162fSJim Ingham else 5775aee162fSJim Ingham { 578144f3a9cSGreg Clayton result.AppendErrorWithFormat ("attach failed: %s\n", error.AsCString()); 5795aee162fSJim Ingham result.SetStatus (eReturnStatusFailed); 5805aee162fSJim Ingham return false; 5815aee162fSJim Ingham } 582bb3a283bSJim Ingham // If we're synchronous, wait for the stopped event and report that. 583bb3a283bSJim Ingham // Otherwise just return. 584bb3a283bSJim Ingham // FIXME: in the async case it will now be possible to get to the command 585bb3a283bSJim Ingham // interpreter with a state eStateAttaching. Make sure we handle that correctly. 586bb3a283bSJim Ingham StateType state = process->WaitForProcessToStop (NULL); 587bb3a283bSJim Ingham 588bb3a283bSJim Ingham result.SetDidChangeProcessState (true); 589aa739093SJohnny Chen 590aa739093SJohnny Chen if (state == eStateStopped) 591aa739093SJohnny Chen { 59281c22f61SGreg Clayton result.AppendMessageWithFormat ("Process %llu %s\n", process->GetID(), StateAsCString (state)); 593bb3a283bSJim Ingham result.SetStatus (eReturnStatusSuccessFinishNoResult); 594bb3a283bSJim Ingham } 595aa739093SJohnny Chen else 596aa739093SJohnny Chen { 597aa739093SJohnny Chen result.AppendError ("attach failed: process did not stop (no such process or permission problem?)"); 598cfc0935eSJim Ingham process->Destroy(); 599aa739093SJohnny Chen result.SetStatus (eReturnStatusFailed); 600aa739093SJohnny Chen return false; 601aa739093SJohnny Chen } 602aa739093SJohnny Chen } 6035aee162fSJim Ingham } 6045aee162fSJim Ingham } 6055aee162fSJim Ingham 6065aee162fSJim Ingham if (result.Succeeded()) 6075aee162fSJim Ingham { 6085aee162fSJim Ingham // Okay, we're done. Last step is to warn if the executable module has changed: 609513c26ceSGreg Clayton char new_path[PATH_MAX]; 610aa149cbdSGreg Clayton ModuleSP new_exec_module_sp (target->GetExecutableModule()); 6115aee162fSJim Ingham if (!old_exec_module_sp) 6125aee162fSJim Ingham { 613513c26ceSGreg Clayton // We might not have a module if we attached to a raw pid... 614aa149cbdSGreg Clayton if (new_exec_module_sp) 615513c26ceSGreg Clayton { 616aa149cbdSGreg Clayton new_exec_module_sp->GetFileSpec().GetPath(new_path, PATH_MAX); 617513c26ceSGreg Clayton result.AppendMessageWithFormat("Executable module set to \"%s\".\n", new_path); 618513c26ceSGreg Clayton } 6195aee162fSJim Ingham } 620aa149cbdSGreg Clayton else if (old_exec_module_sp->GetFileSpec() != new_exec_module_sp->GetFileSpec()) 6215aee162fSJim Ingham { 622513c26ceSGreg Clayton char old_path[PATH_MAX]; 6235aee162fSJim Ingham 6245aee162fSJim Ingham old_exec_module_sp->GetFileSpec().GetPath (old_path, PATH_MAX); 625aa149cbdSGreg Clayton new_exec_module_sp->GetFileSpec().GetPath (new_path, PATH_MAX); 6265aee162fSJim Ingham 6275aee162fSJim Ingham result.AppendWarningWithFormat("Executable module changed from \"%s\" to \"%s\".\n", 6285aee162fSJim Ingham old_path, new_path); 6295aee162fSJim Ingham } 6305aee162fSJim Ingham 6315aee162fSJim Ingham if (!old_arch_spec.IsValid()) 6325aee162fSJim Ingham { 633c1b1f1eaSGreg Clayton result.AppendMessageWithFormat ("Architecture set to: %s.\n", target->GetArchitecture().GetTriple().getTriple().c_str()); 6345aee162fSJim Ingham } 6355aee162fSJim Ingham else if (old_arch_spec != target->GetArchitecture()) 6365aee162fSJim Ingham { 6375aee162fSJim Ingham result.AppendWarningWithFormat("Architecture changed from %s to %s.\n", 638c1b1f1eaSGreg Clayton old_arch_spec.GetTriple().getTriple().c_str(), 639c1b1f1eaSGreg Clayton target->GetArchitecture().GetTriple().getTriple().c_str()); 6405aee162fSJim Ingham } 641a95ce623SJohnny Chen 642a95ce623SJohnny Chen // This supports the use-case scenario of immediately continuing the process once attached. 643a95ce623SJohnny Chen if (m_options.attach_info.GetContinueOnceAttached()) 6445bcaf583SSean Callanan m_interpreter.HandleCommand("process continue", eLazyBoolNo, result); 6455aee162fSJim Ingham } 6465aee162fSJim Ingham return result.Succeeded(); 6475aee162fSJim Ingham } 6485aee162fSJim Ingham 64930fdc8d8SChris Lattner CommandOptions m_options; 65030fdc8d8SChris Lattner }; 65130fdc8d8SChris Lattner 65230fdc8d8SChris Lattner 653e0d378b3SGreg Clayton OptionDefinition 65430fdc8d8SChris Lattner CommandObjectProcessAttach::CommandOptions::g_option_table[] = 65530fdc8d8SChris Lattner { 656a95ce623SJohnny Chen { LLDB_OPT_SET_ALL, false, "continue",'c', no_argument, NULL, 0, eArgTypeNone, "Immediately continue the process once attached."}, 657deaab222SCaroline Tice { LLDB_OPT_SET_ALL, false, "plugin", 'P', required_argument, NULL, 0, eArgTypePlugin, "Name of the process plugin you want to use."}, 658deaab222SCaroline Tice { LLDB_OPT_SET_1, false, "pid", 'p', required_argument, NULL, 0, eArgTypePid, "The process ID of an existing process to attach to."}, 659deaab222SCaroline Tice { LLDB_OPT_SET_2, false, "name", 'n', required_argument, NULL, 0, eArgTypeProcessName, "The name of the process to attach to."}, 660cd16df91SJim Ingham { LLDB_OPT_SET_2, false, "include-existing", 'i', no_argument, NULL, 0, eArgTypeNone, "Include existing processes when doing attach -w."}, 6614ce37abbSJohnny Chen { LLDB_OPT_SET_2, false, "waitfor", 'w', no_argument, NULL, 0, eArgTypeNone, "Wait for the process with <process-name> to launch."}, 662deaab222SCaroline Tice { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL } 66330fdc8d8SChris Lattner }; 66430fdc8d8SChris Lattner 66530fdc8d8SChris Lattner //------------------------------------------------------------------------- 66630fdc8d8SChris Lattner // CommandObjectProcessContinue 66730fdc8d8SChris Lattner //------------------------------------------------------------------------- 668bb9caf73SJim Ingham #pragma mark CommandObjectProcessContinue 66930fdc8d8SChris Lattner 6705a988416SJim Ingham class CommandObjectProcessContinue : public CommandObjectParsed 67130fdc8d8SChris Lattner { 67230fdc8d8SChris Lattner public: 67330fdc8d8SChris Lattner 674a7015092SGreg Clayton CommandObjectProcessContinue (CommandInterpreter &interpreter) : 6755a988416SJim Ingham CommandObjectParsed (interpreter, 676a7015092SGreg Clayton "process continue", 677e3d26315SCaroline Tice "Continue execution of all threads in the current process.", 67830fdc8d8SChris Lattner "process continue", 6790e41084aSJim Ingham eFlagProcessMustBeLaunched | eFlagProcessMustBePaused), 6800e41084aSJim Ingham m_options(interpreter) 68130fdc8d8SChris Lattner { 68230fdc8d8SChris Lattner } 68330fdc8d8SChris Lattner 68430fdc8d8SChris Lattner 68530fdc8d8SChris Lattner ~CommandObjectProcessContinue () 68630fdc8d8SChris Lattner { 68730fdc8d8SChris Lattner } 68830fdc8d8SChris Lattner 6895a988416SJim Ingham protected: 6900e41084aSJim Ingham 6910e41084aSJim Ingham class CommandOptions : public Options 6920e41084aSJim Ingham { 6930e41084aSJim Ingham public: 6940e41084aSJim Ingham 6950e41084aSJim Ingham CommandOptions (CommandInterpreter &interpreter) : 6960e41084aSJim Ingham Options(interpreter) 6970e41084aSJim Ingham { 6980e41084aSJim Ingham // Keep default values of all options in one place: OptionParsingStarting () 6990e41084aSJim Ingham OptionParsingStarting (); 7000e41084aSJim Ingham } 7010e41084aSJim Ingham 7020e41084aSJim Ingham ~CommandOptions () 7030e41084aSJim Ingham { 7040e41084aSJim Ingham } 7050e41084aSJim Ingham 7060e41084aSJim Ingham Error 7070e41084aSJim Ingham SetOptionValue (uint32_t option_idx, const char *option_arg) 7080e41084aSJim Ingham { 7090e41084aSJim Ingham Error error; 7100e41084aSJim Ingham char short_option = (char) m_getopt_table[option_idx].val; 7110e41084aSJim Ingham bool success = false; 7120e41084aSJim Ingham switch (short_option) 7130e41084aSJim Ingham { 7140e41084aSJim Ingham case 'i': 7150e41084aSJim Ingham m_ignore = Args::StringToUInt32 (option_arg, 0, 0, &success); 7160e41084aSJim Ingham if (!success) 7170e41084aSJim Ingham error.SetErrorStringWithFormat ("invalid value for ignore option: \"%s\", should be a number.", option_arg); 7180e41084aSJim Ingham break; 7190e41084aSJim Ingham 7200e41084aSJim Ingham default: 7210e41084aSJim Ingham error.SetErrorStringWithFormat("invalid short option character '%c'", short_option); 7220e41084aSJim Ingham break; 7230e41084aSJim Ingham } 7240e41084aSJim Ingham return error; 7250e41084aSJim Ingham } 7260e41084aSJim Ingham 7270e41084aSJim Ingham void 7280e41084aSJim Ingham OptionParsingStarting () 7290e41084aSJim Ingham { 7300e41084aSJim Ingham m_ignore = 0; 7310e41084aSJim Ingham } 7320e41084aSJim Ingham 7330e41084aSJim Ingham const OptionDefinition* 7340e41084aSJim Ingham GetDefinitions () 7350e41084aSJim Ingham { 7360e41084aSJim Ingham return g_option_table; 7370e41084aSJim Ingham } 7380e41084aSJim Ingham 7390e41084aSJim Ingham // Options table: Required for subclasses of Options. 7400e41084aSJim Ingham 7410e41084aSJim Ingham static OptionDefinition g_option_table[]; 7420e41084aSJim Ingham 7430e41084aSJim Ingham uint32_t m_ignore; 7440e41084aSJim Ingham }; 7450e41084aSJim Ingham 74630fdc8d8SChris Lattner bool 7475a988416SJim Ingham DoExecute (Args& command, 74830fdc8d8SChris Lattner CommandReturnObject &result) 74930fdc8d8SChris Lattner { 750c14ee32dSGreg Clayton Process *process = m_interpreter.GetExecutionContext().GetProcessPtr(); 751a7015092SGreg Clayton bool synchronous_execution = m_interpreter.GetSynchronous (); 75230fdc8d8SChris Lattner 75330fdc8d8SChris Lattner if (process == NULL) 75430fdc8d8SChris Lattner { 75530fdc8d8SChris Lattner result.AppendError ("no process to continue"); 75630fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 75730fdc8d8SChris Lattner return false; 75830fdc8d8SChris Lattner } 75930fdc8d8SChris Lattner 76030fdc8d8SChris Lattner StateType state = process->GetState(); 76130fdc8d8SChris Lattner if (state == eStateStopped) 76230fdc8d8SChris Lattner { 76330fdc8d8SChris Lattner if (command.GetArgumentCount() != 0) 76430fdc8d8SChris Lattner { 76530fdc8d8SChris Lattner result.AppendErrorWithFormat ("The '%s' command does not take any arguments.\n", m_cmd_name.c_str()); 76630fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 76730fdc8d8SChris Lattner return false; 76830fdc8d8SChris Lattner } 76930fdc8d8SChris Lattner 7700e41084aSJim Ingham if (m_options.m_ignore > 0) 7710e41084aSJim Ingham { 7720e41084aSJim Ingham ThreadSP sel_thread_sp(process->GetThreadList().GetSelectedThread()); 7730e41084aSJim Ingham if (sel_thread_sp) 7740e41084aSJim Ingham { 7750e41084aSJim Ingham StopInfoSP stop_info_sp = sel_thread_sp->GetStopInfo(); 7760e41084aSJim Ingham if (stop_info_sp && stop_info_sp->GetStopReason() == eStopReasonBreakpoint) 7770e41084aSJim Ingham { 7780e41084aSJim Ingham uint64_t bp_site_id = stop_info_sp->GetValue(); 7790e41084aSJim Ingham BreakpointSiteSP bp_site_sp(process->GetBreakpointSiteList().FindByID(bp_site_id)); 7800e41084aSJim Ingham if (bp_site_sp) 7810e41084aSJim Ingham { 7820e41084aSJim Ingham uint32_t num_owners = bp_site_sp->GetNumberOfOwners(); 7830e41084aSJim Ingham for (uint32_t i = 0; i < num_owners; i++) 7840e41084aSJim Ingham { 7850e41084aSJim Ingham Breakpoint &bp_ref = bp_site_sp->GetOwnerAtIndex(i)->GetBreakpoint(); 7860e41084aSJim Ingham if (!bp_ref.IsInternal()) 7870e41084aSJim Ingham { 7880e41084aSJim Ingham bp_ref.SetIgnoreCount(m_options.m_ignore); 7890e41084aSJim Ingham } 7900e41084aSJim Ingham } 7910e41084aSJim Ingham } 7920e41084aSJim Ingham } 7930e41084aSJim Ingham } 7940e41084aSJim Ingham } 7950e41084aSJim Ingham 79641f2b940SJim Ingham { // Scope for thread list mutex: 79741f2b940SJim Ingham Mutex::Locker locker (process->GetThreadList().GetMutex()); 79830fdc8d8SChris Lattner const uint32_t num_threads = process->GetThreadList().GetSize(); 79930fdc8d8SChris Lattner 80030fdc8d8SChris Lattner // Set the actions that the threads should each take when resuming 80130fdc8d8SChris Lattner for (uint32_t idx=0; idx<num_threads; ++idx) 80230fdc8d8SChris Lattner { 80330fdc8d8SChris Lattner process->GetThreadList().GetThreadAtIndex(idx)->SetResumeState (eStateRunning); 80430fdc8d8SChris Lattner } 80541f2b940SJim Ingham } 80630fdc8d8SChris Lattner 80730fdc8d8SChris Lattner Error error(process->Resume()); 80830fdc8d8SChris Lattner if (error.Success()) 80930fdc8d8SChris Lattner { 81081c22f61SGreg Clayton result.AppendMessageWithFormat ("Process %llu resuming\n", process->GetID()); 81130fdc8d8SChris Lattner if (synchronous_execution) 81230fdc8d8SChris Lattner { 813b132097bSGreg Clayton state = process->WaitForProcessToStop (NULL); 81430fdc8d8SChris Lattner 81530fdc8d8SChris Lattner result.SetDidChangeProcessState (true); 81681c22f61SGreg Clayton result.AppendMessageWithFormat ("Process %llu %s\n", process->GetID(), StateAsCString (state)); 81730fdc8d8SChris Lattner result.SetStatus (eReturnStatusSuccessFinishNoResult); 81830fdc8d8SChris Lattner } 81930fdc8d8SChris Lattner else 82030fdc8d8SChris Lattner { 82130fdc8d8SChris Lattner result.SetStatus (eReturnStatusSuccessContinuingNoResult); 82230fdc8d8SChris Lattner } 82330fdc8d8SChris Lattner } 82430fdc8d8SChris Lattner else 82530fdc8d8SChris Lattner { 82630fdc8d8SChris Lattner result.AppendErrorWithFormat("Failed to resume process: %s.\n", error.AsCString()); 82730fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 82830fdc8d8SChris Lattner } 82930fdc8d8SChris Lattner } 83030fdc8d8SChris Lattner else 83130fdc8d8SChris Lattner { 83230fdc8d8SChris Lattner result.AppendErrorWithFormat ("Process cannot be continued from its current state (%s).\n", 83330fdc8d8SChris Lattner StateAsCString(state)); 83430fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 83530fdc8d8SChris Lattner } 83630fdc8d8SChris Lattner return result.Succeeded(); 83730fdc8d8SChris Lattner } 8380e41084aSJim Ingham 8390e41084aSJim Ingham Options * 8400e41084aSJim Ingham GetOptions () 8410e41084aSJim Ingham { 8420e41084aSJim Ingham return &m_options; 8430e41084aSJim Ingham } 8440e41084aSJim Ingham 8450e41084aSJim Ingham CommandOptions m_options; 8460e41084aSJim Ingham 8470e41084aSJim Ingham }; 8480e41084aSJim Ingham 8490e41084aSJim Ingham OptionDefinition 8500e41084aSJim Ingham CommandObjectProcessContinue::CommandOptions::g_option_table[] = 8510e41084aSJim Ingham { 8520e41084aSJim Ingham { LLDB_OPT_SET_ALL, false, "ignore-count",'i', required_argument, NULL, 0, eArgTypeUnsignedInteger, 8530e41084aSJim Ingham "Ignore <N> crossings of the breakpoint (if it exists) for the currently selected thread."}, 8540e41084aSJim Ingham { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL } 85530fdc8d8SChris Lattner }; 85630fdc8d8SChris Lattner 85730fdc8d8SChris Lattner //------------------------------------------------------------------------- 85830fdc8d8SChris Lattner // CommandObjectProcessDetach 85930fdc8d8SChris Lattner //------------------------------------------------------------------------- 860bb9caf73SJim Ingham #pragma mark CommandObjectProcessDetach 86130fdc8d8SChris Lattner 8625a988416SJim Ingham class CommandObjectProcessDetach : public CommandObjectParsed 86330fdc8d8SChris Lattner { 86430fdc8d8SChris Lattner public: 86530fdc8d8SChris Lattner 866a7015092SGreg Clayton CommandObjectProcessDetach (CommandInterpreter &interpreter) : 8675a988416SJim Ingham CommandObjectParsed (interpreter, 868a7015092SGreg Clayton "process detach", 869e3d26315SCaroline Tice "Detach from the current process being debugged.", 87030fdc8d8SChris Lattner "process detach", 87130fdc8d8SChris Lattner eFlagProcessMustBeLaunched) 87230fdc8d8SChris Lattner { 87330fdc8d8SChris Lattner } 87430fdc8d8SChris Lattner 87530fdc8d8SChris Lattner ~CommandObjectProcessDetach () 87630fdc8d8SChris Lattner { 87730fdc8d8SChris Lattner } 87830fdc8d8SChris Lattner 8795a988416SJim Ingham protected: 88030fdc8d8SChris Lattner bool 8815a988416SJim Ingham DoExecute (Args& command, 88230fdc8d8SChris Lattner CommandReturnObject &result) 88330fdc8d8SChris Lattner { 884c14ee32dSGreg Clayton Process *process = m_interpreter.GetExecutionContext().GetProcessPtr(); 88530fdc8d8SChris Lattner if (process == NULL) 88630fdc8d8SChris Lattner { 88730fdc8d8SChris Lattner result.AppendError ("must have a valid process in order to detach"); 88830fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 88930fdc8d8SChris Lattner return false; 89030fdc8d8SChris Lattner } 89130fdc8d8SChris Lattner 89281c22f61SGreg Clayton result.AppendMessageWithFormat ("Detaching from process %llu\n", process->GetID()); 89330fdc8d8SChris Lattner Error error (process->Detach()); 89430fdc8d8SChris Lattner if (error.Success()) 89530fdc8d8SChris Lattner { 89630fdc8d8SChris Lattner result.SetStatus (eReturnStatusSuccessFinishResult); 89730fdc8d8SChris Lattner } 89830fdc8d8SChris Lattner else 89930fdc8d8SChris Lattner { 90030fdc8d8SChris Lattner result.AppendErrorWithFormat ("Detach failed: %s\n", error.AsCString()); 90130fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 90230fdc8d8SChris Lattner return false; 90330fdc8d8SChris Lattner } 90430fdc8d8SChris Lattner return result.Succeeded(); 90530fdc8d8SChris Lattner } 90630fdc8d8SChris Lattner }; 90730fdc8d8SChris Lattner 90830fdc8d8SChris Lattner //------------------------------------------------------------------------- 909b766a73dSGreg Clayton // CommandObjectProcessConnect 910b766a73dSGreg Clayton //------------------------------------------------------------------------- 911b766a73dSGreg Clayton #pragma mark CommandObjectProcessConnect 912b766a73dSGreg Clayton 9135a988416SJim Ingham class CommandObjectProcessConnect : public CommandObjectParsed 914b766a73dSGreg Clayton { 915b766a73dSGreg Clayton public: 916b766a73dSGreg Clayton 917b766a73dSGreg Clayton class CommandOptions : public Options 918b766a73dSGreg Clayton { 919b766a73dSGreg Clayton public: 920b766a73dSGreg Clayton 921eb0103f2SGreg Clayton CommandOptions (CommandInterpreter &interpreter) : 922eb0103f2SGreg Clayton Options(interpreter) 923b766a73dSGreg Clayton { 924f6b8b581SGreg Clayton // Keep default values of all options in one place: OptionParsingStarting () 925f6b8b581SGreg Clayton OptionParsingStarting (); 926b766a73dSGreg Clayton } 927b766a73dSGreg Clayton 928b766a73dSGreg Clayton ~CommandOptions () 929b766a73dSGreg Clayton { 930b766a73dSGreg Clayton } 931b766a73dSGreg Clayton 932b766a73dSGreg Clayton Error 933f6b8b581SGreg Clayton SetOptionValue (uint32_t option_idx, const char *option_arg) 934b766a73dSGreg Clayton { 935b766a73dSGreg Clayton Error error; 936b766a73dSGreg Clayton char short_option = (char) m_getopt_table[option_idx].val; 937b766a73dSGreg Clayton 938b766a73dSGreg Clayton switch (short_option) 939b766a73dSGreg Clayton { 940b766a73dSGreg Clayton case 'p': 941b766a73dSGreg Clayton plugin_name.assign (option_arg); 942b766a73dSGreg Clayton break; 943b766a73dSGreg Clayton 944b766a73dSGreg Clayton default: 94586edbf41SGreg Clayton error.SetErrorStringWithFormat("invalid short option character '%c'", short_option); 946b766a73dSGreg Clayton break; 947b766a73dSGreg Clayton } 948b766a73dSGreg Clayton return error; 949b766a73dSGreg Clayton } 950b766a73dSGreg Clayton 951b766a73dSGreg Clayton void 952f6b8b581SGreg Clayton OptionParsingStarting () 953b766a73dSGreg Clayton { 954b766a73dSGreg Clayton plugin_name.clear(); 955b766a73dSGreg Clayton } 956b766a73dSGreg Clayton 957e0d378b3SGreg Clayton const OptionDefinition* 958b766a73dSGreg Clayton GetDefinitions () 959b766a73dSGreg Clayton { 960b766a73dSGreg Clayton return g_option_table; 961b766a73dSGreg Clayton } 962b766a73dSGreg Clayton 963b766a73dSGreg Clayton // Options table: Required for subclasses of Options. 964b766a73dSGreg Clayton 965e0d378b3SGreg Clayton static OptionDefinition g_option_table[]; 966b766a73dSGreg Clayton 967b766a73dSGreg Clayton // Instance variables to hold the values for command options. 968b766a73dSGreg Clayton 969b766a73dSGreg Clayton std::string plugin_name; 970b766a73dSGreg Clayton }; 971b766a73dSGreg Clayton 972b766a73dSGreg Clayton CommandObjectProcessConnect (CommandInterpreter &interpreter) : 9735a988416SJim Ingham CommandObjectParsed (interpreter, 974b766a73dSGreg Clayton "process connect", 975b766a73dSGreg Clayton "Connect to a remote debug service.", 976b766a73dSGreg Clayton "process connect <remote-url>", 977eb0103f2SGreg Clayton 0), 978eb0103f2SGreg Clayton m_options (interpreter) 979b766a73dSGreg Clayton { 980b766a73dSGreg Clayton } 981b766a73dSGreg Clayton 982b766a73dSGreg Clayton ~CommandObjectProcessConnect () 983b766a73dSGreg Clayton { 984b766a73dSGreg Clayton } 985b766a73dSGreg Clayton 986b766a73dSGreg Clayton 9875a988416SJim Ingham Options * 9885a988416SJim Ingham GetOptions () 9895a988416SJim Ingham { 9905a988416SJim Ingham return &m_options; 9915a988416SJim Ingham } 9925a988416SJim Ingham 9935a988416SJim Ingham protected: 994b766a73dSGreg Clayton bool 9955a988416SJim Ingham DoExecute (Args& command, 996b766a73dSGreg Clayton CommandReturnObject &result) 997b766a73dSGreg Clayton { 998b766a73dSGreg Clayton 999b766a73dSGreg Clayton TargetSP target_sp (m_interpreter.GetDebugger().GetSelectedTarget()); 1000b766a73dSGreg Clayton Error error; 1001c14ee32dSGreg Clayton Process *process = m_interpreter.GetExecutionContext().GetProcessPtr(); 1002b766a73dSGreg Clayton if (process) 1003b766a73dSGreg Clayton { 1004b766a73dSGreg Clayton if (process->IsAlive()) 1005b766a73dSGreg Clayton { 100681c22f61SGreg Clayton result.AppendErrorWithFormat ("Process %llu is currently being debugged, kill the process before connecting.\n", 1007b766a73dSGreg Clayton process->GetID()); 1008b766a73dSGreg Clayton result.SetStatus (eReturnStatusFailed); 1009b766a73dSGreg Clayton return false; 1010b766a73dSGreg Clayton } 1011b766a73dSGreg Clayton } 1012b766a73dSGreg Clayton 1013b766a73dSGreg Clayton if (!target_sp) 1014b766a73dSGreg Clayton { 1015b766a73dSGreg Clayton // If there isn't a current target create one. 1016b766a73dSGreg Clayton FileSpec emptyFileSpec; 1017b766a73dSGreg Clayton 1018b766a73dSGreg Clayton error = m_interpreter.GetDebugger().GetTargetList().CreateTarget (m_interpreter.GetDebugger(), 1019b766a73dSGreg Clayton emptyFileSpec, 1020cac9c5f9SGreg Clayton NULL, 1021b766a73dSGreg Clayton false, 1022cac9c5f9SGreg Clayton NULL, // No platform options 1023b766a73dSGreg Clayton target_sp); 1024b766a73dSGreg Clayton if (!target_sp || error.Fail()) 1025b766a73dSGreg Clayton { 1026b766a73dSGreg Clayton result.AppendError(error.AsCString("Error creating target")); 1027b766a73dSGreg Clayton result.SetStatus (eReturnStatusFailed); 1028b766a73dSGreg Clayton return false; 1029b766a73dSGreg Clayton } 1030b766a73dSGreg Clayton m_interpreter.GetDebugger().GetTargetList().SetSelectedTarget(target_sp.get()); 1031b766a73dSGreg Clayton } 1032b766a73dSGreg Clayton 1033b766a73dSGreg Clayton if (command.GetArgumentCount() == 1) 1034b766a73dSGreg Clayton { 1035b766a73dSGreg Clayton const char *plugin_name = NULL; 1036b766a73dSGreg Clayton if (!m_options.plugin_name.empty()) 1037b766a73dSGreg Clayton plugin_name = m_options.plugin_name.c_str(); 1038b766a73dSGreg Clayton 1039b766a73dSGreg Clayton const char *remote_url = command.GetArgumentAtIndex(0); 1040c3776bf2SGreg Clayton process = target_sp->CreateProcess (m_interpreter.GetDebugger().GetListener(), plugin_name, NULL).get(); 1041b766a73dSGreg Clayton 1042b766a73dSGreg Clayton if (process) 1043b766a73dSGreg Clayton { 1044*4bd4e7e3SJason Molenda error = process->ConnectRemote (&process->GetTarget().GetDebugger().GetOutputStream(), remote_url); 1045b766a73dSGreg Clayton 1046b766a73dSGreg Clayton if (error.Fail()) 1047b766a73dSGreg Clayton { 1048b766a73dSGreg Clayton result.AppendError(error.AsCString("Remote connect failed")); 1049b766a73dSGreg Clayton result.SetStatus (eReturnStatusFailed); 10501517dd33SGreg Clayton target_sp->DeleteCurrentProcess(); 1051b766a73dSGreg Clayton return false; 1052b766a73dSGreg Clayton } 1053b766a73dSGreg Clayton } 1054b766a73dSGreg Clayton else 1055b766a73dSGreg Clayton { 1056fd54b368SJason 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", 1057fd54b368SJason Molenda m_cmd_name.c_str()); 1058b766a73dSGreg Clayton result.SetStatus (eReturnStatusFailed); 1059b766a73dSGreg Clayton } 1060b766a73dSGreg Clayton } 1061b766a73dSGreg Clayton else 1062b766a73dSGreg Clayton { 1063fd54b368SJason Molenda result.AppendErrorWithFormat ("'%s' takes exactly one argument:\nUsage: %s\n", 1064b766a73dSGreg Clayton m_cmd_name.c_str(), 1065b766a73dSGreg Clayton m_cmd_syntax.c_str()); 1066b766a73dSGreg Clayton result.SetStatus (eReturnStatusFailed); 1067b766a73dSGreg Clayton } 1068b766a73dSGreg Clayton return result.Succeeded(); 1069b766a73dSGreg Clayton } 1070b766a73dSGreg Clayton 1071b766a73dSGreg Clayton CommandOptions m_options; 1072b766a73dSGreg Clayton }; 1073b766a73dSGreg Clayton 1074b766a73dSGreg Clayton 1075e0d378b3SGreg Clayton OptionDefinition 1076b766a73dSGreg Clayton CommandObjectProcessConnect::CommandOptions::g_option_table[] = 1077b766a73dSGreg Clayton { 1078b766a73dSGreg Clayton { LLDB_OPT_SET_ALL, false, "plugin", 'p', required_argument, NULL, 0, eArgTypePlugin, "Name of the process plugin you want to use."}, 1079b766a73dSGreg Clayton { 0, false, NULL, 0 , 0, NULL, 0, eArgTypeNone, NULL } 1080b766a73dSGreg Clayton }; 1081b766a73dSGreg Clayton 1082b766a73dSGreg Clayton //------------------------------------------------------------------------- 10838f343b09SGreg Clayton // CommandObjectProcessLoad 10848f343b09SGreg Clayton //------------------------------------------------------------------------- 1085bb9caf73SJim Ingham #pragma mark CommandObjectProcessLoad 10868f343b09SGreg Clayton 10875a988416SJim Ingham class CommandObjectProcessLoad : public CommandObjectParsed 10888f343b09SGreg Clayton { 10898f343b09SGreg Clayton public: 10908f343b09SGreg Clayton 10918f343b09SGreg Clayton CommandObjectProcessLoad (CommandInterpreter &interpreter) : 10925a988416SJim Ingham CommandObjectParsed (interpreter, 10938f343b09SGreg Clayton "process load", 10948f343b09SGreg Clayton "Load a shared library into the current process.", 10958f343b09SGreg Clayton "process load <filename> [<filename> ...]", 10968f343b09SGreg Clayton eFlagProcessMustBeLaunched | eFlagProcessMustBePaused) 10978f343b09SGreg Clayton { 10988f343b09SGreg Clayton } 10998f343b09SGreg Clayton 11008f343b09SGreg Clayton ~CommandObjectProcessLoad () 11018f343b09SGreg Clayton { 11028f343b09SGreg Clayton } 11038f343b09SGreg Clayton 11045a988416SJim Ingham protected: 11058f343b09SGreg Clayton bool 11065a988416SJim Ingham DoExecute (Args& command, 11078f343b09SGreg Clayton CommandReturnObject &result) 11088f343b09SGreg Clayton { 1109c14ee32dSGreg Clayton Process *process = m_interpreter.GetExecutionContext().GetProcessPtr(); 11108f343b09SGreg Clayton if (process == NULL) 11118f343b09SGreg Clayton { 11128f343b09SGreg Clayton result.AppendError ("must have a valid process in order to load a shared library"); 11138f343b09SGreg Clayton result.SetStatus (eReturnStatusFailed); 11148f343b09SGreg Clayton return false; 11158f343b09SGreg Clayton } 11168f343b09SGreg Clayton 11178f343b09SGreg Clayton const uint32_t argc = command.GetArgumentCount(); 11188f343b09SGreg Clayton 11198f343b09SGreg Clayton for (uint32_t i=0; i<argc; ++i) 11208f343b09SGreg Clayton { 11218f343b09SGreg Clayton Error error; 11228f343b09SGreg Clayton const char *image_path = command.GetArgumentAtIndex(i); 11238f343b09SGreg Clayton FileSpec image_spec (image_path, false); 1124aa516843SGreg Clayton process->GetTarget().GetPlatform()->ResolveRemotePath(image_spec, image_spec); 11258f343b09SGreg Clayton uint32_t image_token = process->LoadImage(image_spec, error); 11268f343b09SGreg Clayton if (image_token != LLDB_INVALID_IMAGE_TOKEN) 11278f343b09SGreg Clayton { 11288f343b09SGreg Clayton result.AppendMessageWithFormat ("Loading \"%s\"...ok\nImage %u loaded.\n", image_path, image_token); 11298f343b09SGreg Clayton result.SetStatus (eReturnStatusSuccessFinishResult); 11308f343b09SGreg Clayton } 11318f343b09SGreg Clayton else 11328f343b09SGreg Clayton { 11338f343b09SGreg Clayton result.AppendErrorWithFormat ("failed to load '%s': %s", image_path, error.AsCString()); 11348f343b09SGreg Clayton result.SetStatus (eReturnStatusFailed); 11358f343b09SGreg Clayton } 11368f343b09SGreg Clayton } 11378f343b09SGreg Clayton return result.Succeeded(); 11388f343b09SGreg Clayton } 11398f343b09SGreg Clayton }; 11408f343b09SGreg Clayton 11418f343b09SGreg Clayton 11428f343b09SGreg Clayton //------------------------------------------------------------------------- 11438f343b09SGreg Clayton // CommandObjectProcessUnload 11448f343b09SGreg Clayton //------------------------------------------------------------------------- 1145bb9caf73SJim Ingham #pragma mark CommandObjectProcessUnload 11468f343b09SGreg Clayton 11475a988416SJim Ingham class CommandObjectProcessUnload : public CommandObjectParsed 11488f343b09SGreg Clayton { 11498f343b09SGreg Clayton public: 11508f343b09SGreg Clayton 11518f343b09SGreg Clayton CommandObjectProcessUnload (CommandInterpreter &interpreter) : 11525a988416SJim Ingham CommandObjectParsed (interpreter, 11538f343b09SGreg Clayton "process unload", 11548f343b09SGreg Clayton "Unload a shared library from the current process using the index returned by a previous call to \"process load\".", 11558f343b09SGreg Clayton "process unload <index>", 11568f343b09SGreg Clayton eFlagProcessMustBeLaunched | eFlagProcessMustBePaused) 11578f343b09SGreg Clayton { 11588f343b09SGreg Clayton } 11598f343b09SGreg Clayton 11608f343b09SGreg Clayton ~CommandObjectProcessUnload () 11618f343b09SGreg Clayton { 11628f343b09SGreg Clayton } 11638f343b09SGreg Clayton 11645a988416SJim Ingham protected: 11658f343b09SGreg Clayton bool 11665a988416SJim Ingham DoExecute (Args& command, 11678f343b09SGreg Clayton CommandReturnObject &result) 11688f343b09SGreg Clayton { 1169c14ee32dSGreg Clayton Process *process = m_interpreter.GetExecutionContext().GetProcessPtr(); 11708f343b09SGreg Clayton if (process == NULL) 11718f343b09SGreg Clayton { 11728f343b09SGreg Clayton result.AppendError ("must have a valid process in order to load a shared library"); 11738f343b09SGreg Clayton result.SetStatus (eReturnStatusFailed); 11748f343b09SGreg Clayton return false; 11758f343b09SGreg Clayton } 11768f343b09SGreg Clayton 11778f343b09SGreg Clayton const uint32_t argc = command.GetArgumentCount(); 11788f343b09SGreg Clayton 11798f343b09SGreg Clayton for (uint32_t i=0; i<argc; ++i) 11808f343b09SGreg Clayton { 11818f343b09SGreg Clayton const char *image_token_cstr = command.GetArgumentAtIndex(i); 11828f343b09SGreg Clayton uint32_t image_token = Args::StringToUInt32(image_token_cstr, LLDB_INVALID_IMAGE_TOKEN, 0); 11838f343b09SGreg Clayton if (image_token == LLDB_INVALID_IMAGE_TOKEN) 11848f343b09SGreg Clayton { 11858f343b09SGreg Clayton result.AppendErrorWithFormat ("invalid image index argument '%s'", image_token_cstr); 11868f343b09SGreg Clayton result.SetStatus (eReturnStatusFailed); 11878f343b09SGreg Clayton break; 11888f343b09SGreg Clayton } 11898f343b09SGreg Clayton else 11908f343b09SGreg Clayton { 11918f343b09SGreg Clayton Error error (process->UnloadImage(image_token)); 11928f343b09SGreg Clayton if (error.Success()) 11938f343b09SGreg Clayton { 11948f343b09SGreg Clayton result.AppendMessageWithFormat ("Unloading shared library with index %u...ok\n", image_token); 11958f343b09SGreg Clayton result.SetStatus (eReturnStatusSuccessFinishResult); 11968f343b09SGreg Clayton } 11978f343b09SGreg Clayton else 11988f343b09SGreg Clayton { 11998f343b09SGreg Clayton result.AppendErrorWithFormat ("failed to unload image: %s", error.AsCString()); 12008f343b09SGreg Clayton result.SetStatus (eReturnStatusFailed); 12018f343b09SGreg Clayton break; 12028f343b09SGreg Clayton } 12038f343b09SGreg Clayton } 12048f343b09SGreg Clayton } 12058f343b09SGreg Clayton return result.Succeeded(); 12068f343b09SGreg Clayton } 12078f343b09SGreg Clayton }; 12088f343b09SGreg Clayton 12098f343b09SGreg Clayton //------------------------------------------------------------------------- 121030fdc8d8SChris Lattner // CommandObjectProcessSignal 121130fdc8d8SChris Lattner //------------------------------------------------------------------------- 1212bb9caf73SJim Ingham #pragma mark CommandObjectProcessSignal 121330fdc8d8SChris Lattner 12145a988416SJim Ingham class CommandObjectProcessSignal : public CommandObjectParsed 121530fdc8d8SChris Lattner { 121630fdc8d8SChris Lattner public: 121730fdc8d8SChris Lattner 1218a7015092SGreg Clayton CommandObjectProcessSignal (CommandInterpreter &interpreter) : 12195a988416SJim Ingham CommandObjectParsed (interpreter, 1220a7015092SGreg Clayton "process signal", 1221e3d26315SCaroline Tice "Send a UNIX signal to the current process being debugged.", 1222405fe67fSCaroline Tice NULL) 122330fdc8d8SChris Lattner { 1224405fe67fSCaroline Tice CommandArgumentEntry arg; 1225405fe67fSCaroline Tice CommandArgumentData signal_arg; 1226405fe67fSCaroline Tice 1227405fe67fSCaroline Tice // Define the first (and only) variant of this arg. 1228c0dbdfb6SCaroline Tice signal_arg.arg_type = eArgTypeUnixSignal; 1229405fe67fSCaroline Tice signal_arg.arg_repetition = eArgRepeatPlain; 1230405fe67fSCaroline Tice 1231405fe67fSCaroline Tice // There is only one variant this argument could be; put it into the argument entry. 1232405fe67fSCaroline Tice arg.push_back (signal_arg); 1233405fe67fSCaroline Tice 1234405fe67fSCaroline Tice // Push the data for the first argument into the m_arguments vector. 1235405fe67fSCaroline Tice m_arguments.push_back (arg); 123630fdc8d8SChris Lattner } 123730fdc8d8SChris Lattner 123830fdc8d8SChris Lattner ~CommandObjectProcessSignal () 123930fdc8d8SChris Lattner { 124030fdc8d8SChris Lattner } 124130fdc8d8SChris Lattner 12425a988416SJim Ingham protected: 124330fdc8d8SChris Lattner bool 12445a988416SJim Ingham DoExecute (Args& command, 124530fdc8d8SChris Lattner CommandReturnObject &result) 124630fdc8d8SChris Lattner { 1247c14ee32dSGreg Clayton Process *process = m_interpreter.GetExecutionContext().GetProcessPtr(); 124830fdc8d8SChris Lattner if (process == NULL) 124930fdc8d8SChris Lattner { 125030fdc8d8SChris Lattner result.AppendError ("no process to signal"); 125130fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 125230fdc8d8SChris Lattner return false; 125330fdc8d8SChris Lattner } 125430fdc8d8SChris Lattner 125530fdc8d8SChris Lattner if (command.GetArgumentCount() == 1) 125630fdc8d8SChris Lattner { 1257237cd906SGreg Clayton int signo = LLDB_INVALID_SIGNAL_NUMBER; 1258237cd906SGreg Clayton 1259237cd906SGreg Clayton const char *signal_name = command.GetArgumentAtIndex(0); 1260237cd906SGreg Clayton if (::isxdigit (signal_name[0])) 1261237cd906SGreg Clayton signo = Args::StringToSInt32(signal_name, LLDB_INVALID_SIGNAL_NUMBER, 0); 1262237cd906SGreg Clayton else 1263237cd906SGreg Clayton signo = process->GetUnixSignals().GetSignalNumberFromName (signal_name); 1264237cd906SGreg Clayton 1265237cd906SGreg Clayton if (signo == LLDB_INVALID_SIGNAL_NUMBER) 126630fdc8d8SChris Lattner { 126730fdc8d8SChris Lattner result.AppendErrorWithFormat ("Invalid signal argument '%s'.\n", command.GetArgumentAtIndex(0)); 126830fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 126930fdc8d8SChris Lattner } 127030fdc8d8SChris Lattner else 127130fdc8d8SChris Lattner { 127230fdc8d8SChris Lattner Error error (process->Signal (signo)); 127330fdc8d8SChris Lattner if (error.Success()) 127430fdc8d8SChris Lattner { 127530fdc8d8SChris Lattner result.SetStatus (eReturnStatusSuccessFinishResult); 127630fdc8d8SChris Lattner } 127730fdc8d8SChris Lattner else 127830fdc8d8SChris Lattner { 127930fdc8d8SChris Lattner result.AppendErrorWithFormat ("Failed to send signal %i: %s\n", signo, error.AsCString()); 128030fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 128130fdc8d8SChris Lattner } 128230fdc8d8SChris Lattner } 128330fdc8d8SChris Lattner } 128430fdc8d8SChris Lattner else 128530fdc8d8SChris Lattner { 1286fd54b368SJason Molenda result.AppendErrorWithFormat("'%s' takes exactly one signal number argument:\nUsage: %s\n", m_cmd_name.c_str(), 128730fdc8d8SChris Lattner m_cmd_syntax.c_str()); 128830fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 128930fdc8d8SChris Lattner } 129030fdc8d8SChris Lattner return result.Succeeded(); 129130fdc8d8SChris Lattner } 129230fdc8d8SChris Lattner }; 129330fdc8d8SChris Lattner 129430fdc8d8SChris Lattner 129530fdc8d8SChris Lattner //------------------------------------------------------------------------- 129630fdc8d8SChris Lattner // CommandObjectProcessInterrupt 129730fdc8d8SChris Lattner //------------------------------------------------------------------------- 1298bb9caf73SJim Ingham #pragma mark CommandObjectProcessInterrupt 129930fdc8d8SChris Lattner 13005a988416SJim Ingham class CommandObjectProcessInterrupt : public CommandObjectParsed 130130fdc8d8SChris Lattner { 130230fdc8d8SChris Lattner public: 130330fdc8d8SChris Lattner 130430fdc8d8SChris Lattner 1305a7015092SGreg Clayton CommandObjectProcessInterrupt (CommandInterpreter &interpreter) : 13065a988416SJim Ingham CommandObjectParsed (interpreter, 1307a7015092SGreg Clayton "process interrupt", 1308e3d26315SCaroline Tice "Interrupt the current process being debugged.", 130930fdc8d8SChris Lattner "process interrupt", 131030fdc8d8SChris Lattner eFlagProcessMustBeLaunched) 131130fdc8d8SChris Lattner { 131230fdc8d8SChris Lattner } 131330fdc8d8SChris Lattner 131430fdc8d8SChris Lattner ~CommandObjectProcessInterrupt () 131530fdc8d8SChris Lattner { 131630fdc8d8SChris Lattner } 131730fdc8d8SChris Lattner 13185a988416SJim Ingham protected: 131930fdc8d8SChris Lattner bool 13205a988416SJim Ingham DoExecute (Args& command, 132130fdc8d8SChris Lattner CommandReturnObject &result) 132230fdc8d8SChris Lattner { 1323c14ee32dSGreg Clayton Process *process = m_interpreter.GetExecutionContext().GetProcessPtr(); 132430fdc8d8SChris Lattner if (process == NULL) 132530fdc8d8SChris Lattner { 132630fdc8d8SChris Lattner result.AppendError ("no process to halt"); 132730fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 132830fdc8d8SChris Lattner return false; 132930fdc8d8SChris Lattner } 133030fdc8d8SChris Lattner 133130fdc8d8SChris Lattner if (command.GetArgumentCount() == 0) 133230fdc8d8SChris Lattner { 133330fdc8d8SChris Lattner Error error(process->Halt ()); 133430fdc8d8SChris Lattner if (error.Success()) 133530fdc8d8SChris Lattner { 133630fdc8d8SChris Lattner result.SetStatus (eReturnStatusSuccessFinishResult); 133730fdc8d8SChris Lattner 133830fdc8d8SChris Lattner // Maybe we should add a "SuspendThreadPlans so we 133930fdc8d8SChris Lattner // can halt, and keep in place all the current thread plans. 134030fdc8d8SChris Lattner process->GetThreadList().DiscardThreadPlans(); 134130fdc8d8SChris Lattner } 134230fdc8d8SChris Lattner else 134330fdc8d8SChris Lattner { 134430fdc8d8SChris Lattner result.AppendErrorWithFormat ("Failed to halt process: %s\n", error.AsCString()); 134530fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 134630fdc8d8SChris Lattner } 134730fdc8d8SChris Lattner } 134830fdc8d8SChris Lattner else 134930fdc8d8SChris Lattner { 1350fd54b368SJason Molenda result.AppendErrorWithFormat("'%s' takes no arguments:\nUsage: %s\n", 135130fdc8d8SChris Lattner m_cmd_name.c_str(), 135230fdc8d8SChris Lattner m_cmd_syntax.c_str()); 135330fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 135430fdc8d8SChris Lattner } 135530fdc8d8SChris Lattner return result.Succeeded(); 135630fdc8d8SChris Lattner } 135730fdc8d8SChris Lattner }; 135830fdc8d8SChris Lattner 135930fdc8d8SChris Lattner //------------------------------------------------------------------------- 136030fdc8d8SChris Lattner // CommandObjectProcessKill 136130fdc8d8SChris Lattner //------------------------------------------------------------------------- 1362bb9caf73SJim Ingham #pragma mark CommandObjectProcessKill 136330fdc8d8SChris Lattner 13645a988416SJim Ingham class CommandObjectProcessKill : public CommandObjectParsed 136530fdc8d8SChris Lattner { 136630fdc8d8SChris Lattner public: 136730fdc8d8SChris Lattner 1368a7015092SGreg Clayton CommandObjectProcessKill (CommandInterpreter &interpreter) : 13695a988416SJim Ingham CommandObjectParsed (interpreter, 1370a7015092SGreg Clayton "process kill", 1371e3d26315SCaroline Tice "Terminate the current process being debugged.", 137230fdc8d8SChris Lattner "process kill", 137330fdc8d8SChris Lattner eFlagProcessMustBeLaunched) 137430fdc8d8SChris Lattner { 137530fdc8d8SChris Lattner } 137630fdc8d8SChris Lattner 137730fdc8d8SChris Lattner ~CommandObjectProcessKill () 137830fdc8d8SChris Lattner { 137930fdc8d8SChris Lattner } 138030fdc8d8SChris Lattner 13815a988416SJim Ingham protected: 138230fdc8d8SChris Lattner bool 13835a988416SJim Ingham DoExecute (Args& command, 138430fdc8d8SChris Lattner CommandReturnObject &result) 138530fdc8d8SChris Lattner { 1386c14ee32dSGreg Clayton Process *process = m_interpreter.GetExecutionContext().GetProcessPtr(); 138730fdc8d8SChris Lattner if (process == NULL) 138830fdc8d8SChris Lattner { 138930fdc8d8SChris Lattner result.AppendError ("no process to kill"); 139030fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 139130fdc8d8SChris Lattner return false; 139230fdc8d8SChris Lattner } 139330fdc8d8SChris Lattner 139430fdc8d8SChris Lattner if (command.GetArgumentCount() == 0) 139530fdc8d8SChris Lattner { 139630fdc8d8SChris Lattner Error error (process->Destroy()); 139730fdc8d8SChris Lattner if (error.Success()) 139830fdc8d8SChris Lattner { 139930fdc8d8SChris Lattner result.SetStatus (eReturnStatusSuccessFinishResult); 140030fdc8d8SChris Lattner } 140130fdc8d8SChris Lattner else 140230fdc8d8SChris Lattner { 140330fdc8d8SChris Lattner result.AppendErrorWithFormat ("Failed to kill process: %s\n", error.AsCString()); 140430fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 140530fdc8d8SChris Lattner } 140630fdc8d8SChris Lattner } 140730fdc8d8SChris Lattner else 140830fdc8d8SChris Lattner { 1409fd54b368SJason Molenda result.AppendErrorWithFormat("'%s' takes no arguments:\nUsage: %s\n", 141030fdc8d8SChris Lattner m_cmd_name.c_str(), 141130fdc8d8SChris Lattner m_cmd_syntax.c_str()); 141230fdc8d8SChris Lattner result.SetStatus (eReturnStatusFailed); 141330fdc8d8SChris Lattner } 141430fdc8d8SChris Lattner return result.Succeeded(); 141530fdc8d8SChris Lattner } 141630fdc8d8SChris Lattner }; 141730fdc8d8SChris Lattner 141830fdc8d8SChris Lattner //------------------------------------------------------------------------- 14194b9bea87SJim Ingham // CommandObjectProcessStatus 14204b9bea87SJim Ingham //------------------------------------------------------------------------- 1421bb9caf73SJim Ingham #pragma mark CommandObjectProcessStatus 1422bb9caf73SJim Ingham 14235a988416SJim Ingham class CommandObjectProcessStatus : public CommandObjectParsed 14244b9bea87SJim Ingham { 14254b9bea87SJim Ingham public: 1426a7015092SGreg Clayton CommandObjectProcessStatus (CommandInterpreter &interpreter) : 14275a988416SJim Ingham CommandObjectParsed (interpreter, 1428a7015092SGreg Clayton "process status", 1429e3d26315SCaroline Tice "Show the current status and location of executing process.", 1430e3d26315SCaroline Tice "process status", 14314b9bea87SJim Ingham 0) 14324b9bea87SJim Ingham { 14334b9bea87SJim Ingham } 14344b9bea87SJim Ingham 14354b9bea87SJim Ingham ~CommandObjectProcessStatus() 14364b9bea87SJim Ingham { 14374b9bea87SJim Ingham } 14384b9bea87SJim Ingham 14394b9bea87SJim Ingham 14404b9bea87SJim Ingham bool 14415a988416SJim Ingham DoExecute (Args& command, CommandReturnObject &result) 14424b9bea87SJim Ingham { 14437260f620SGreg Clayton Stream &strm = result.GetOutputStream(); 14444b9bea87SJim Ingham result.SetStatus (eReturnStatusSuccessFinishNoResult); 14458b82f087SGreg Clayton ExecutionContext exe_ctx(m_interpreter.GetExecutionContext()); 1446c14ee32dSGreg Clayton Process *process = exe_ctx.GetProcessPtr(); 1447c14ee32dSGreg Clayton if (process) 14484b9bea87SJim Ingham { 14497260f620SGreg Clayton const bool only_threads_with_stop_reason = true; 14507260f620SGreg Clayton const uint32_t start_frame = 0; 14517260f620SGreg Clayton const uint32_t num_frames = 1; 14527260f620SGreg Clayton const uint32_t num_frames_with_source = 1; 1453c14ee32dSGreg Clayton process->GetStatus(strm); 1454c14ee32dSGreg Clayton process->GetThreadStatus (strm, 14557260f620SGreg Clayton only_threads_with_stop_reason, 14567260f620SGreg Clayton start_frame, 14577260f620SGreg Clayton num_frames, 14587260f620SGreg Clayton num_frames_with_source); 14597260f620SGreg Clayton 14604b9bea87SJim Ingham } 14614b9bea87SJim Ingham else 14624b9bea87SJim Ingham { 14637260f620SGreg Clayton result.AppendError ("No process."); 14644b9bea87SJim Ingham result.SetStatus (eReturnStatusFailed); 14654b9bea87SJim Ingham } 14664b9bea87SJim Ingham return result.Succeeded(); 14674b9bea87SJim Ingham } 14684b9bea87SJim Ingham }; 14694b9bea87SJim Ingham 14704b9bea87SJim Ingham //------------------------------------------------------------------------- 147135731357SCaroline Tice // CommandObjectProcessHandle 147235731357SCaroline Tice //------------------------------------------------------------------------- 1473bb9caf73SJim Ingham #pragma mark CommandObjectProcessHandle 147435731357SCaroline Tice 14755a988416SJim Ingham class CommandObjectProcessHandle : public CommandObjectParsed 147635731357SCaroline Tice { 147735731357SCaroline Tice public: 147835731357SCaroline Tice 147935731357SCaroline Tice class CommandOptions : public Options 148035731357SCaroline Tice { 148135731357SCaroline Tice public: 148235731357SCaroline Tice 1483eb0103f2SGreg Clayton CommandOptions (CommandInterpreter &interpreter) : 1484eb0103f2SGreg Clayton Options (interpreter) 148535731357SCaroline Tice { 1486f6b8b581SGreg Clayton OptionParsingStarting (); 148735731357SCaroline Tice } 148835731357SCaroline Tice 148935731357SCaroline Tice ~CommandOptions () 149035731357SCaroline Tice { 149135731357SCaroline Tice } 149235731357SCaroline Tice 149335731357SCaroline Tice Error 1494f6b8b581SGreg Clayton SetOptionValue (uint32_t option_idx, const char *option_arg) 149535731357SCaroline Tice { 149635731357SCaroline Tice Error error; 149735731357SCaroline Tice char short_option = (char) m_getopt_table[option_idx].val; 149835731357SCaroline Tice 149935731357SCaroline Tice switch (short_option) 150035731357SCaroline Tice { 150135731357SCaroline Tice case 's': 150235731357SCaroline Tice stop = option_arg; 150335731357SCaroline Tice break; 150435731357SCaroline Tice case 'n': 150535731357SCaroline Tice notify = option_arg; 150635731357SCaroline Tice break; 150735731357SCaroline Tice case 'p': 150835731357SCaroline Tice pass = option_arg; 150935731357SCaroline Tice break; 151035731357SCaroline Tice default: 151186edbf41SGreg Clayton error.SetErrorStringWithFormat("invalid short option character '%c'", short_option); 151235731357SCaroline Tice break; 151335731357SCaroline Tice } 151435731357SCaroline Tice return error; 151535731357SCaroline Tice } 151635731357SCaroline Tice 151735731357SCaroline Tice void 1518f6b8b581SGreg Clayton OptionParsingStarting () 151935731357SCaroline Tice { 152035731357SCaroline Tice stop.clear(); 152135731357SCaroline Tice notify.clear(); 152235731357SCaroline Tice pass.clear(); 152335731357SCaroline Tice } 152435731357SCaroline Tice 1525e0d378b3SGreg Clayton const OptionDefinition* 152635731357SCaroline Tice GetDefinitions () 152735731357SCaroline Tice { 152835731357SCaroline Tice return g_option_table; 152935731357SCaroline Tice } 153035731357SCaroline Tice 153135731357SCaroline Tice // Options table: Required for subclasses of Options. 153235731357SCaroline Tice 1533e0d378b3SGreg Clayton static OptionDefinition g_option_table[]; 153435731357SCaroline Tice 153535731357SCaroline Tice // Instance variables to hold the values for command options. 153635731357SCaroline Tice 153735731357SCaroline Tice std::string stop; 153835731357SCaroline Tice std::string notify; 153935731357SCaroline Tice std::string pass; 154035731357SCaroline Tice }; 154135731357SCaroline Tice 154235731357SCaroline Tice 154335731357SCaroline Tice CommandObjectProcessHandle (CommandInterpreter &interpreter) : 15445a988416SJim Ingham CommandObjectParsed (interpreter, 154535731357SCaroline Tice "process handle", 154610ad7993SCaroline Tice "Show or update what the process and debugger should do with various signals received from the OS.", 1547eb0103f2SGreg Clayton NULL), 1548eb0103f2SGreg Clayton m_options (interpreter) 154935731357SCaroline Tice { 155010ad7993SCaroline Tice SetHelpLong ("If no signals are specified, update them all. If no update option is specified, list the current values.\n"); 155135731357SCaroline Tice CommandArgumentEntry arg; 1552c0dbdfb6SCaroline Tice CommandArgumentData signal_arg; 155335731357SCaroline Tice 1554c0dbdfb6SCaroline Tice signal_arg.arg_type = eArgTypeUnixSignal; 1555c0dbdfb6SCaroline Tice signal_arg.arg_repetition = eArgRepeatStar; 155635731357SCaroline Tice 1557c0dbdfb6SCaroline Tice arg.push_back (signal_arg); 155835731357SCaroline Tice 155935731357SCaroline Tice m_arguments.push_back (arg); 156035731357SCaroline Tice } 156135731357SCaroline Tice 156235731357SCaroline Tice ~CommandObjectProcessHandle () 156335731357SCaroline Tice { 156435731357SCaroline Tice } 156535731357SCaroline Tice 156635731357SCaroline Tice Options * 156735731357SCaroline Tice GetOptions () 156835731357SCaroline Tice { 156935731357SCaroline Tice return &m_options; 157035731357SCaroline Tice } 157135731357SCaroline Tice 157235731357SCaroline Tice bool 157310ad7993SCaroline Tice VerifyCommandOptionValue (const std::string &option, int &real_value) 157435731357SCaroline Tice { 157535731357SCaroline Tice bool okay = true; 157635731357SCaroline Tice 157710ad7993SCaroline Tice bool success = false; 157810ad7993SCaroline Tice bool tmp_value = Args::StringToBoolean (option.c_str(), false, &success); 157910ad7993SCaroline Tice 158010ad7993SCaroline Tice if (success && tmp_value) 158110ad7993SCaroline Tice real_value = 1; 158210ad7993SCaroline Tice else if (success && !tmp_value) 158310ad7993SCaroline Tice real_value = 0; 158435731357SCaroline Tice else 158535731357SCaroline Tice { 158635731357SCaroline Tice // If the value isn't 'true' or 'false', it had better be 0 or 1. 158710ad7993SCaroline Tice real_value = Args::StringToUInt32 (option.c_str(), 3); 158810ad7993SCaroline Tice if (real_value != 0 && real_value != 1) 158935731357SCaroline Tice okay = false; 159035731357SCaroline Tice } 159135731357SCaroline Tice 159235731357SCaroline Tice return okay; 159335731357SCaroline Tice } 159435731357SCaroline Tice 159510ad7993SCaroline Tice void 159610ad7993SCaroline Tice PrintSignalHeader (Stream &str) 159710ad7993SCaroline Tice { 159810ad7993SCaroline Tice str.Printf ("NAME PASS STOP NOTIFY\n"); 159910ad7993SCaroline Tice str.Printf ("========== ===== ===== ======\n"); 160010ad7993SCaroline Tice } 160110ad7993SCaroline Tice 160210ad7993SCaroline Tice void 160310ad7993SCaroline Tice PrintSignal (Stream &str, int32_t signo, const char *sig_name, UnixSignals &signals) 160410ad7993SCaroline Tice { 160510ad7993SCaroline Tice bool stop; 160610ad7993SCaroline Tice bool suppress; 160710ad7993SCaroline Tice bool notify; 160810ad7993SCaroline Tice 160910ad7993SCaroline Tice str.Printf ("%-10s ", sig_name); 161010ad7993SCaroline Tice if (signals.GetSignalInfo (signo, suppress, stop, notify)) 161110ad7993SCaroline Tice { 161210ad7993SCaroline Tice bool pass = !suppress; 161310ad7993SCaroline Tice str.Printf ("%s %s %s", 161410ad7993SCaroline Tice (pass ? "true " : "false"), 161510ad7993SCaroline Tice (stop ? "true " : "false"), 161610ad7993SCaroline Tice (notify ? "true " : "false")); 161710ad7993SCaroline Tice } 161810ad7993SCaroline Tice str.Printf ("\n"); 161910ad7993SCaroline Tice } 162010ad7993SCaroline Tice 162110ad7993SCaroline Tice void 162210ad7993SCaroline Tice PrintSignalInformation (Stream &str, Args &signal_args, int num_valid_signals, UnixSignals &signals) 162310ad7993SCaroline Tice { 162410ad7993SCaroline Tice PrintSignalHeader (str); 162510ad7993SCaroline Tice 162610ad7993SCaroline Tice if (num_valid_signals > 0) 162710ad7993SCaroline Tice { 162810ad7993SCaroline Tice size_t num_args = signal_args.GetArgumentCount(); 162910ad7993SCaroline Tice for (size_t i = 0; i < num_args; ++i) 163010ad7993SCaroline Tice { 163110ad7993SCaroline Tice int32_t signo = signals.GetSignalNumberFromName (signal_args.GetArgumentAtIndex (i)); 163210ad7993SCaroline Tice if (signo != LLDB_INVALID_SIGNAL_NUMBER) 163310ad7993SCaroline Tice PrintSignal (str, signo, signal_args.GetArgumentAtIndex (i), signals); 163410ad7993SCaroline Tice } 163510ad7993SCaroline Tice } 163610ad7993SCaroline Tice else // Print info for ALL signals 163710ad7993SCaroline Tice { 163810ad7993SCaroline Tice int32_t signo = signals.GetFirstSignalNumber(); 163910ad7993SCaroline Tice while (signo != LLDB_INVALID_SIGNAL_NUMBER) 164010ad7993SCaroline Tice { 164110ad7993SCaroline Tice PrintSignal (str, signo, signals.GetSignalAsCString (signo), signals); 164210ad7993SCaroline Tice signo = signals.GetNextSignalNumber (signo); 164310ad7993SCaroline Tice } 164410ad7993SCaroline Tice } 164510ad7993SCaroline Tice } 164610ad7993SCaroline Tice 16475a988416SJim Ingham protected: 164835731357SCaroline Tice bool 16495a988416SJim Ingham DoExecute (Args &signal_args, CommandReturnObject &result) 165035731357SCaroline Tice { 165135731357SCaroline Tice TargetSP target_sp = m_interpreter.GetDebugger().GetSelectedTarget(); 165235731357SCaroline Tice 165335731357SCaroline Tice if (!target_sp) 165435731357SCaroline Tice { 165535731357SCaroline Tice result.AppendError ("No current target;" 165635731357SCaroline Tice " cannot handle signals until you have a valid target and process.\n"); 165735731357SCaroline Tice result.SetStatus (eReturnStatusFailed); 165835731357SCaroline Tice return false; 165935731357SCaroline Tice } 166035731357SCaroline Tice 166135731357SCaroline Tice ProcessSP process_sp = target_sp->GetProcessSP(); 166235731357SCaroline Tice 166335731357SCaroline Tice if (!process_sp) 166435731357SCaroline Tice { 166535731357SCaroline Tice result.AppendError ("No current process; cannot handle signals until you have a valid process.\n"); 166635731357SCaroline Tice result.SetStatus (eReturnStatusFailed); 166735731357SCaroline Tice return false; 166835731357SCaroline Tice } 166935731357SCaroline Tice 167035731357SCaroline Tice int stop_action = -1; // -1 means leave the current setting alone 167135731357SCaroline Tice int pass_action = -1; // -1 means leave the current setting alone 167235731357SCaroline Tice int notify_action = -1; // -1 means leave the current setting alone 167335731357SCaroline Tice 167435731357SCaroline Tice if (! m_options.stop.empty() 167510ad7993SCaroline Tice && ! VerifyCommandOptionValue (m_options.stop, stop_action)) 167635731357SCaroline Tice { 167735731357SCaroline Tice result.AppendError ("Invalid argument for command option --stop; must be true or false.\n"); 167835731357SCaroline Tice result.SetStatus (eReturnStatusFailed); 167935731357SCaroline Tice return false; 168035731357SCaroline Tice } 168135731357SCaroline Tice 168235731357SCaroline Tice if (! m_options.notify.empty() 168310ad7993SCaroline Tice && ! VerifyCommandOptionValue (m_options.notify, notify_action)) 168435731357SCaroline Tice { 168535731357SCaroline Tice result.AppendError ("Invalid argument for command option --notify; must be true or false.\n"); 168635731357SCaroline Tice result.SetStatus (eReturnStatusFailed); 168735731357SCaroline Tice return false; 168835731357SCaroline Tice } 168935731357SCaroline Tice 169035731357SCaroline Tice if (! m_options.pass.empty() 169110ad7993SCaroline Tice && ! VerifyCommandOptionValue (m_options.pass, pass_action)) 169235731357SCaroline Tice { 169335731357SCaroline Tice result.AppendError ("Invalid argument for command option --pass; must be true or false.\n"); 169435731357SCaroline Tice result.SetStatus (eReturnStatusFailed); 169535731357SCaroline Tice return false; 169635731357SCaroline Tice } 169735731357SCaroline Tice 169835731357SCaroline Tice size_t num_args = signal_args.GetArgumentCount(); 169935731357SCaroline Tice UnixSignals &signals = process_sp->GetUnixSignals(); 170035731357SCaroline Tice int num_signals_set = 0; 170135731357SCaroline Tice 170210ad7993SCaroline Tice if (num_args > 0) 170310ad7993SCaroline Tice { 170435731357SCaroline Tice for (size_t i = 0; i < num_args; ++i) 170535731357SCaroline Tice { 170635731357SCaroline Tice int32_t signo = signals.GetSignalNumberFromName (signal_args.GetArgumentAtIndex (i)); 170735731357SCaroline Tice if (signo != LLDB_INVALID_SIGNAL_NUMBER) 170835731357SCaroline Tice { 170910ad7993SCaroline Tice // Casting the actions as bools here should be okay, because VerifyCommandOptionValue guarantees 171035731357SCaroline Tice // the value is either 0 or 1. 171135731357SCaroline Tice if (stop_action != -1) 171235731357SCaroline Tice signals.SetShouldStop (signo, (bool) stop_action); 171335731357SCaroline Tice if (pass_action != -1) 171435731357SCaroline Tice { 171510ad7993SCaroline Tice bool suppress = ! ((bool) pass_action); 171610ad7993SCaroline Tice signals.SetShouldSuppress (signo, suppress); 171735731357SCaroline Tice } 171835731357SCaroline Tice if (notify_action != -1) 171935731357SCaroline Tice signals.SetShouldNotify (signo, (bool) notify_action); 172035731357SCaroline Tice ++num_signals_set; 172135731357SCaroline Tice } 172235731357SCaroline Tice else 172335731357SCaroline Tice { 172435731357SCaroline Tice result.AppendErrorWithFormat ("Invalid signal name '%s'\n", signal_args.GetArgumentAtIndex (i)); 172535731357SCaroline Tice } 172635731357SCaroline Tice } 172710ad7993SCaroline Tice } 172810ad7993SCaroline Tice else 172910ad7993SCaroline Tice { 173010ad7993SCaroline Tice // No signal specified, if any command options were specified, update ALL signals. 173110ad7993SCaroline Tice if ((notify_action != -1) || (stop_action != -1) || (pass_action != -1)) 173210ad7993SCaroline Tice { 173310ad7993SCaroline Tice if (m_interpreter.Confirm ("Do you really want to update all the signals?", false)) 173410ad7993SCaroline Tice { 173510ad7993SCaroline Tice int32_t signo = signals.GetFirstSignalNumber(); 173610ad7993SCaroline Tice while (signo != LLDB_INVALID_SIGNAL_NUMBER) 173710ad7993SCaroline Tice { 173810ad7993SCaroline Tice if (notify_action != -1) 173910ad7993SCaroline Tice signals.SetShouldNotify (signo, (bool) notify_action); 174010ad7993SCaroline Tice if (stop_action != -1) 174110ad7993SCaroline Tice signals.SetShouldStop (signo, (bool) stop_action); 174210ad7993SCaroline Tice if (pass_action != -1) 174310ad7993SCaroline Tice { 174410ad7993SCaroline Tice bool suppress = ! ((bool) pass_action); 174510ad7993SCaroline Tice signals.SetShouldSuppress (signo, suppress); 174610ad7993SCaroline Tice } 174710ad7993SCaroline Tice signo = signals.GetNextSignalNumber (signo); 174810ad7993SCaroline Tice } 174910ad7993SCaroline Tice } 175010ad7993SCaroline Tice } 175110ad7993SCaroline Tice } 175210ad7993SCaroline Tice 175310ad7993SCaroline Tice PrintSignalInformation (result.GetOutputStream(), signal_args, num_signals_set, signals); 175435731357SCaroline Tice 175535731357SCaroline Tice if (num_signals_set > 0) 175635731357SCaroline Tice result.SetStatus (eReturnStatusSuccessFinishNoResult); 175735731357SCaroline Tice else 175835731357SCaroline Tice result.SetStatus (eReturnStatusFailed); 175935731357SCaroline Tice 176035731357SCaroline Tice return result.Succeeded(); 176135731357SCaroline Tice } 176235731357SCaroline Tice 176335731357SCaroline Tice CommandOptions m_options; 176435731357SCaroline Tice }; 176535731357SCaroline Tice 1766e0d378b3SGreg Clayton OptionDefinition 176735731357SCaroline Tice CommandObjectProcessHandle::CommandOptions::g_option_table[] = 176835731357SCaroline Tice { 176935731357SCaroline 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." }, 177035731357SCaroline 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." }, 177135731357SCaroline Tice { LLDB_OPT_SET_1, false, "pass", 'p', required_argument, NULL, 0, eArgTypeBoolean, "Whether or not the signal should be passed to the process." }, 177235731357SCaroline Tice { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL } 177335731357SCaroline Tice }; 177435731357SCaroline Tice 177535731357SCaroline Tice //------------------------------------------------------------------------- 177630fdc8d8SChris Lattner // CommandObjectMultiwordProcess 177730fdc8d8SChris Lattner //------------------------------------------------------------------------- 177830fdc8d8SChris Lattner 17796611103cSGreg Clayton CommandObjectMultiwordProcess::CommandObjectMultiwordProcess (CommandInterpreter &interpreter) : 1780a7015092SGreg Clayton CommandObjectMultiword (interpreter, 1781a7015092SGreg Clayton "process", 178230fdc8d8SChris Lattner "A set of commands for operating on a process.", 178330fdc8d8SChris Lattner "process <subcommand> [<subcommand-options>]") 178430fdc8d8SChris Lattner { 1785a7015092SGreg Clayton LoadSubCommand ("attach", CommandObjectSP (new CommandObjectProcessAttach (interpreter))); 1786a7015092SGreg Clayton LoadSubCommand ("launch", CommandObjectSP (new CommandObjectProcessLaunch (interpreter))); 1787a7015092SGreg Clayton LoadSubCommand ("continue", CommandObjectSP (new CommandObjectProcessContinue (interpreter))); 1788b766a73dSGreg Clayton LoadSubCommand ("connect", CommandObjectSP (new CommandObjectProcessConnect (interpreter))); 1789a7015092SGreg Clayton LoadSubCommand ("detach", CommandObjectSP (new CommandObjectProcessDetach (interpreter))); 17908f343b09SGreg Clayton LoadSubCommand ("load", CommandObjectSP (new CommandObjectProcessLoad (interpreter))); 17918f343b09SGreg Clayton LoadSubCommand ("unload", CommandObjectSP (new CommandObjectProcessUnload (interpreter))); 1792a7015092SGreg Clayton LoadSubCommand ("signal", CommandObjectSP (new CommandObjectProcessSignal (interpreter))); 179335731357SCaroline Tice LoadSubCommand ("handle", CommandObjectSP (new CommandObjectProcessHandle (interpreter))); 1794a7015092SGreg Clayton LoadSubCommand ("status", CommandObjectSP (new CommandObjectProcessStatus (interpreter))); 1795a7015092SGreg Clayton LoadSubCommand ("interrupt", CommandObjectSP (new CommandObjectProcessInterrupt (interpreter))); 1796a7015092SGreg Clayton LoadSubCommand ("kill", CommandObjectSP (new CommandObjectProcessKill (interpreter))); 179730fdc8d8SChris Lattner } 179830fdc8d8SChris Lattner 179930fdc8d8SChris Lattner CommandObjectMultiwordProcess::~CommandObjectMultiwordProcess () 180030fdc8d8SChris Lattner { 180130fdc8d8SChris Lattner } 180230fdc8d8SChris Lattner 1803