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
1640af72e1SJim Ingham #include "lldb/Interpreter/Args.h"
1740af72e1SJim Ingham #include "lldb/Interpreter/Options.h"
1830fdc8d8SChris Lattner #include "lldb/Core/State.h"
197260f620SGreg Clayton #include "lldb/Host/Host.h"
2030fdc8d8SChris Lattner #include "lldb/Interpreter/CommandInterpreter.h"
2130fdc8d8SChris Lattner #include "lldb/Interpreter/CommandReturnObject.h"
22e996fd30SGreg Clayton #include "lldb/Target/Platform.h"
2330fdc8d8SChris Lattner #include "lldb/Target/Process.h"
2430fdc8d8SChris Lattner #include "lldb/Target/Target.h"
2530fdc8d8SChris Lattner #include "lldb/Target/Thread.h"
2630fdc8d8SChris Lattner 
2730fdc8d8SChris Lattner using namespace lldb;
2830fdc8d8SChris Lattner using namespace lldb_private;
2930fdc8d8SChris Lattner 
3030fdc8d8SChris Lattner //-------------------------------------------------------------------------
3130fdc8d8SChris Lattner // CommandObjectProcessLaunch
3230fdc8d8SChris Lattner //-------------------------------------------------------------------------
33bb9caf73SJim Ingham #pragma mark CommandObjectProjectLaunch
3430fdc8d8SChris Lattner class CommandObjectProcessLaunch : public CommandObject
3530fdc8d8SChris Lattner {
3630fdc8d8SChris Lattner public:
3730fdc8d8SChris Lattner 
38982c9762SGreg Clayton //    class CommandOptions : public Options
39982c9762SGreg Clayton //    {
40982c9762SGreg Clayton //    public:
41982c9762SGreg Clayton //
42982c9762SGreg Clayton //        CommandOptions (CommandInterpreter &interpreter) :
43982c9762SGreg Clayton //            Options(interpreter)
44982c9762SGreg Clayton //        {
45982c9762SGreg Clayton //            // Keep default values of all options in one place: OptionParsingStarting ()
46982c9762SGreg Clayton //            OptionParsingStarting ();
47982c9762SGreg Clayton //        }
48982c9762SGreg Clayton //
49982c9762SGreg Clayton //        ~CommandOptions ()
50982c9762SGreg Clayton //        {
51982c9762SGreg Clayton //        }
52982c9762SGreg Clayton //
53982c9762SGreg Clayton //        Error
54982c9762SGreg Clayton //        SetOptionValue (uint32_t option_idx, const char *option_arg)
55982c9762SGreg Clayton //        {
56982c9762SGreg Clayton //            Error error;
57982c9762SGreg Clayton //            char short_option = (char) m_getopt_table[option_idx].val;
58982c9762SGreg Clayton //
59982c9762SGreg Clayton //            switch (short_option)
60982c9762SGreg Clayton //            {
61982c9762SGreg Clayton //                case 's':   stop_at_entry = true;               break;
62982c9762SGreg Clayton //                case 'e':   stderr_path.assign (option_arg);    break;
63982c9762SGreg Clayton //                case 'i':   stdin_path.assign (option_arg);     break;
64982c9762SGreg Clayton //                case 'o':   stdout_path.assign (option_arg);    break;
65982c9762SGreg Clayton //                case 'p':   plugin_name.assign (option_arg);    break;
66982c9762SGreg Clayton //                case 'n':   no_stdio = true;                    break;
67982c9762SGreg Clayton //                case 'w':   working_dir.assign (option_arg);    break;
68982c9762SGreg Clayton //                case 't':
69982c9762SGreg Clayton //                    if (option_arg && option_arg[0])
70982c9762SGreg Clayton //                        tty_name.assign (option_arg);
71982c9762SGreg Clayton //                    in_new_tty = true;
72982c9762SGreg Clayton //                    break;
73982c9762SGreg Clayton //                default:
74982c9762SGreg Clayton //                    error.SetErrorStringWithFormat("invalid short option character '%c'", short_option);
75982c9762SGreg Clayton //                    break;
76982c9762SGreg Clayton //
77982c9762SGreg Clayton //            }
78982c9762SGreg Clayton //            return error;
79982c9762SGreg Clayton //        }
80982c9762SGreg Clayton //
81982c9762SGreg Clayton //        void
82982c9762SGreg Clayton //        OptionParsingStarting ()
83982c9762SGreg Clayton //        {
84982c9762SGreg Clayton //            stop_at_entry = false;
85982c9762SGreg Clayton //            in_new_tty = false;
86982c9762SGreg Clayton //            tty_name.clear();
87982c9762SGreg Clayton //            stdin_path.clear();
88982c9762SGreg Clayton //            stdout_path.clear();
89982c9762SGreg Clayton //            stderr_path.clear();
90982c9762SGreg Clayton //            plugin_name.clear();
91982c9762SGreg Clayton //            working_dir.clear();
92982c9762SGreg Clayton //            no_stdio = false;
93982c9762SGreg Clayton //        }
94982c9762SGreg Clayton //
95982c9762SGreg Clayton //        const OptionDefinition*
96982c9762SGreg Clayton //        GetDefinitions ()
97982c9762SGreg Clayton //        {
98982c9762SGreg Clayton //            return g_option_table;
99982c9762SGreg Clayton //        }
100982c9762SGreg Clayton //
101982c9762SGreg Clayton //        // Options table: Required for subclasses of Options.
102982c9762SGreg Clayton //
103982c9762SGreg Clayton //        static OptionDefinition g_option_table[];
104982c9762SGreg Clayton //
105982c9762SGreg Clayton //        // Instance variables to hold the values for command options.
106982c9762SGreg Clayton //
107982c9762SGreg Clayton //        bool stop_at_entry;
108982c9762SGreg Clayton //        bool in_new_tty;
109982c9762SGreg Clayton //        bool no_stdio;
110982c9762SGreg Clayton //        std::string tty_name;
111982c9762SGreg Clayton //        std::string stderr_path;
112982c9762SGreg Clayton //        std::string stdin_path;
113982c9762SGreg Clayton //        std::string stdout_path;
114982c9762SGreg Clayton //        std::string plugin_name;
115982c9762SGreg Clayton //        std::string working_dir;
116982c9762SGreg Clayton //
117982c9762SGreg Clayton //    };
11830fdc8d8SChris Lattner 
119a7015092SGreg Clayton     CommandObjectProcessLaunch (CommandInterpreter &interpreter) :
120a7015092SGreg Clayton         CommandObject (interpreter,
121a7015092SGreg Clayton                        "process launch",
122e3d26315SCaroline Tice                        "Launch the executable in the debugger.",
123eb0103f2SGreg Clayton                        NULL),
124eb0103f2SGreg Clayton         m_options (interpreter)
12530fdc8d8SChris Lattner     {
126405fe67fSCaroline Tice         CommandArgumentEntry arg;
127405fe67fSCaroline Tice         CommandArgumentData run_args_arg;
128405fe67fSCaroline Tice 
129405fe67fSCaroline Tice         // Define the first (and only) variant of this arg.
130405fe67fSCaroline Tice         run_args_arg.arg_type = eArgTypeRunArgs;
131405fe67fSCaroline Tice         run_args_arg.arg_repetition = eArgRepeatOptional;
132405fe67fSCaroline Tice 
133405fe67fSCaroline Tice         // There is only one variant this argument could be; put it into the argument entry.
134405fe67fSCaroline Tice         arg.push_back (run_args_arg);
135405fe67fSCaroline Tice 
136405fe67fSCaroline Tice         // Push the data for the first argument into the m_arguments vector.
137405fe67fSCaroline Tice         m_arguments.push_back (arg);
13830fdc8d8SChris Lattner     }
13930fdc8d8SChris Lattner 
14030fdc8d8SChris Lattner 
14130fdc8d8SChris Lattner     ~CommandObjectProcessLaunch ()
14230fdc8d8SChris Lattner     {
14330fdc8d8SChris Lattner     }
14430fdc8d8SChris Lattner 
14530fdc8d8SChris Lattner     Options *
14630fdc8d8SChris Lattner     GetOptions ()
14730fdc8d8SChris Lattner     {
14830fdc8d8SChris Lattner         return &m_options;
14930fdc8d8SChris Lattner     }
15030fdc8d8SChris Lattner 
15130fdc8d8SChris Lattner     bool
15205faeb71SGreg Clayton     Execute (Args& launch_args, CommandReturnObject &result)
15330fdc8d8SChris Lattner     {
1541d885966SGreg Clayton         Debugger &debugger = m_interpreter.GetDebugger();
1551d885966SGreg Clayton         Target *target = debugger.GetSelectedTarget().get();
1561d885966SGreg Clayton         Error error;
15730fdc8d8SChris Lattner 
15830fdc8d8SChris Lattner         if (target == NULL)
15930fdc8d8SChris Lattner         {
160effe5c95SGreg Clayton             result.AppendError ("invalid target, create a debug target using the 'target create' command");
16130fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
16230fdc8d8SChris Lattner             return false;
16330fdc8d8SChris Lattner         }
16430fdc8d8SChris Lattner         // If our listener is NULL, users aren't allows to launch
16530fdc8d8SChris Lattner         char filename[PATH_MAX];
166aa149cbdSGreg Clayton         const Module *exe_module = target->GetExecutableModulePointer();
16771337622SGreg Clayton 
16871337622SGreg Clayton         if (exe_module == NULL)
16971337622SGreg Clayton         {
170effe5c95SGreg Clayton             result.AppendError ("no file in target, create a debug target using the 'target create' command");
17171337622SGreg Clayton             result.SetStatus (eReturnStatusFailed);
17271337622SGreg Clayton             return false;
17371337622SGreg Clayton         }
17471337622SGreg Clayton 
17530fdc8d8SChris Lattner         exe_module->GetFileSpec().GetPath (filename, sizeof(filename));
17630fdc8d8SChris Lattner 
177982c9762SGreg Clayton         const bool add_exe_file_as_first_arg = true;
178982c9762SGreg Clayton         m_options.launch_info.SetExecutableFile(exe_module->GetFileSpec(), add_exe_file_as_first_arg);
179982c9762SGreg Clayton 
18071337622SGreg Clayton         StateType state = eStateInvalid;
181c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
18271337622SGreg Clayton         if (process)
18371337622SGreg Clayton         {
18471337622SGreg Clayton             state = process->GetState();
18571337622SGreg Clayton 
18671337622SGreg Clayton             if (process->IsAlive() && state != eStateConnected)
18730fdc8d8SChris Lattner             {
188513c26ceSGreg Clayton                 char message[1024];
189513c26ceSGreg Clayton                 if (process->GetState() == eStateAttaching)
190513c26ceSGreg Clayton                     ::strncpy (message, "There is a pending attach, abort it and launch a new process?", sizeof(message));
191513c26ceSGreg Clayton                 else
192513c26ceSGreg Clayton                     ::strncpy (message, "There is a running process, kill it and restart?", sizeof(message));
193513c26ceSGreg Clayton 
194513c26ceSGreg Clayton                 if (!m_interpreter.Confirm (message, true))
195bb9caf73SJim Ingham                 {
19630fdc8d8SChris Lattner                     result.SetStatus (eReturnStatusFailed);
19730fdc8d8SChris Lattner                     return false;
19830fdc8d8SChris Lattner                 }
199bb9caf73SJim Ingham                 else
200bb9caf73SJim Ingham                 {
2011d885966SGreg Clayton                     Error destroy_error (process->Destroy());
2021d885966SGreg Clayton                     if (destroy_error.Success())
203bb9caf73SJim Ingham                     {
204bb9caf73SJim Ingham                         result.SetStatus (eReturnStatusSuccessFinishResult);
205bb9caf73SJim Ingham                     }
206bb9caf73SJim Ingham                     else
207bb9caf73SJim Ingham                     {
2081d885966SGreg Clayton                         result.AppendErrorWithFormat ("Failed to kill process: %s\n", destroy_error.AsCString());
209bb9caf73SJim Ingham                         result.SetStatus (eReturnStatusFailed);
210bb9caf73SJim Ingham                     }
211bb9caf73SJim Ingham                 }
212bb9caf73SJim Ingham             }
21371337622SGreg Clayton         }
21430fdc8d8SChris Lattner 
215144f3a9cSGreg Clayton         if (launch_args.GetArgumentCount() == 0)
216144f3a9cSGreg Clayton         {
217144f3a9cSGreg Clayton             const Args &process_args = target->GetRunArguments();
218144f3a9cSGreg Clayton             if (process_args.GetArgumentCount() > 0)
219144f3a9cSGreg Clayton                 m_options.launch_info.GetArguments().AppendArguments (process_args);
220144f3a9cSGreg Clayton         }
221144f3a9cSGreg Clayton         else
22230fdc8d8SChris Lattner         {
223982c9762SGreg Clayton             m_options.launch_info.GetArguments().AppendArguments (launch_args);
224982c9762SGreg Clayton         }
2251d885966SGreg Clayton 
226144f3a9cSGreg Clayton         if (target->GetDisableASLR())
227144f3a9cSGreg Clayton             m_options.launch_info.GetFlags().Set (eLaunchFlagDisableASLR);
228144f3a9cSGreg Clayton 
229144f3a9cSGreg Clayton         if (target->GetDisableSTDIO())
230144f3a9cSGreg Clayton             m_options.launch_info.GetFlags().Set (eLaunchFlagDisableSTDIO);
231144f3a9cSGreg Clayton 
232144f3a9cSGreg Clayton         m_options.launch_info.GetFlags().Set (eLaunchFlagDebug);
233144f3a9cSGreg Clayton 
234144f3a9cSGreg Clayton         Args environment;
235144f3a9cSGreg Clayton         target->GetEnvironmentAsArgs (environment);
236144f3a9cSGreg Clayton         if (environment.GetArgumentCount() > 0)
237144f3a9cSGreg Clayton             m_options.launch_info.GetEnvironmentEntries ().AppendArguments (environment);
238144f3a9cSGreg Clayton 
239*ee95ed50SGreg Clayton         // Finalize the file actions, and if none were given, default to opening
240*ee95ed50SGreg Clayton         // up a pseudo terminal
241*ee95ed50SGreg Clayton         const bool default_to_use_pty = true;
242*ee95ed50SGreg Clayton         m_options.launch_info.FinalizeFileActions (target, default_to_use_pty);
2431d885966SGreg Clayton 
2441d885966SGreg Clayton         if (state == eStateConnected)
2451d885966SGreg Clayton         {
2461d885966SGreg Clayton             if (m_options.launch_info.GetFlags().Test (eLaunchFlagLaunchInTTY))
2471d885966SGreg Clayton             {
2481d885966SGreg Clayton                 result.AppendWarning("can't launch in tty when launching through a remote connection");
2491d885966SGreg Clayton                 m_options.launch_info.GetFlags().Clear (eLaunchFlagLaunchInTTY);
2501d885966SGreg Clayton             }
2511d885966SGreg Clayton         }
252982c9762SGreg Clayton         else
253982c9762SGreg Clayton         {
254144f3a9cSGreg Clayton             if (!m_options.launch_info.GetArchitecture().IsValid())
255c9ed478aSGreg Clayton                 m_options.launch_info.GetArchitecture() = target->GetArchitecture();
256c9ed478aSGreg Clayton 
2571d885966SGreg Clayton             process = target->GetPlatform()->DebugProcess (m_options.launch_info,
2581d885966SGreg Clayton                                                            debugger,
2591d885966SGreg Clayton                                                            target,
2601d885966SGreg Clayton                                                            debugger.GetListener(),
2611d885966SGreg Clayton                                                            error).get();
2621d885966SGreg Clayton 
2631d885966SGreg Clayton             if (process == NULL)
2641d885966SGreg Clayton             {
265144f3a9cSGreg Clayton                 result.SetError (error, "failed to launch or debug process");
2661d885966SGreg Clayton                 return false;
2671d885966SGreg Clayton             }
2681d885966SGreg Clayton         }
26930fdc8d8SChris Lattner 
27030fdc8d8SChris Lattner         if (error.Success())
27130fdc8d8SChris Lattner         {
27264195a2cSGreg Clayton             const char *archname = exe_module->GetArchitecture().GetArchitectureName();
27319388cfcSGreg Clayton 
27481c22f61SGreg Clayton             result.AppendMessageWithFormat ("Process %llu launched: '%s' (%s)\n", process->GetID(), filename, archname);
27505faeb71SGreg Clayton             result.SetDidChangeProcessState (true);
276982c9762SGreg Clayton             if (m_options.launch_info.GetFlags().Test(eLaunchFlagStopAtEntry) == false)
27730fdc8d8SChris Lattner             {
27805faeb71SGreg Clayton                 result.SetStatus (eReturnStatusSuccessContinuingNoResult);
27930fdc8d8SChris Lattner                 StateType state = process->WaitForProcessToStop (NULL);
28030fdc8d8SChris Lattner 
28130fdc8d8SChris Lattner                 if (state == eStateStopped)
28230fdc8d8SChris Lattner                 {
28305faeb71SGreg Clayton                     error = process->Resume();
28405faeb71SGreg Clayton                     if (error.Success())
28505faeb71SGreg Clayton                     {
28605faeb71SGreg Clayton                         bool synchronous_execution = m_interpreter.GetSynchronous ();
28730fdc8d8SChris Lattner                         if (synchronous_execution)
28830fdc8d8SChris Lattner                         {
28905faeb71SGreg Clayton                             state = process->WaitForProcessToStop (NULL);
2902637f825SGreg Clayton                             const bool must_be_alive = true;
2912637f825SGreg Clayton                             if (!StateIsStoppedState(state, must_be_alive))
292514487e8SGreg Clayton                             {
293144f3a9cSGreg Clayton                                 result.AppendErrorWithFormat ("process isn't stopped: %s", StateAsCString(state));
294514487e8SGreg Clayton                             }
29530fdc8d8SChris Lattner                             result.SetDidChangeProcessState (true);
29605faeb71SGreg Clayton                             result.SetStatus (eReturnStatusSuccessFinishResult);
29705faeb71SGreg Clayton                         }
29805faeb71SGreg Clayton                         else
29905faeb71SGreg Clayton                         {
30005faeb71SGreg Clayton                             result.SetStatus (eReturnStatusSuccessContinuingNoResult);
30105faeb71SGreg Clayton                         }
30205faeb71SGreg Clayton                     }
303514487e8SGreg Clayton                     else
304514487e8SGreg Clayton                     {
305144f3a9cSGreg Clayton                         result.AppendErrorWithFormat ("process resume at entry point failed: %s", error.AsCString());
306514487e8SGreg Clayton                         result.SetStatus (eReturnStatusFailed);
30730fdc8d8SChris Lattner                     }
30830fdc8d8SChris Lattner                 }
309514487e8SGreg Clayton                 else
310514487e8SGreg Clayton                 {
311144f3a9cSGreg Clayton                     result.AppendErrorWithFormat ("initial process state wasn't stopped: %s", StateAsCString(state));
312514487e8SGreg Clayton                     result.SetStatus (eReturnStatusFailed);
313514487e8SGreg Clayton                 }
314514487e8SGreg Clayton             }
315514487e8SGreg Clayton         }
316514487e8SGreg Clayton         else
317514487e8SGreg Clayton         {
318197bacffSGreg Clayton             result.AppendErrorWithFormat ("process launch failed: %s", error.AsCString());
319514487e8SGreg Clayton             result.SetStatus (eReturnStatusFailed);
32030fdc8d8SChris Lattner         }
32130fdc8d8SChris Lattner 
32230fdc8d8SChris Lattner         return result.Succeeded();
32330fdc8d8SChris Lattner     }
32430fdc8d8SChris Lattner 
325ebc09c36SJim Ingham     virtual const char *GetRepeatCommand (Args &current_command_args, uint32_t index)
326ebc09c36SJim Ingham     {
327ebc09c36SJim Ingham         // No repeat for "process launch"...
328ebc09c36SJim Ingham         return "";
329ebc09c36SJim Ingham     }
330ebc09c36SJim Ingham 
33130fdc8d8SChris Lattner protected:
332982c9762SGreg Clayton     ProcessLaunchCommandOptions m_options;
33330fdc8d8SChris Lattner };
33430fdc8d8SChris Lattner 
33530fdc8d8SChris Lattner 
336982c9762SGreg Clayton //#define SET1 LLDB_OPT_SET_1
337982c9762SGreg Clayton //#define SET2 LLDB_OPT_SET_2
338982c9762SGreg Clayton //#define SET3 LLDB_OPT_SET_3
339982c9762SGreg Clayton //
340982c9762SGreg Clayton //OptionDefinition
341982c9762SGreg Clayton //CommandObjectProcessLaunch::CommandOptions::g_option_table[] =
342982c9762SGreg Clayton //{
343982c9762SGreg 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."},
344982c9762SGreg Clayton //{ SET1              , false, "stdin",         'i', required_argument, NULL, 0, eArgTypePath,    "Redirect stdin for the process to <path>."},
345982c9762SGreg Clayton //{ SET1              , false, "stdout",        'o', required_argument, NULL, 0, eArgTypePath,    "Redirect stdout for the process to <path>."},
346982c9762SGreg Clayton //{ SET1              , false, "stderr",        'e', required_argument, NULL, 0, eArgTypePath,    "Redirect stderr for the process to <path>."},
347982c9762SGreg Clayton //{ SET1 | SET2 | SET3, false, "plugin",        'p', required_argument, NULL, 0, eArgTypePlugin,  "Name of the process plugin you want to use."},
348982c9762SGreg 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."},
349982c9762SGreg Clayton //{               SET3, false, "no-stdio",      'n', no_argument,       NULL, 0, eArgTypeNone,    "Do not set up for terminal I/O to go to running process."},
350982c9762SGreg Clayton //{ SET1 | SET2 | SET3, false, "working-dir",   'w', required_argument, NULL, 0, eArgTypePath,    "Set the current working directory to <path> when running the inferior."},
351982c9762SGreg Clayton //{ 0,                  false, NULL,             0,  0,                 NULL, 0, eArgTypeNone,    NULL }
352982c9762SGreg Clayton //};
353982c9762SGreg Clayton //
354982c9762SGreg Clayton //#undef SET1
355982c9762SGreg Clayton //#undef SET2
356982c9762SGreg Clayton //#undef SET3
35730fdc8d8SChris Lattner 
35830fdc8d8SChris Lattner //-------------------------------------------------------------------------
35930fdc8d8SChris Lattner // CommandObjectProcessAttach
36030fdc8d8SChris Lattner //-------------------------------------------------------------------------
361bb9caf73SJim Ingham #pragma mark CommandObjectProcessAttach
36230fdc8d8SChris Lattner class CommandObjectProcessAttach : public CommandObject
36330fdc8d8SChris Lattner {
36430fdc8d8SChris Lattner public:
36530fdc8d8SChris Lattner 
36630fdc8d8SChris Lattner     class CommandOptions : public Options
36730fdc8d8SChris Lattner     {
36830fdc8d8SChris Lattner     public:
36930fdc8d8SChris Lattner 
370eb0103f2SGreg Clayton         CommandOptions (CommandInterpreter &interpreter) :
371eb0103f2SGreg Clayton             Options(interpreter)
37230fdc8d8SChris Lattner         {
373f6b8b581SGreg Clayton             // Keep default values of all options in one place: OptionParsingStarting ()
374f6b8b581SGreg Clayton             OptionParsingStarting ();
37530fdc8d8SChris Lattner         }
37630fdc8d8SChris Lattner 
37730fdc8d8SChris Lattner         ~CommandOptions ()
37830fdc8d8SChris Lattner         {
37930fdc8d8SChris Lattner         }
38030fdc8d8SChris Lattner 
38130fdc8d8SChris Lattner         Error
382f6b8b581SGreg Clayton         SetOptionValue (uint32_t option_idx, const char *option_arg)
38330fdc8d8SChris Lattner         {
38430fdc8d8SChris Lattner             Error error;
38530fdc8d8SChris Lattner             char short_option = (char) m_getopt_table[option_idx].val;
38630fdc8d8SChris Lattner             bool success = false;
38730fdc8d8SChris Lattner             switch (short_option)
38830fdc8d8SChris Lattner             {
38930fdc8d8SChris Lattner                 case 'p':
390144f3a9cSGreg Clayton                     {
391144f3a9cSGreg Clayton                         lldb::pid_t pid = Args::StringToUInt32 (option_arg, LLDB_INVALID_PROCESS_ID, 0, &success);
39230fdc8d8SChris Lattner                         if (!success || pid == LLDB_INVALID_PROCESS_ID)
39330fdc8d8SChris Lattner                         {
39486edbf41SGreg Clayton                             error.SetErrorStringWithFormat("invalid process ID '%s'", option_arg);
39530fdc8d8SChris Lattner                         }
396144f3a9cSGreg Clayton                         else
397144f3a9cSGreg Clayton                         {
398144f3a9cSGreg Clayton                             attach_info.SetProcessID (pid);
399144f3a9cSGreg Clayton                         }
400144f3a9cSGreg Clayton                     }
40130fdc8d8SChris Lattner                     break;
40230fdc8d8SChris Lattner 
40330fdc8d8SChris Lattner                 case 'P':
404144f3a9cSGreg Clayton                     attach_info.SetProcessPluginName (option_arg);
40530fdc8d8SChris Lattner                     break;
40630fdc8d8SChris Lattner 
40730fdc8d8SChris Lattner                 case 'n':
408144f3a9cSGreg Clayton                     attach_info.GetExecutableFile().SetFile(option_arg, false);
40930fdc8d8SChris Lattner                     break;
41030fdc8d8SChris Lattner 
41130fdc8d8SChris Lattner                 case 'w':
412144f3a9cSGreg Clayton                     attach_info.SetWaitForLaunch(true);
41330fdc8d8SChris Lattner                     break;
41430fdc8d8SChris Lattner 
41530fdc8d8SChris Lattner                 default:
41686edbf41SGreg Clayton                     error.SetErrorStringWithFormat("invalid short option character '%c'", short_option);
41730fdc8d8SChris Lattner                     break;
41830fdc8d8SChris Lattner             }
41930fdc8d8SChris Lattner             return error;
42030fdc8d8SChris Lattner         }
42130fdc8d8SChris Lattner 
42230fdc8d8SChris Lattner         void
423f6b8b581SGreg Clayton         OptionParsingStarting ()
42430fdc8d8SChris Lattner         {
425144f3a9cSGreg Clayton             attach_info.Clear();
42630fdc8d8SChris Lattner         }
42730fdc8d8SChris Lattner 
428e0d378b3SGreg Clayton         const OptionDefinition*
42930fdc8d8SChris Lattner         GetDefinitions ()
43030fdc8d8SChris Lattner         {
43130fdc8d8SChris Lattner             return g_option_table;
43230fdc8d8SChris Lattner         }
43330fdc8d8SChris Lattner 
4345aee162fSJim Ingham         virtual bool
435eb0103f2SGreg Clayton         HandleOptionArgumentCompletion (Args &input,
4365aee162fSJim Ingham                                         int cursor_index,
4375aee162fSJim Ingham                                         int char_pos,
4385aee162fSJim Ingham                                         OptionElementVector &opt_element_vector,
4395aee162fSJim Ingham                                         int opt_element_index,
4405aee162fSJim Ingham                                         int match_start_point,
4415aee162fSJim Ingham                                         int max_return_elements,
4425aee162fSJim Ingham                                         bool &word_complete,
4435aee162fSJim Ingham                                         StringList &matches)
4445aee162fSJim Ingham         {
4455aee162fSJim Ingham             int opt_arg_pos = opt_element_vector[opt_element_index].opt_arg_pos;
4465aee162fSJim Ingham             int opt_defs_index = opt_element_vector[opt_element_index].opt_defs_index;
4475aee162fSJim Ingham 
4485aee162fSJim Ingham             // We are only completing the name option for now...
4495aee162fSJim Ingham 
450e0d378b3SGreg Clayton             const OptionDefinition *opt_defs = GetDefinitions();
4515aee162fSJim Ingham             if (opt_defs[opt_defs_index].short_option == 'n')
4525aee162fSJim Ingham             {
4535aee162fSJim Ingham                 // Are we in the name?
4545aee162fSJim Ingham 
4555aee162fSJim Ingham                 // Look to see if there is a -P argument provided, and if so use that plugin, otherwise
4565aee162fSJim Ingham                 // use the default plugin.
4575aee162fSJim Ingham 
4585aee162fSJim Ingham                 const char *partial_name = NULL;
4595aee162fSJim Ingham                 partial_name = input.GetArgumentAtIndex(opt_arg_pos);
4605aee162fSJim Ingham 
4618b82f087SGreg Clayton                 PlatformSP platform_sp (m_interpreter.GetPlatform (true));
462e996fd30SGreg Clayton                 if (platform_sp)
4635aee162fSJim Ingham                 {
4648b82f087SGreg Clayton                     ProcessInstanceInfoList process_infos;
4658b82f087SGreg Clayton                     ProcessInstanceInfoMatch match_info;
46632e0a750SGreg Clayton                     if (partial_name)
46732e0a750SGreg Clayton                     {
468144f3a9cSGreg Clayton                         match_info.GetProcessInfo().GetExecutableFile().SetFile(partial_name, false);
46932e0a750SGreg Clayton                         match_info.SetNameMatchType(eNameMatchStartsWith);
47032e0a750SGreg Clayton                     }
47132e0a750SGreg Clayton                     platform_sp->FindProcesses (match_info, process_infos);
472e996fd30SGreg Clayton                     const uint32_t num_matches = process_infos.GetSize();
473e996fd30SGreg Clayton                     if (num_matches > 0)
474e996fd30SGreg Clayton                     {
475e996fd30SGreg Clayton                         for (uint32_t i=0; i<num_matches; ++i)
476e996fd30SGreg Clayton                         {
477e996fd30SGreg Clayton                             matches.AppendString (process_infos.GetProcessNameAtIndex(i),
478e996fd30SGreg Clayton                                                   process_infos.GetProcessNameLengthAtIndex(i));
4795aee162fSJim Ingham                         }
4805aee162fSJim Ingham                     }
4815aee162fSJim Ingham                 }
4825aee162fSJim Ingham             }
4835aee162fSJim Ingham 
4845aee162fSJim Ingham             return false;
4855aee162fSJim Ingham         }
4865aee162fSJim Ingham 
48730fdc8d8SChris Lattner         // Options table: Required for subclasses of Options.
48830fdc8d8SChris Lattner 
489e0d378b3SGreg Clayton         static OptionDefinition g_option_table[];
49030fdc8d8SChris Lattner 
49130fdc8d8SChris Lattner         // Instance variables to hold the values for command options.
49230fdc8d8SChris Lattner 
493144f3a9cSGreg Clayton         ProcessAttachInfo attach_info;
49430fdc8d8SChris Lattner     };
49530fdc8d8SChris Lattner 
496a7015092SGreg Clayton     CommandObjectProcessAttach (CommandInterpreter &interpreter) :
497a7015092SGreg Clayton         CommandObject (interpreter,
498a7015092SGreg Clayton                        "process attach",
499e3d26315SCaroline Tice                        "Attach to a process.",
500eb0103f2SGreg Clayton                        "process attach <cmd-options>"),
501eb0103f2SGreg Clayton         m_options (interpreter)
5025aee162fSJim Ingham     {
5035aee162fSJim Ingham     }
5045aee162fSJim Ingham 
5055aee162fSJim Ingham     ~CommandObjectProcessAttach ()
5065aee162fSJim Ingham     {
5075aee162fSJim Ingham     }
5085aee162fSJim Ingham 
5095aee162fSJim Ingham     bool
510a7015092SGreg Clayton     Execute (Args& command,
5115aee162fSJim Ingham              CommandReturnObject &result)
5125aee162fSJim Ingham     {
513a7015092SGreg Clayton         Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get();
51431412642SJim Ingham         // N.B. The attach should be synchronous.  It doesn't help much to get the prompt back between initiating the attach
51531412642SJim Ingham         // and the target actually stopping.  So even if the interpreter is set to be asynchronous, we wait for the stop
51631412642SJim Ingham         // ourselves here.
5175aee162fSJim Ingham 
518c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
51971337622SGreg Clayton         StateType state = eStateInvalid;
5205aee162fSJim Ingham         if (process)
5215aee162fSJim Ingham         {
52271337622SGreg Clayton             state = process->GetState();
52371337622SGreg Clayton             if (process->IsAlive() && state != eStateConnected)
5245aee162fSJim Ingham             {
52581c22f61SGreg Clayton                 result.AppendErrorWithFormat ("Process %llu is currently being debugged, kill the process before attaching.\n",
5265aee162fSJim Ingham                                               process->GetID());
5275aee162fSJim Ingham                 result.SetStatus (eReturnStatusFailed);
5285aee162fSJim Ingham                 return false;
5295aee162fSJim Ingham             }
5305aee162fSJim Ingham         }
5315aee162fSJim Ingham 
5325aee162fSJim Ingham         if (target == NULL)
5335aee162fSJim Ingham         {
5345aee162fSJim Ingham             // If there isn't a current target create one.
5355aee162fSJim Ingham             TargetSP new_target_sp;
5365aee162fSJim Ingham             FileSpec emptyFileSpec;
5375aee162fSJim Ingham             Error error;
5385aee162fSJim Ingham 
539a7015092SGreg Clayton             error = m_interpreter.GetDebugger().GetTargetList().CreateTarget (m_interpreter.GetDebugger(),
5405aee162fSJim Ingham                                                                               emptyFileSpec,
541cac9c5f9SGreg Clayton                                                                               NULL,
5425aee162fSJim Ingham                                                                               false,
543cac9c5f9SGreg Clayton                                                                               NULL, // No platform options
5445aee162fSJim Ingham                                                                               new_target_sp);
5455aee162fSJim Ingham             target = new_target_sp.get();
5465aee162fSJim Ingham             if (target == NULL || error.Fail())
5475aee162fSJim Ingham             {
548b766a73dSGreg Clayton                 result.AppendError(error.AsCString("Error creating target"));
5495aee162fSJim Ingham                 return false;
5505aee162fSJim Ingham             }
551a7015092SGreg Clayton             m_interpreter.GetDebugger().GetTargetList().SetSelectedTarget(target);
5525aee162fSJim Ingham         }
5535aee162fSJim Ingham 
5545aee162fSJim Ingham         // Record the old executable module, we want to issue a warning if the process of attaching changed the
5555aee162fSJim Ingham         // current executable (like somebody said "file foo" then attached to a PID whose executable was bar.)
5565aee162fSJim Ingham 
5575aee162fSJim Ingham         ModuleSP old_exec_module_sp = target->GetExecutableModule();
5585aee162fSJim Ingham         ArchSpec old_arch_spec = target->GetArchitecture();
5595aee162fSJim Ingham 
5605aee162fSJim Ingham         if (command.GetArgumentCount())
5615aee162fSJim Ingham         {
562fd54b368SJason Molenda             result.AppendErrorWithFormat("Invalid arguments for '%s'.\nUsage: %s\n", m_cmd_name.c_str(), m_cmd_syntax.c_str());
5635aee162fSJim Ingham             result.SetStatus (eReturnStatusFailed);
5645aee162fSJim Ingham         }
5655aee162fSJim Ingham         else
5665aee162fSJim Ingham         {
56771337622SGreg Clayton             if (state != eStateConnected)
56871337622SGreg Clayton             {
569144f3a9cSGreg Clayton                 const char *plugin_name = m_options.attach_info.GetProcessPluginName();
570a7015092SGreg Clayton                 process = target->CreateProcess (m_interpreter.GetDebugger().GetListener(), plugin_name).get();
57171337622SGreg Clayton             }
5725aee162fSJim Ingham 
5735aee162fSJim Ingham             if (process)
5745aee162fSJim Ingham             {
5755aee162fSJim Ingham                 Error error;
576144f3a9cSGreg Clayton                 // If no process info was specified, then use the target executable
577144f3a9cSGreg Clayton                 // name as the process to attach to by default
578144f3a9cSGreg Clayton                 if (!m_options.attach_info.ProcessInfoSpecified ())
5795aee162fSJim Ingham                 {
5803a0b9cdfSJim Ingham                     if (old_exec_module_sp)
581144f3a9cSGreg Clayton                         m_options.attach_info.GetExecutableFile().GetFilename() = old_exec_module_sp->GetFileSpec().GetFilename();
582144f3a9cSGreg Clayton 
583144f3a9cSGreg Clayton                     if (!m_options.attach_info.ProcessInfoSpecified ())
5843a0b9cdfSJim Ingham                     {
585144f3a9cSGreg Clayton                         error.SetErrorString ("no process specified, create a target with a file, or specify the --pid or --name command option");
5863a0b9cdfSJim Ingham                     }
5875aee162fSJim Ingham                 }
5883a0b9cdfSJim Ingham 
589144f3a9cSGreg Clayton                 if (error.Success())
5903a0b9cdfSJim Ingham                 {
591144f3a9cSGreg Clayton                     error = process->Attach (m_options.attach_info);
5923a0b9cdfSJim Ingham 
5935aee162fSJim Ingham                     if (error.Success())
5945aee162fSJim Ingham                     {
5955aee162fSJim Ingham                         result.SetStatus (eReturnStatusSuccessContinuingNoResult);
5965aee162fSJim Ingham                     }
5975aee162fSJim Ingham                     else
5985aee162fSJim Ingham                     {
599144f3a9cSGreg Clayton                         result.AppendErrorWithFormat ("attach failed: %s\n", error.AsCString());
6005aee162fSJim Ingham                         result.SetStatus (eReturnStatusFailed);
6015aee162fSJim Ingham                         return false;
6025aee162fSJim Ingham                     }
603bb3a283bSJim Ingham                     // If we're synchronous, wait for the stopped event and report that.
604bb3a283bSJim Ingham                     // Otherwise just return.
605bb3a283bSJim Ingham                     // FIXME: in the async case it will now be possible to get to the command
606bb3a283bSJim Ingham                     // interpreter with a state eStateAttaching.  Make sure we handle that correctly.
607bb3a283bSJim Ingham                     StateType state = process->WaitForProcessToStop (NULL);
608bb3a283bSJim Ingham 
609bb3a283bSJim Ingham                     result.SetDidChangeProcessState (true);
61081c22f61SGreg Clayton                     result.AppendMessageWithFormat ("Process %llu %s\n", process->GetID(), StateAsCString (state));
611bb3a283bSJim Ingham                     result.SetStatus (eReturnStatusSuccessFinishNoResult);
612bb3a283bSJim Ingham                 }
6135aee162fSJim Ingham             }
6145aee162fSJim Ingham         }
6155aee162fSJim Ingham 
6165aee162fSJim Ingham         if (result.Succeeded())
6175aee162fSJim Ingham         {
6185aee162fSJim Ingham             // Okay, we're done.  Last step is to warn if the executable module has changed:
619513c26ceSGreg Clayton             char new_path[PATH_MAX];
620aa149cbdSGreg Clayton             ModuleSP new_exec_module_sp (target->GetExecutableModule());
6215aee162fSJim Ingham             if (!old_exec_module_sp)
6225aee162fSJim Ingham             {
623513c26ceSGreg Clayton                 // We might not have a module if we attached to a raw pid...
624aa149cbdSGreg Clayton                 if (new_exec_module_sp)
625513c26ceSGreg Clayton                 {
626aa149cbdSGreg Clayton                     new_exec_module_sp->GetFileSpec().GetPath(new_path, PATH_MAX);
627513c26ceSGreg Clayton                     result.AppendMessageWithFormat("Executable module set to \"%s\".\n", new_path);
628513c26ceSGreg Clayton                 }
6295aee162fSJim Ingham             }
630aa149cbdSGreg Clayton             else if (old_exec_module_sp->GetFileSpec() != new_exec_module_sp->GetFileSpec())
6315aee162fSJim Ingham             {
632513c26ceSGreg Clayton                 char old_path[PATH_MAX];
6335aee162fSJim Ingham 
6345aee162fSJim Ingham                 old_exec_module_sp->GetFileSpec().GetPath (old_path, PATH_MAX);
635aa149cbdSGreg Clayton                 new_exec_module_sp->GetFileSpec().GetPath (new_path, PATH_MAX);
6365aee162fSJim Ingham 
6375aee162fSJim Ingham                 result.AppendWarningWithFormat("Executable module changed from \"%s\" to \"%s\".\n",
6385aee162fSJim Ingham                                                     old_path, new_path);
6395aee162fSJim Ingham             }
6405aee162fSJim Ingham 
6415aee162fSJim Ingham             if (!old_arch_spec.IsValid())
6425aee162fSJim Ingham             {
64364195a2cSGreg Clayton                 result.AppendMessageWithFormat ("Architecture set to: %s.\n", target->GetArchitecture().GetArchitectureName());
6445aee162fSJim Ingham             }
6455aee162fSJim Ingham             else if (old_arch_spec != target->GetArchitecture())
6465aee162fSJim Ingham             {
6475aee162fSJim Ingham                 result.AppendWarningWithFormat("Architecture changed from %s to %s.\n",
64864195a2cSGreg Clayton                                                 old_arch_spec.GetArchitectureName(), target->GetArchitecture().GetArchitectureName());
6495aee162fSJim Ingham             }
6505aee162fSJim Ingham         }
6515aee162fSJim Ingham         return result.Succeeded();
6525aee162fSJim Ingham     }
6535aee162fSJim Ingham 
6545aee162fSJim Ingham     Options *
6555aee162fSJim Ingham     GetOptions ()
6565aee162fSJim Ingham     {
6575aee162fSJim Ingham         return &m_options;
6585aee162fSJim Ingham     }
6595aee162fSJim Ingham 
66030fdc8d8SChris Lattner protected:
66130fdc8d8SChris Lattner 
66230fdc8d8SChris Lattner     CommandOptions m_options;
66330fdc8d8SChris Lattner };
66430fdc8d8SChris Lattner 
66530fdc8d8SChris Lattner 
666e0d378b3SGreg Clayton OptionDefinition
66730fdc8d8SChris Lattner CommandObjectProcessAttach::CommandOptions::g_option_table[] =
66830fdc8d8SChris Lattner {
669deaab222SCaroline Tice { LLDB_OPT_SET_ALL, false, "plugin", 'P', required_argument, NULL, 0, eArgTypePlugin,        "Name of the process plugin you want to use."},
670deaab222SCaroline Tice { LLDB_OPT_SET_1,   false, "pid",    'p', required_argument, NULL, 0, eArgTypePid,           "The process ID of an existing process to attach to."},
671deaab222SCaroline Tice { LLDB_OPT_SET_2,   false, "name",   'n', required_argument, NULL, 0, eArgTypeProcessName,  "The name of the process to attach to."},
672deaab222SCaroline Tice { LLDB_OPT_SET_2,   false, "waitfor",'w', no_argument,       NULL, 0, eArgTypeNone,              "Wait for the the process with <process-name> to launch."},
673deaab222SCaroline Tice { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL }
67430fdc8d8SChris Lattner };
67530fdc8d8SChris Lattner 
67630fdc8d8SChris Lattner //-------------------------------------------------------------------------
67730fdc8d8SChris Lattner // CommandObjectProcessContinue
67830fdc8d8SChris Lattner //-------------------------------------------------------------------------
679bb9caf73SJim Ingham #pragma mark CommandObjectProcessContinue
68030fdc8d8SChris Lattner 
68130fdc8d8SChris Lattner class CommandObjectProcessContinue : public CommandObject
68230fdc8d8SChris Lattner {
68330fdc8d8SChris Lattner public:
68430fdc8d8SChris Lattner 
685a7015092SGreg Clayton     CommandObjectProcessContinue (CommandInterpreter &interpreter) :
686a7015092SGreg Clayton         CommandObject (interpreter,
687a7015092SGreg Clayton                        "process continue",
688e3d26315SCaroline Tice                        "Continue execution of all threads in the current process.",
68930fdc8d8SChris Lattner                        "process continue",
69030fdc8d8SChris Lattner                        eFlagProcessMustBeLaunched | eFlagProcessMustBePaused)
69130fdc8d8SChris Lattner     {
69230fdc8d8SChris Lattner     }
69330fdc8d8SChris Lattner 
69430fdc8d8SChris Lattner 
69530fdc8d8SChris Lattner     ~CommandObjectProcessContinue ()
69630fdc8d8SChris Lattner     {
69730fdc8d8SChris Lattner     }
69830fdc8d8SChris Lattner 
69930fdc8d8SChris Lattner     bool
700a7015092SGreg Clayton     Execute (Args& command,
70130fdc8d8SChris Lattner              CommandReturnObject &result)
70230fdc8d8SChris Lattner     {
703c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
704a7015092SGreg Clayton         bool synchronous_execution = m_interpreter.GetSynchronous ();
70530fdc8d8SChris Lattner 
70630fdc8d8SChris Lattner         if (process == NULL)
70730fdc8d8SChris Lattner         {
70830fdc8d8SChris Lattner             result.AppendError ("no process to continue");
70930fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
71030fdc8d8SChris Lattner             return false;
71130fdc8d8SChris Lattner          }
71230fdc8d8SChris Lattner 
71330fdc8d8SChris Lattner         StateType state = process->GetState();
71430fdc8d8SChris Lattner         if (state == eStateStopped)
71530fdc8d8SChris Lattner         {
71630fdc8d8SChris Lattner             if (command.GetArgumentCount() != 0)
71730fdc8d8SChris Lattner             {
71830fdc8d8SChris Lattner                 result.AppendErrorWithFormat ("The '%s' command does not take any arguments.\n", m_cmd_name.c_str());
71930fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusFailed);
72030fdc8d8SChris Lattner                 return false;
72130fdc8d8SChris Lattner             }
72230fdc8d8SChris Lattner 
72330fdc8d8SChris Lattner             const uint32_t num_threads = process->GetThreadList().GetSize();
72430fdc8d8SChris Lattner 
72530fdc8d8SChris Lattner             // Set the actions that the threads should each take when resuming
72630fdc8d8SChris Lattner             for (uint32_t idx=0; idx<num_threads; ++idx)
72730fdc8d8SChris Lattner             {
72830fdc8d8SChris Lattner                 process->GetThreadList().GetThreadAtIndex(idx)->SetResumeState (eStateRunning);
72930fdc8d8SChris Lattner             }
73030fdc8d8SChris Lattner 
73130fdc8d8SChris Lattner             Error error(process->Resume());
73230fdc8d8SChris Lattner             if (error.Success())
73330fdc8d8SChris Lattner             {
73481c22f61SGreg Clayton                 result.AppendMessageWithFormat ("Process %llu resuming\n", process->GetID());
73530fdc8d8SChris Lattner                 if (synchronous_execution)
73630fdc8d8SChris Lattner                 {
737b132097bSGreg Clayton                     state = process->WaitForProcessToStop (NULL);
73830fdc8d8SChris Lattner 
73930fdc8d8SChris Lattner                     result.SetDidChangeProcessState (true);
74081c22f61SGreg Clayton                     result.AppendMessageWithFormat ("Process %llu %s\n", process->GetID(), StateAsCString (state));
74130fdc8d8SChris Lattner                     result.SetStatus (eReturnStatusSuccessFinishNoResult);
74230fdc8d8SChris Lattner                 }
74330fdc8d8SChris Lattner                 else
74430fdc8d8SChris Lattner                 {
74530fdc8d8SChris Lattner                     result.SetStatus (eReturnStatusSuccessContinuingNoResult);
74630fdc8d8SChris Lattner                 }
74730fdc8d8SChris Lattner             }
74830fdc8d8SChris Lattner             else
74930fdc8d8SChris Lattner             {
75030fdc8d8SChris Lattner                 result.AppendErrorWithFormat("Failed to resume process: %s.\n", error.AsCString());
75130fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusFailed);
75230fdc8d8SChris Lattner             }
75330fdc8d8SChris Lattner         }
75430fdc8d8SChris Lattner         else
75530fdc8d8SChris Lattner         {
75630fdc8d8SChris Lattner             result.AppendErrorWithFormat ("Process cannot be continued from its current state (%s).\n",
75730fdc8d8SChris Lattner                                          StateAsCString(state));
75830fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
75930fdc8d8SChris Lattner         }
76030fdc8d8SChris Lattner         return result.Succeeded();
76130fdc8d8SChris Lattner     }
76230fdc8d8SChris Lattner };
76330fdc8d8SChris Lattner 
76430fdc8d8SChris Lattner //-------------------------------------------------------------------------
76530fdc8d8SChris Lattner // CommandObjectProcessDetach
76630fdc8d8SChris Lattner //-------------------------------------------------------------------------
767bb9caf73SJim Ingham #pragma mark CommandObjectProcessDetach
76830fdc8d8SChris Lattner 
76930fdc8d8SChris Lattner class CommandObjectProcessDetach : public CommandObject
77030fdc8d8SChris Lattner {
77130fdc8d8SChris Lattner public:
77230fdc8d8SChris Lattner 
773a7015092SGreg Clayton     CommandObjectProcessDetach (CommandInterpreter &interpreter) :
774a7015092SGreg Clayton         CommandObject (interpreter,
775a7015092SGreg Clayton                        "process detach",
776e3d26315SCaroline Tice                        "Detach from the current process being debugged.",
77730fdc8d8SChris Lattner                        "process detach",
77830fdc8d8SChris Lattner                        eFlagProcessMustBeLaunched)
77930fdc8d8SChris Lattner     {
78030fdc8d8SChris Lattner     }
78130fdc8d8SChris Lattner 
78230fdc8d8SChris Lattner     ~CommandObjectProcessDetach ()
78330fdc8d8SChris Lattner     {
78430fdc8d8SChris Lattner     }
78530fdc8d8SChris Lattner 
78630fdc8d8SChris Lattner     bool
787a7015092SGreg Clayton     Execute (Args& command,
78830fdc8d8SChris Lattner              CommandReturnObject &result)
78930fdc8d8SChris Lattner     {
790c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
79130fdc8d8SChris Lattner         if (process == NULL)
79230fdc8d8SChris Lattner         {
79330fdc8d8SChris Lattner             result.AppendError ("must have a valid process in order to detach");
79430fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
79530fdc8d8SChris Lattner             return false;
79630fdc8d8SChris Lattner         }
79730fdc8d8SChris Lattner 
79881c22f61SGreg Clayton         result.AppendMessageWithFormat ("Detaching from process %llu\n", process->GetID());
79930fdc8d8SChris Lattner         Error error (process->Detach());
80030fdc8d8SChris Lattner         if (error.Success())
80130fdc8d8SChris Lattner         {
80230fdc8d8SChris Lattner             result.SetStatus (eReturnStatusSuccessFinishResult);
80330fdc8d8SChris Lattner         }
80430fdc8d8SChris Lattner         else
80530fdc8d8SChris Lattner         {
80630fdc8d8SChris Lattner             result.AppendErrorWithFormat ("Detach failed: %s\n", error.AsCString());
80730fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
80830fdc8d8SChris Lattner             return false;
80930fdc8d8SChris Lattner         }
81030fdc8d8SChris Lattner         return result.Succeeded();
81130fdc8d8SChris Lattner     }
81230fdc8d8SChris Lattner };
81330fdc8d8SChris Lattner 
81430fdc8d8SChris Lattner //-------------------------------------------------------------------------
815b766a73dSGreg Clayton // CommandObjectProcessConnect
816b766a73dSGreg Clayton //-------------------------------------------------------------------------
817b766a73dSGreg Clayton #pragma mark CommandObjectProcessConnect
818b766a73dSGreg Clayton 
819b766a73dSGreg Clayton class CommandObjectProcessConnect : public CommandObject
820b766a73dSGreg Clayton {
821b766a73dSGreg Clayton public:
822b766a73dSGreg Clayton 
823b766a73dSGreg Clayton     class CommandOptions : public Options
824b766a73dSGreg Clayton     {
825b766a73dSGreg Clayton     public:
826b766a73dSGreg Clayton 
827eb0103f2SGreg Clayton         CommandOptions (CommandInterpreter &interpreter) :
828eb0103f2SGreg Clayton             Options(interpreter)
829b766a73dSGreg Clayton         {
830f6b8b581SGreg Clayton             // Keep default values of all options in one place: OptionParsingStarting ()
831f6b8b581SGreg Clayton             OptionParsingStarting ();
832b766a73dSGreg Clayton         }
833b766a73dSGreg Clayton 
834b766a73dSGreg Clayton         ~CommandOptions ()
835b766a73dSGreg Clayton         {
836b766a73dSGreg Clayton         }
837b766a73dSGreg Clayton 
838b766a73dSGreg Clayton         Error
839f6b8b581SGreg Clayton         SetOptionValue (uint32_t option_idx, const char *option_arg)
840b766a73dSGreg Clayton         {
841b766a73dSGreg Clayton             Error error;
842b766a73dSGreg Clayton             char short_option = (char) m_getopt_table[option_idx].val;
843b766a73dSGreg Clayton 
844b766a73dSGreg Clayton             switch (short_option)
845b766a73dSGreg Clayton             {
846b766a73dSGreg Clayton             case 'p':
847b766a73dSGreg Clayton                 plugin_name.assign (option_arg);
848b766a73dSGreg Clayton                 break;
849b766a73dSGreg Clayton 
850b766a73dSGreg Clayton             default:
85186edbf41SGreg Clayton                 error.SetErrorStringWithFormat("invalid short option character '%c'", short_option);
852b766a73dSGreg Clayton                 break;
853b766a73dSGreg Clayton             }
854b766a73dSGreg Clayton             return error;
855b766a73dSGreg Clayton         }
856b766a73dSGreg Clayton 
857b766a73dSGreg Clayton         void
858f6b8b581SGreg Clayton         OptionParsingStarting ()
859b766a73dSGreg Clayton         {
860b766a73dSGreg Clayton             plugin_name.clear();
861b766a73dSGreg Clayton         }
862b766a73dSGreg Clayton 
863e0d378b3SGreg Clayton         const OptionDefinition*
864b766a73dSGreg Clayton         GetDefinitions ()
865b766a73dSGreg Clayton         {
866b766a73dSGreg Clayton             return g_option_table;
867b766a73dSGreg Clayton         }
868b766a73dSGreg Clayton 
869b766a73dSGreg Clayton         // Options table: Required for subclasses of Options.
870b766a73dSGreg Clayton 
871e0d378b3SGreg Clayton         static OptionDefinition g_option_table[];
872b766a73dSGreg Clayton 
873b766a73dSGreg Clayton         // Instance variables to hold the values for command options.
874b766a73dSGreg Clayton 
875b766a73dSGreg Clayton         std::string plugin_name;
876b766a73dSGreg Clayton     };
877b766a73dSGreg Clayton 
878b766a73dSGreg Clayton     CommandObjectProcessConnect (CommandInterpreter &interpreter) :
879b766a73dSGreg Clayton         CommandObject (interpreter,
880b766a73dSGreg Clayton                        "process connect",
881b766a73dSGreg Clayton                        "Connect to a remote debug service.",
882b766a73dSGreg Clayton                        "process connect <remote-url>",
883eb0103f2SGreg Clayton                        0),
884eb0103f2SGreg Clayton         m_options (interpreter)
885b766a73dSGreg Clayton     {
886b766a73dSGreg Clayton     }
887b766a73dSGreg Clayton 
888b766a73dSGreg Clayton     ~CommandObjectProcessConnect ()
889b766a73dSGreg Clayton     {
890b766a73dSGreg Clayton     }
891b766a73dSGreg Clayton 
892b766a73dSGreg Clayton 
893b766a73dSGreg Clayton     bool
894b766a73dSGreg Clayton     Execute (Args& command,
895b766a73dSGreg Clayton              CommandReturnObject &result)
896b766a73dSGreg Clayton     {
897b766a73dSGreg Clayton 
898b766a73dSGreg Clayton         TargetSP target_sp (m_interpreter.GetDebugger().GetSelectedTarget());
899b766a73dSGreg Clayton         Error error;
900c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
901b766a73dSGreg Clayton         if (process)
902b766a73dSGreg Clayton         {
903b766a73dSGreg Clayton             if (process->IsAlive())
904b766a73dSGreg Clayton             {
90581c22f61SGreg Clayton                 result.AppendErrorWithFormat ("Process %llu is currently being debugged, kill the process before connecting.\n",
906b766a73dSGreg Clayton                                               process->GetID());
907b766a73dSGreg Clayton                 result.SetStatus (eReturnStatusFailed);
908b766a73dSGreg Clayton                 return false;
909b766a73dSGreg Clayton             }
910b766a73dSGreg Clayton         }
911b766a73dSGreg Clayton 
912b766a73dSGreg Clayton         if (!target_sp)
913b766a73dSGreg Clayton         {
914b766a73dSGreg Clayton             // If there isn't a current target create one.
915b766a73dSGreg Clayton             FileSpec emptyFileSpec;
916b766a73dSGreg Clayton 
917b766a73dSGreg Clayton             error = m_interpreter.GetDebugger().GetTargetList().CreateTarget (m_interpreter.GetDebugger(),
918b766a73dSGreg Clayton                                                                               emptyFileSpec,
919cac9c5f9SGreg Clayton                                                                               NULL,
920b766a73dSGreg Clayton                                                                               false,
921cac9c5f9SGreg Clayton                                                                               NULL, // No platform options
922b766a73dSGreg Clayton                                                                               target_sp);
923b766a73dSGreg Clayton             if (!target_sp || error.Fail())
924b766a73dSGreg Clayton             {
925b766a73dSGreg Clayton                 result.AppendError(error.AsCString("Error creating target"));
926b766a73dSGreg Clayton                 result.SetStatus (eReturnStatusFailed);
927b766a73dSGreg Clayton                 return false;
928b766a73dSGreg Clayton             }
929b766a73dSGreg Clayton             m_interpreter.GetDebugger().GetTargetList().SetSelectedTarget(target_sp.get());
930b766a73dSGreg Clayton         }
931b766a73dSGreg Clayton 
932b766a73dSGreg Clayton         if (command.GetArgumentCount() == 1)
933b766a73dSGreg Clayton         {
934b766a73dSGreg Clayton             const char *plugin_name = NULL;
935b766a73dSGreg Clayton             if (!m_options.plugin_name.empty())
936b766a73dSGreg Clayton                 plugin_name = m_options.plugin_name.c_str();
937b766a73dSGreg Clayton 
938b766a73dSGreg Clayton             const char *remote_url = command.GetArgumentAtIndex(0);
939b766a73dSGreg Clayton             process = target_sp->CreateProcess (m_interpreter.GetDebugger().GetListener(), plugin_name).get();
940b766a73dSGreg Clayton 
941b766a73dSGreg Clayton             if (process)
942b766a73dSGreg Clayton             {
943b766a73dSGreg Clayton                 error = process->ConnectRemote (remote_url);
944b766a73dSGreg Clayton 
945b766a73dSGreg Clayton                 if (error.Fail())
946b766a73dSGreg Clayton                 {
947b766a73dSGreg Clayton                     result.AppendError(error.AsCString("Remote connect failed"));
948b766a73dSGreg Clayton                     result.SetStatus (eReturnStatusFailed);
949b766a73dSGreg Clayton                     return false;
950b766a73dSGreg Clayton                 }
951b766a73dSGreg Clayton             }
952b766a73dSGreg Clayton             else
953b766a73dSGreg Clayton             {
954fd54b368SJason 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",
955fd54b368SJason Molenda                                               m_cmd_name.c_str());
956b766a73dSGreg Clayton                 result.SetStatus (eReturnStatusFailed);
957b766a73dSGreg Clayton             }
958b766a73dSGreg Clayton         }
959b766a73dSGreg Clayton         else
960b766a73dSGreg Clayton         {
961fd54b368SJason Molenda             result.AppendErrorWithFormat ("'%s' takes exactly one argument:\nUsage: %s\n",
962b766a73dSGreg Clayton                                           m_cmd_name.c_str(),
963b766a73dSGreg Clayton                                           m_cmd_syntax.c_str());
964b766a73dSGreg Clayton             result.SetStatus (eReturnStatusFailed);
965b766a73dSGreg Clayton         }
966b766a73dSGreg Clayton         return result.Succeeded();
967b766a73dSGreg Clayton     }
968b766a73dSGreg Clayton 
969b766a73dSGreg Clayton     Options *
970b766a73dSGreg Clayton     GetOptions ()
971b766a73dSGreg Clayton     {
972b766a73dSGreg Clayton         return &m_options;
973b766a73dSGreg Clayton     }
974b766a73dSGreg Clayton 
975b766a73dSGreg Clayton protected:
976b766a73dSGreg Clayton 
977b766a73dSGreg Clayton     CommandOptions m_options;
978b766a73dSGreg Clayton };
979b766a73dSGreg Clayton 
980b766a73dSGreg Clayton 
981e0d378b3SGreg Clayton OptionDefinition
982b766a73dSGreg Clayton CommandObjectProcessConnect::CommandOptions::g_option_table[] =
983b766a73dSGreg Clayton {
984b766a73dSGreg Clayton     { LLDB_OPT_SET_ALL, false, "plugin", 'p', required_argument, NULL, 0, eArgTypePlugin, "Name of the process plugin you want to use."},
985b766a73dSGreg Clayton     { 0,                false, NULL,      0 , 0,                 NULL, 0, eArgTypeNone,   NULL }
986b766a73dSGreg Clayton };
987b766a73dSGreg Clayton 
988b766a73dSGreg Clayton //-------------------------------------------------------------------------
9898f343b09SGreg Clayton // CommandObjectProcessLoad
9908f343b09SGreg Clayton //-------------------------------------------------------------------------
991bb9caf73SJim Ingham #pragma mark CommandObjectProcessLoad
9928f343b09SGreg Clayton 
9938f343b09SGreg Clayton class CommandObjectProcessLoad : public CommandObject
9948f343b09SGreg Clayton {
9958f343b09SGreg Clayton public:
9968f343b09SGreg Clayton 
9978f343b09SGreg Clayton     CommandObjectProcessLoad (CommandInterpreter &interpreter) :
9988f343b09SGreg Clayton         CommandObject (interpreter,
9998f343b09SGreg Clayton                        "process load",
10008f343b09SGreg Clayton                        "Load a shared library into the current process.",
10018f343b09SGreg Clayton                        "process load <filename> [<filename> ...]",
10028f343b09SGreg Clayton                        eFlagProcessMustBeLaunched | eFlagProcessMustBePaused)
10038f343b09SGreg Clayton     {
10048f343b09SGreg Clayton     }
10058f343b09SGreg Clayton 
10068f343b09SGreg Clayton     ~CommandObjectProcessLoad ()
10078f343b09SGreg Clayton     {
10088f343b09SGreg Clayton     }
10098f343b09SGreg Clayton 
10108f343b09SGreg Clayton     bool
10118f343b09SGreg Clayton     Execute (Args& command,
10128f343b09SGreg Clayton              CommandReturnObject &result)
10138f343b09SGreg Clayton     {
1014c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
10158f343b09SGreg Clayton         if (process == NULL)
10168f343b09SGreg Clayton         {
10178f343b09SGreg Clayton             result.AppendError ("must have a valid process in order to load a shared library");
10188f343b09SGreg Clayton             result.SetStatus (eReturnStatusFailed);
10198f343b09SGreg Clayton             return false;
10208f343b09SGreg Clayton         }
10218f343b09SGreg Clayton 
10228f343b09SGreg Clayton         const uint32_t argc = command.GetArgumentCount();
10238f343b09SGreg Clayton 
10248f343b09SGreg Clayton         for (uint32_t i=0; i<argc; ++i)
10258f343b09SGreg Clayton         {
10268f343b09SGreg Clayton             Error error;
10278f343b09SGreg Clayton             const char *image_path = command.GetArgumentAtIndex(i);
10288f343b09SGreg Clayton             FileSpec image_spec (image_path, false);
1029aa516843SGreg Clayton             process->GetTarget().GetPlatform()->ResolveRemotePath(image_spec, image_spec);
10308f343b09SGreg Clayton             uint32_t image_token = process->LoadImage(image_spec, error);
10318f343b09SGreg Clayton             if (image_token != LLDB_INVALID_IMAGE_TOKEN)
10328f343b09SGreg Clayton             {
10338f343b09SGreg Clayton                 result.AppendMessageWithFormat ("Loading \"%s\"...ok\nImage %u loaded.\n", image_path, image_token);
10348f343b09SGreg Clayton                 result.SetStatus (eReturnStatusSuccessFinishResult);
10358f343b09SGreg Clayton             }
10368f343b09SGreg Clayton             else
10378f343b09SGreg Clayton             {
10388f343b09SGreg Clayton                 result.AppendErrorWithFormat ("failed to load '%s': %s", image_path, error.AsCString());
10398f343b09SGreg Clayton                 result.SetStatus (eReturnStatusFailed);
10408f343b09SGreg Clayton             }
10418f343b09SGreg Clayton         }
10428f343b09SGreg Clayton         return result.Succeeded();
10438f343b09SGreg Clayton     }
10448f343b09SGreg Clayton };
10458f343b09SGreg Clayton 
10468f343b09SGreg Clayton 
10478f343b09SGreg Clayton //-------------------------------------------------------------------------
10488f343b09SGreg Clayton // CommandObjectProcessUnload
10498f343b09SGreg Clayton //-------------------------------------------------------------------------
1050bb9caf73SJim Ingham #pragma mark CommandObjectProcessUnload
10518f343b09SGreg Clayton 
10528f343b09SGreg Clayton class CommandObjectProcessUnload : public CommandObject
10538f343b09SGreg Clayton {
10548f343b09SGreg Clayton public:
10558f343b09SGreg Clayton 
10568f343b09SGreg Clayton     CommandObjectProcessUnload (CommandInterpreter &interpreter) :
10578f343b09SGreg Clayton         CommandObject (interpreter,
10588f343b09SGreg Clayton                        "process unload",
10598f343b09SGreg Clayton                        "Unload a shared library from the current process using the index returned by a previous call to \"process load\".",
10608f343b09SGreg Clayton                        "process unload <index>",
10618f343b09SGreg Clayton                        eFlagProcessMustBeLaunched | eFlagProcessMustBePaused)
10628f343b09SGreg Clayton     {
10638f343b09SGreg Clayton     }
10648f343b09SGreg Clayton 
10658f343b09SGreg Clayton     ~CommandObjectProcessUnload ()
10668f343b09SGreg Clayton     {
10678f343b09SGreg Clayton     }
10688f343b09SGreg Clayton 
10698f343b09SGreg Clayton     bool
10708f343b09SGreg Clayton     Execute (Args& command,
10718f343b09SGreg Clayton              CommandReturnObject &result)
10728f343b09SGreg Clayton     {
1073c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
10748f343b09SGreg Clayton         if (process == NULL)
10758f343b09SGreg Clayton         {
10768f343b09SGreg Clayton             result.AppendError ("must have a valid process in order to load a shared library");
10778f343b09SGreg Clayton             result.SetStatus (eReturnStatusFailed);
10788f343b09SGreg Clayton             return false;
10798f343b09SGreg Clayton         }
10808f343b09SGreg Clayton 
10818f343b09SGreg Clayton         const uint32_t argc = command.GetArgumentCount();
10828f343b09SGreg Clayton 
10838f343b09SGreg Clayton         for (uint32_t i=0; i<argc; ++i)
10848f343b09SGreg Clayton         {
10858f343b09SGreg Clayton             const char *image_token_cstr = command.GetArgumentAtIndex(i);
10868f343b09SGreg Clayton             uint32_t image_token = Args::StringToUInt32(image_token_cstr, LLDB_INVALID_IMAGE_TOKEN, 0);
10878f343b09SGreg Clayton             if (image_token == LLDB_INVALID_IMAGE_TOKEN)
10888f343b09SGreg Clayton             {
10898f343b09SGreg Clayton                 result.AppendErrorWithFormat ("invalid image index argument '%s'", image_token_cstr);
10908f343b09SGreg Clayton                 result.SetStatus (eReturnStatusFailed);
10918f343b09SGreg Clayton                 break;
10928f343b09SGreg Clayton             }
10938f343b09SGreg Clayton             else
10948f343b09SGreg Clayton             {
10958f343b09SGreg Clayton                 Error error (process->UnloadImage(image_token));
10968f343b09SGreg Clayton                 if (error.Success())
10978f343b09SGreg Clayton                 {
10988f343b09SGreg Clayton                     result.AppendMessageWithFormat ("Unloading shared library with index %u...ok\n", image_token);
10998f343b09SGreg Clayton                     result.SetStatus (eReturnStatusSuccessFinishResult);
11008f343b09SGreg Clayton                 }
11018f343b09SGreg Clayton                 else
11028f343b09SGreg Clayton                 {
11038f343b09SGreg Clayton                     result.AppendErrorWithFormat ("failed to unload image: %s", error.AsCString());
11048f343b09SGreg Clayton                     result.SetStatus (eReturnStatusFailed);
11058f343b09SGreg Clayton                     break;
11068f343b09SGreg Clayton                 }
11078f343b09SGreg Clayton             }
11088f343b09SGreg Clayton         }
11098f343b09SGreg Clayton         return result.Succeeded();
11108f343b09SGreg Clayton     }
11118f343b09SGreg Clayton };
11128f343b09SGreg Clayton 
11138f343b09SGreg Clayton //-------------------------------------------------------------------------
111430fdc8d8SChris Lattner // CommandObjectProcessSignal
111530fdc8d8SChris Lattner //-------------------------------------------------------------------------
1116bb9caf73SJim Ingham #pragma mark CommandObjectProcessSignal
111730fdc8d8SChris Lattner 
111830fdc8d8SChris Lattner class CommandObjectProcessSignal : public CommandObject
111930fdc8d8SChris Lattner {
112030fdc8d8SChris Lattner public:
112130fdc8d8SChris Lattner 
1122a7015092SGreg Clayton     CommandObjectProcessSignal (CommandInterpreter &interpreter) :
1123a7015092SGreg Clayton         CommandObject (interpreter,
1124a7015092SGreg Clayton                        "process signal",
1125e3d26315SCaroline Tice                        "Send a UNIX signal to the current process being debugged.",
1126405fe67fSCaroline Tice                        NULL)
112730fdc8d8SChris Lattner     {
1128405fe67fSCaroline Tice         CommandArgumentEntry arg;
1129405fe67fSCaroline Tice         CommandArgumentData signal_arg;
1130405fe67fSCaroline Tice 
1131405fe67fSCaroline Tice         // Define the first (and only) variant of this arg.
1132c0dbdfb6SCaroline Tice         signal_arg.arg_type = eArgTypeUnixSignal;
1133405fe67fSCaroline Tice         signal_arg.arg_repetition = eArgRepeatPlain;
1134405fe67fSCaroline Tice 
1135405fe67fSCaroline Tice         // There is only one variant this argument could be; put it into the argument entry.
1136405fe67fSCaroline Tice         arg.push_back (signal_arg);
1137405fe67fSCaroline Tice 
1138405fe67fSCaroline Tice         // Push the data for the first argument into the m_arguments vector.
1139405fe67fSCaroline Tice         m_arguments.push_back (arg);
114030fdc8d8SChris Lattner     }
114130fdc8d8SChris Lattner 
114230fdc8d8SChris Lattner     ~CommandObjectProcessSignal ()
114330fdc8d8SChris Lattner     {
114430fdc8d8SChris Lattner     }
114530fdc8d8SChris Lattner 
114630fdc8d8SChris Lattner     bool
1147a7015092SGreg Clayton     Execute (Args& command,
114830fdc8d8SChris Lattner              CommandReturnObject &result)
114930fdc8d8SChris Lattner     {
1150c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
115130fdc8d8SChris Lattner         if (process == NULL)
115230fdc8d8SChris Lattner         {
115330fdc8d8SChris Lattner             result.AppendError ("no process to signal");
115430fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
115530fdc8d8SChris Lattner             return false;
115630fdc8d8SChris Lattner         }
115730fdc8d8SChris Lattner 
115830fdc8d8SChris Lattner         if (command.GetArgumentCount() == 1)
115930fdc8d8SChris Lattner         {
1160237cd906SGreg Clayton             int signo = LLDB_INVALID_SIGNAL_NUMBER;
1161237cd906SGreg Clayton 
1162237cd906SGreg Clayton             const char *signal_name = command.GetArgumentAtIndex(0);
1163237cd906SGreg Clayton             if (::isxdigit (signal_name[0]))
1164237cd906SGreg Clayton                 signo = Args::StringToSInt32(signal_name, LLDB_INVALID_SIGNAL_NUMBER, 0);
1165237cd906SGreg Clayton             else
1166237cd906SGreg Clayton                 signo = process->GetUnixSignals().GetSignalNumberFromName (signal_name);
1167237cd906SGreg Clayton 
1168237cd906SGreg Clayton             if (signo == LLDB_INVALID_SIGNAL_NUMBER)
116930fdc8d8SChris Lattner             {
117030fdc8d8SChris Lattner                 result.AppendErrorWithFormat ("Invalid signal argument '%s'.\n", command.GetArgumentAtIndex(0));
117130fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusFailed);
117230fdc8d8SChris Lattner             }
117330fdc8d8SChris Lattner             else
117430fdc8d8SChris Lattner             {
117530fdc8d8SChris Lattner                 Error error (process->Signal (signo));
117630fdc8d8SChris Lattner                 if (error.Success())
117730fdc8d8SChris Lattner                 {
117830fdc8d8SChris Lattner                     result.SetStatus (eReturnStatusSuccessFinishResult);
117930fdc8d8SChris Lattner                 }
118030fdc8d8SChris Lattner                 else
118130fdc8d8SChris Lattner                 {
118230fdc8d8SChris Lattner                     result.AppendErrorWithFormat ("Failed to send signal %i: %s\n", signo, error.AsCString());
118330fdc8d8SChris Lattner                     result.SetStatus (eReturnStatusFailed);
118430fdc8d8SChris Lattner                 }
118530fdc8d8SChris Lattner             }
118630fdc8d8SChris Lattner         }
118730fdc8d8SChris Lattner         else
118830fdc8d8SChris Lattner         {
1189fd54b368SJason Molenda             result.AppendErrorWithFormat("'%s' takes exactly one signal number argument:\nUsage: %s\n", m_cmd_name.c_str(),
119030fdc8d8SChris Lattner                                         m_cmd_syntax.c_str());
119130fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
119230fdc8d8SChris Lattner         }
119330fdc8d8SChris Lattner         return result.Succeeded();
119430fdc8d8SChris Lattner     }
119530fdc8d8SChris Lattner };
119630fdc8d8SChris Lattner 
119730fdc8d8SChris Lattner 
119830fdc8d8SChris Lattner //-------------------------------------------------------------------------
119930fdc8d8SChris Lattner // CommandObjectProcessInterrupt
120030fdc8d8SChris Lattner //-------------------------------------------------------------------------
1201bb9caf73SJim Ingham #pragma mark CommandObjectProcessInterrupt
120230fdc8d8SChris Lattner 
120330fdc8d8SChris Lattner class CommandObjectProcessInterrupt : public CommandObject
120430fdc8d8SChris Lattner {
120530fdc8d8SChris Lattner public:
120630fdc8d8SChris Lattner 
120730fdc8d8SChris Lattner 
1208a7015092SGreg Clayton     CommandObjectProcessInterrupt (CommandInterpreter &interpreter) :
1209a7015092SGreg Clayton     CommandObject (interpreter,
1210a7015092SGreg Clayton                    "process interrupt",
1211e3d26315SCaroline Tice                    "Interrupt the current process being debugged.",
121230fdc8d8SChris Lattner                    "process interrupt",
121330fdc8d8SChris Lattner                    eFlagProcessMustBeLaunched)
121430fdc8d8SChris Lattner     {
121530fdc8d8SChris Lattner     }
121630fdc8d8SChris Lattner 
121730fdc8d8SChris Lattner     ~CommandObjectProcessInterrupt ()
121830fdc8d8SChris Lattner     {
121930fdc8d8SChris Lattner     }
122030fdc8d8SChris Lattner 
122130fdc8d8SChris Lattner     bool
1222a7015092SGreg Clayton     Execute (Args& command,
122330fdc8d8SChris Lattner              CommandReturnObject &result)
122430fdc8d8SChris Lattner     {
1225c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
122630fdc8d8SChris Lattner         if (process == NULL)
122730fdc8d8SChris Lattner         {
122830fdc8d8SChris Lattner             result.AppendError ("no process to halt");
122930fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
123030fdc8d8SChris Lattner             return false;
123130fdc8d8SChris Lattner         }
123230fdc8d8SChris Lattner 
123330fdc8d8SChris Lattner         if (command.GetArgumentCount() == 0)
123430fdc8d8SChris Lattner         {
123530fdc8d8SChris Lattner             Error error(process->Halt ());
123630fdc8d8SChris Lattner             if (error.Success())
123730fdc8d8SChris Lattner             {
123830fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusSuccessFinishResult);
123930fdc8d8SChris Lattner 
124030fdc8d8SChris Lattner                 // Maybe we should add a "SuspendThreadPlans so we
124130fdc8d8SChris Lattner                 // can halt, and keep in place all the current thread plans.
124230fdc8d8SChris Lattner                 process->GetThreadList().DiscardThreadPlans();
124330fdc8d8SChris Lattner             }
124430fdc8d8SChris Lattner             else
124530fdc8d8SChris Lattner             {
124630fdc8d8SChris Lattner                 result.AppendErrorWithFormat ("Failed to halt process: %s\n", error.AsCString());
124730fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusFailed);
124830fdc8d8SChris Lattner             }
124930fdc8d8SChris Lattner         }
125030fdc8d8SChris Lattner         else
125130fdc8d8SChris Lattner         {
1252fd54b368SJason Molenda             result.AppendErrorWithFormat("'%s' takes no arguments:\nUsage: %s\n",
125330fdc8d8SChris Lattner                                         m_cmd_name.c_str(),
125430fdc8d8SChris Lattner                                         m_cmd_syntax.c_str());
125530fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
125630fdc8d8SChris Lattner         }
125730fdc8d8SChris Lattner         return result.Succeeded();
125830fdc8d8SChris Lattner     }
125930fdc8d8SChris Lattner };
126030fdc8d8SChris Lattner 
126130fdc8d8SChris Lattner //-------------------------------------------------------------------------
126230fdc8d8SChris Lattner // CommandObjectProcessKill
126330fdc8d8SChris Lattner //-------------------------------------------------------------------------
1264bb9caf73SJim Ingham #pragma mark CommandObjectProcessKill
126530fdc8d8SChris Lattner 
126630fdc8d8SChris Lattner class CommandObjectProcessKill : public CommandObject
126730fdc8d8SChris Lattner {
126830fdc8d8SChris Lattner public:
126930fdc8d8SChris Lattner 
1270a7015092SGreg Clayton     CommandObjectProcessKill (CommandInterpreter &interpreter) :
1271a7015092SGreg Clayton     CommandObject (interpreter,
1272a7015092SGreg Clayton                    "process kill",
1273e3d26315SCaroline Tice                    "Terminate the current process being debugged.",
127430fdc8d8SChris Lattner                    "process kill",
127530fdc8d8SChris Lattner                    eFlagProcessMustBeLaunched)
127630fdc8d8SChris Lattner     {
127730fdc8d8SChris Lattner     }
127830fdc8d8SChris Lattner 
127930fdc8d8SChris Lattner     ~CommandObjectProcessKill ()
128030fdc8d8SChris Lattner     {
128130fdc8d8SChris Lattner     }
128230fdc8d8SChris Lattner 
128330fdc8d8SChris Lattner     bool
1284a7015092SGreg Clayton     Execute (Args& command,
128530fdc8d8SChris Lattner              CommandReturnObject &result)
128630fdc8d8SChris Lattner     {
1287c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
128830fdc8d8SChris Lattner         if (process == NULL)
128930fdc8d8SChris Lattner         {
129030fdc8d8SChris Lattner             result.AppendError ("no process to kill");
129130fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
129230fdc8d8SChris Lattner             return false;
129330fdc8d8SChris Lattner         }
129430fdc8d8SChris Lattner 
129530fdc8d8SChris Lattner         if (command.GetArgumentCount() == 0)
129630fdc8d8SChris Lattner         {
129730fdc8d8SChris Lattner             Error error (process->Destroy());
129830fdc8d8SChris Lattner             if (error.Success())
129930fdc8d8SChris Lattner             {
130030fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusSuccessFinishResult);
130130fdc8d8SChris Lattner             }
130230fdc8d8SChris Lattner             else
130330fdc8d8SChris Lattner             {
130430fdc8d8SChris Lattner                 result.AppendErrorWithFormat ("Failed to kill process: %s\n", error.AsCString());
130530fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusFailed);
130630fdc8d8SChris Lattner             }
130730fdc8d8SChris Lattner         }
130830fdc8d8SChris Lattner         else
130930fdc8d8SChris Lattner         {
1310fd54b368SJason Molenda             result.AppendErrorWithFormat("'%s' takes no arguments:\nUsage: %s\n",
131130fdc8d8SChris Lattner                                         m_cmd_name.c_str(),
131230fdc8d8SChris Lattner                                         m_cmd_syntax.c_str());
131330fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
131430fdc8d8SChris Lattner         }
131530fdc8d8SChris Lattner         return result.Succeeded();
131630fdc8d8SChris Lattner     }
131730fdc8d8SChris Lattner };
131830fdc8d8SChris Lattner 
131930fdc8d8SChris Lattner //-------------------------------------------------------------------------
13204b9bea87SJim Ingham // CommandObjectProcessStatus
13214b9bea87SJim Ingham //-------------------------------------------------------------------------
1322bb9caf73SJim Ingham #pragma mark CommandObjectProcessStatus
1323bb9caf73SJim Ingham 
13244b9bea87SJim Ingham class CommandObjectProcessStatus : public CommandObject
13254b9bea87SJim Ingham {
13264b9bea87SJim Ingham public:
1327a7015092SGreg Clayton     CommandObjectProcessStatus (CommandInterpreter &interpreter) :
1328a7015092SGreg Clayton     CommandObject (interpreter,
1329a7015092SGreg Clayton                    "process status",
1330e3d26315SCaroline Tice                    "Show the current status and location of executing process.",
1331e3d26315SCaroline Tice                    "process status",
13324b9bea87SJim Ingham                    0)
13334b9bea87SJim Ingham     {
13344b9bea87SJim Ingham     }
13354b9bea87SJim Ingham 
13364b9bea87SJim Ingham     ~CommandObjectProcessStatus()
13374b9bea87SJim Ingham     {
13384b9bea87SJim Ingham     }
13394b9bea87SJim Ingham 
13404b9bea87SJim Ingham 
13414b9bea87SJim Ingham     bool
13424b9bea87SJim Ingham     Execute
13434b9bea87SJim Ingham     (
13444b9bea87SJim Ingham         Args& command,
13454b9bea87SJim Ingham         CommandReturnObject &result
13464b9bea87SJim Ingham     )
13474b9bea87SJim Ingham     {
13487260f620SGreg Clayton         Stream &strm = result.GetOutputStream();
13494b9bea87SJim Ingham         result.SetStatus (eReturnStatusSuccessFinishNoResult);
13508b82f087SGreg Clayton         ExecutionContext exe_ctx(m_interpreter.GetExecutionContext());
1351c14ee32dSGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
1352c14ee32dSGreg Clayton         if (process)
13534b9bea87SJim Ingham         {
13547260f620SGreg Clayton             const bool only_threads_with_stop_reason = true;
13557260f620SGreg Clayton             const uint32_t start_frame = 0;
13567260f620SGreg Clayton             const uint32_t num_frames = 1;
13577260f620SGreg Clayton             const uint32_t num_frames_with_source = 1;
1358c14ee32dSGreg Clayton             process->GetStatus(strm);
1359c14ee32dSGreg Clayton             process->GetThreadStatus (strm,
13607260f620SGreg Clayton                                       only_threads_with_stop_reason,
13617260f620SGreg Clayton                                       start_frame,
13627260f620SGreg Clayton                                       num_frames,
13637260f620SGreg Clayton                                       num_frames_with_source);
13647260f620SGreg Clayton 
13654b9bea87SJim Ingham         }
13664b9bea87SJim Ingham         else
13674b9bea87SJim Ingham         {
13687260f620SGreg Clayton             result.AppendError ("No process.");
13694b9bea87SJim Ingham             result.SetStatus (eReturnStatusFailed);
13704b9bea87SJim Ingham         }
13714b9bea87SJim Ingham         return result.Succeeded();
13724b9bea87SJim Ingham     }
13734b9bea87SJim Ingham };
13744b9bea87SJim Ingham 
13754b9bea87SJim Ingham //-------------------------------------------------------------------------
137635731357SCaroline Tice // CommandObjectProcessHandle
137735731357SCaroline Tice //-------------------------------------------------------------------------
1378bb9caf73SJim Ingham #pragma mark CommandObjectProcessHandle
137935731357SCaroline Tice 
138035731357SCaroline Tice class CommandObjectProcessHandle : public CommandObject
138135731357SCaroline Tice {
138235731357SCaroline Tice public:
138335731357SCaroline Tice 
138435731357SCaroline Tice     class CommandOptions : public Options
138535731357SCaroline Tice     {
138635731357SCaroline Tice     public:
138735731357SCaroline Tice 
1388eb0103f2SGreg Clayton         CommandOptions (CommandInterpreter &interpreter) :
1389eb0103f2SGreg Clayton             Options (interpreter)
139035731357SCaroline Tice         {
1391f6b8b581SGreg Clayton             OptionParsingStarting ();
139235731357SCaroline Tice         }
139335731357SCaroline Tice 
139435731357SCaroline Tice         ~CommandOptions ()
139535731357SCaroline Tice         {
139635731357SCaroline Tice         }
139735731357SCaroline Tice 
139835731357SCaroline Tice         Error
1399f6b8b581SGreg Clayton         SetOptionValue (uint32_t option_idx, const char *option_arg)
140035731357SCaroline Tice         {
140135731357SCaroline Tice             Error error;
140235731357SCaroline Tice             char short_option = (char) m_getopt_table[option_idx].val;
140335731357SCaroline Tice 
140435731357SCaroline Tice             switch (short_option)
140535731357SCaroline Tice             {
140635731357SCaroline Tice                 case 's':
140735731357SCaroline Tice                     stop = option_arg;
140835731357SCaroline Tice                     break;
140935731357SCaroline Tice                 case 'n':
141035731357SCaroline Tice                     notify = option_arg;
141135731357SCaroline Tice                     break;
141235731357SCaroline Tice                 case 'p':
141335731357SCaroline Tice                     pass = option_arg;
141435731357SCaroline Tice                     break;
141535731357SCaroline Tice                 default:
141686edbf41SGreg Clayton                     error.SetErrorStringWithFormat("invalid short option character '%c'", short_option);
141735731357SCaroline Tice                     break;
141835731357SCaroline Tice             }
141935731357SCaroline Tice             return error;
142035731357SCaroline Tice         }
142135731357SCaroline Tice 
142235731357SCaroline Tice         void
1423f6b8b581SGreg Clayton         OptionParsingStarting ()
142435731357SCaroline Tice         {
142535731357SCaroline Tice             stop.clear();
142635731357SCaroline Tice             notify.clear();
142735731357SCaroline Tice             pass.clear();
142835731357SCaroline Tice         }
142935731357SCaroline Tice 
1430e0d378b3SGreg Clayton         const OptionDefinition*
143135731357SCaroline Tice         GetDefinitions ()
143235731357SCaroline Tice         {
143335731357SCaroline Tice             return g_option_table;
143435731357SCaroline Tice         }
143535731357SCaroline Tice 
143635731357SCaroline Tice         // Options table: Required for subclasses of Options.
143735731357SCaroline Tice 
1438e0d378b3SGreg Clayton         static OptionDefinition g_option_table[];
143935731357SCaroline Tice 
144035731357SCaroline Tice         // Instance variables to hold the values for command options.
144135731357SCaroline Tice 
144235731357SCaroline Tice         std::string stop;
144335731357SCaroline Tice         std::string notify;
144435731357SCaroline Tice         std::string pass;
144535731357SCaroline Tice     };
144635731357SCaroline Tice 
144735731357SCaroline Tice 
144835731357SCaroline Tice     CommandObjectProcessHandle (CommandInterpreter &interpreter) :
144935731357SCaroline Tice         CommandObject (interpreter,
145035731357SCaroline Tice                        "process handle",
145110ad7993SCaroline Tice                        "Show or update what the process and debugger should do with various signals received from the OS.",
1452eb0103f2SGreg Clayton                        NULL),
1453eb0103f2SGreg Clayton         m_options (interpreter)
145435731357SCaroline Tice     {
145510ad7993SCaroline Tice         SetHelpLong ("If no signals are specified, update them all.  If no update option is specified, list the current values.\n");
145635731357SCaroline Tice         CommandArgumentEntry arg;
1457c0dbdfb6SCaroline Tice         CommandArgumentData signal_arg;
145835731357SCaroline Tice 
1459c0dbdfb6SCaroline Tice         signal_arg.arg_type = eArgTypeUnixSignal;
1460c0dbdfb6SCaroline Tice         signal_arg.arg_repetition = eArgRepeatStar;
146135731357SCaroline Tice 
1462c0dbdfb6SCaroline Tice         arg.push_back (signal_arg);
146335731357SCaroline Tice 
146435731357SCaroline Tice         m_arguments.push_back (arg);
146535731357SCaroline Tice     }
146635731357SCaroline Tice 
146735731357SCaroline Tice     ~CommandObjectProcessHandle ()
146835731357SCaroline Tice     {
146935731357SCaroline Tice     }
147035731357SCaroline Tice 
147135731357SCaroline Tice     Options *
147235731357SCaroline Tice     GetOptions ()
147335731357SCaroline Tice     {
147435731357SCaroline Tice         return &m_options;
147535731357SCaroline Tice     }
147635731357SCaroline Tice 
147735731357SCaroline Tice     bool
147810ad7993SCaroline Tice     VerifyCommandOptionValue (const std::string &option, int &real_value)
147935731357SCaroline Tice     {
148035731357SCaroline Tice         bool okay = true;
148135731357SCaroline Tice 
148210ad7993SCaroline Tice         bool success = false;
148310ad7993SCaroline Tice         bool tmp_value = Args::StringToBoolean (option.c_str(), false, &success);
148410ad7993SCaroline Tice 
148510ad7993SCaroline Tice         if (success && tmp_value)
148610ad7993SCaroline Tice             real_value = 1;
148710ad7993SCaroline Tice         else if (success && !tmp_value)
148810ad7993SCaroline Tice             real_value = 0;
148935731357SCaroline Tice         else
149035731357SCaroline Tice         {
149135731357SCaroline Tice             // If the value isn't 'true' or 'false', it had better be 0 or 1.
149210ad7993SCaroline Tice             real_value = Args::StringToUInt32 (option.c_str(), 3);
149310ad7993SCaroline Tice             if (real_value != 0 && real_value != 1)
149435731357SCaroline Tice                 okay = false;
149535731357SCaroline Tice         }
149635731357SCaroline Tice 
149735731357SCaroline Tice         return okay;
149835731357SCaroline Tice     }
149935731357SCaroline Tice 
150010ad7993SCaroline Tice     void
150110ad7993SCaroline Tice     PrintSignalHeader (Stream &str)
150210ad7993SCaroline Tice     {
150310ad7993SCaroline Tice         str.Printf ("NAME        PASS   STOP   NOTIFY\n");
150410ad7993SCaroline Tice         str.Printf ("==========  =====  =====  ======\n");
150510ad7993SCaroline Tice     }
150610ad7993SCaroline Tice 
150710ad7993SCaroline Tice     void
150810ad7993SCaroline Tice     PrintSignal (Stream &str, int32_t signo, const char *sig_name, UnixSignals &signals)
150910ad7993SCaroline Tice     {
151010ad7993SCaroline Tice         bool stop;
151110ad7993SCaroline Tice         bool suppress;
151210ad7993SCaroline Tice         bool notify;
151310ad7993SCaroline Tice 
151410ad7993SCaroline Tice         str.Printf ("%-10s  ", sig_name);
151510ad7993SCaroline Tice         if (signals.GetSignalInfo (signo, suppress, stop, notify))
151610ad7993SCaroline Tice         {
151710ad7993SCaroline Tice             bool pass = !suppress;
151810ad7993SCaroline Tice             str.Printf ("%s  %s  %s",
151910ad7993SCaroline Tice                         (pass ? "true " : "false"),
152010ad7993SCaroline Tice                         (stop ? "true " : "false"),
152110ad7993SCaroline Tice                         (notify ? "true " : "false"));
152210ad7993SCaroline Tice         }
152310ad7993SCaroline Tice         str.Printf ("\n");
152410ad7993SCaroline Tice     }
152510ad7993SCaroline Tice 
152610ad7993SCaroline Tice     void
152710ad7993SCaroline Tice     PrintSignalInformation (Stream &str, Args &signal_args, int num_valid_signals, UnixSignals &signals)
152810ad7993SCaroline Tice     {
152910ad7993SCaroline Tice         PrintSignalHeader (str);
153010ad7993SCaroline Tice 
153110ad7993SCaroline Tice         if (num_valid_signals > 0)
153210ad7993SCaroline Tice         {
153310ad7993SCaroline Tice             size_t num_args = signal_args.GetArgumentCount();
153410ad7993SCaroline Tice             for (size_t i = 0; i < num_args; ++i)
153510ad7993SCaroline Tice             {
153610ad7993SCaroline Tice                 int32_t signo = signals.GetSignalNumberFromName (signal_args.GetArgumentAtIndex (i));
153710ad7993SCaroline Tice                 if (signo != LLDB_INVALID_SIGNAL_NUMBER)
153810ad7993SCaroline Tice                     PrintSignal (str, signo, signal_args.GetArgumentAtIndex (i), signals);
153910ad7993SCaroline Tice             }
154010ad7993SCaroline Tice         }
154110ad7993SCaroline Tice         else // Print info for ALL signals
154210ad7993SCaroline Tice         {
154310ad7993SCaroline Tice             int32_t signo = signals.GetFirstSignalNumber();
154410ad7993SCaroline Tice             while (signo != LLDB_INVALID_SIGNAL_NUMBER)
154510ad7993SCaroline Tice             {
154610ad7993SCaroline Tice                 PrintSignal (str, signo, signals.GetSignalAsCString (signo), signals);
154710ad7993SCaroline Tice                 signo = signals.GetNextSignalNumber (signo);
154810ad7993SCaroline Tice             }
154910ad7993SCaroline Tice         }
155010ad7993SCaroline Tice     }
155110ad7993SCaroline Tice 
155235731357SCaroline Tice     bool
155335731357SCaroline Tice     Execute (Args &signal_args, CommandReturnObject &result)
155435731357SCaroline Tice     {
155535731357SCaroline Tice         TargetSP target_sp = m_interpreter.GetDebugger().GetSelectedTarget();
155635731357SCaroline Tice 
155735731357SCaroline Tice         if (!target_sp)
155835731357SCaroline Tice         {
155935731357SCaroline Tice             result.AppendError ("No current target;"
156035731357SCaroline Tice                                 " cannot handle signals until you have a valid target and process.\n");
156135731357SCaroline Tice             result.SetStatus (eReturnStatusFailed);
156235731357SCaroline Tice             return false;
156335731357SCaroline Tice         }
156435731357SCaroline Tice 
156535731357SCaroline Tice         ProcessSP process_sp = target_sp->GetProcessSP();
156635731357SCaroline Tice 
156735731357SCaroline Tice         if (!process_sp)
156835731357SCaroline Tice         {
156935731357SCaroline Tice             result.AppendError ("No current process; cannot handle signals until you have a valid process.\n");
157035731357SCaroline Tice             result.SetStatus (eReturnStatusFailed);
157135731357SCaroline Tice             return false;
157235731357SCaroline Tice         }
157335731357SCaroline Tice 
157435731357SCaroline Tice         int stop_action = -1;   // -1 means leave the current setting alone
157535731357SCaroline Tice         int pass_action = -1;   // -1 means leave the current setting alone
157635731357SCaroline Tice         int notify_action = -1; // -1 means leave the current setting alone
157735731357SCaroline Tice 
157835731357SCaroline Tice         if (! m_options.stop.empty()
157910ad7993SCaroline Tice             && ! VerifyCommandOptionValue (m_options.stop, stop_action))
158035731357SCaroline Tice         {
158135731357SCaroline Tice             result.AppendError ("Invalid argument for command option --stop; must be true or false.\n");
158235731357SCaroline Tice             result.SetStatus (eReturnStatusFailed);
158335731357SCaroline Tice             return false;
158435731357SCaroline Tice         }
158535731357SCaroline Tice 
158635731357SCaroline Tice         if (! m_options.notify.empty()
158710ad7993SCaroline Tice             && ! VerifyCommandOptionValue (m_options.notify, notify_action))
158835731357SCaroline Tice         {
158935731357SCaroline Tice             result.AppendError ("Invalid argument for command option --notify; must be true or false.\n");
159035731357SCaroline Tice             result.SetStatus (eReturnStatusFailed);
159135731357SCaroline Tice             return false;
159235731357SCaroline Tice         }
159335731357SCaroline Tice 
159435731357SCaroline Tice         if (! m_options.pass.empty()
159510ad7993SCaroline Tice             && ! VerifyCommandOptionValue (m_options.pass, pass_action))
159635731357SCaroline Tice         {
159735731357SCaroline Tice             result.AppendError ("Invalid argument for command option --pass; must be true or false.\n");
159835731357SCaroline Tice             result.SetStatus (eReturnStatusFailed);
159935731357SCaroline Tice             return false;
160035731357SCaroline Tice         }
160135731357SCaroline Tice 
160235731357SCaroline Tice         size_t num_args = signal_args.GetArgumentCount();
160335731357SCaroline Tice         UnixSignals &signals = process_sp->GetUnixSignals();
160435731357SCaroline Tice         int num_signals_set = 0;
160535731357SCaroline Tice 
160610ad7993SCaroline Tice         if (num_args > 0)
160710ad7993SCaroline Tice         {
160835731357SCaroline Tice             for (size_t i = 0; i < num_args; ++i)
160935731357SCaroline Tice             {
161035731357SCaroline Tice                 int32_t signo = signals.GetSignalNumberFromName (signal_args.GetArgumentAtIndex (i));
161135731357SCaroline Tice                 if (signo != LLDB_INVALID_SIGNAL_NUMBER)
161235731357SCaroline Tice                 {
161310ad7993SCaroline Tice                     // Casting the actions as bools here should be okay, because VerifyCommandOptionValue guarantees
161435731357SCaroline Tice                     // the value is either 0 or 1.
161535731357SCaroline Tice                     if (stop_action != -1)
161635731357SCaroline Tice                         signals.SetShouldStop (signo, (bool) stop_action);
161735731357SCaroline Tice                     if (pass_action != -1)
161835731357SCaroline Tice                     {
161910ad7993SCaroline Tice                         bool suppress = ! ((bool) pass_action);
162010ad7993SCaroline Tice                         signals.SetShouldSuppress (signo, suppress);
162135731357SCaroline Tice                     }
162235731357SCaroline Tice                     if (notify_action != -1)
162335731357SCaroline Tice                         signals.SetShouldNotify (signo, (bool) notify_action);
162435731357SCaroline Tice                     ++num_signals_set;
162535731357SCaroline Tice                 }
162635731357SCaroline Tice                 else
162735731357SCaroline Tice                 {
162835731357SCaroline Tice                     result.AppendErrorWithFormat ("Invalid signal name '%s'\n", signal_args.GetArgumentAtIndex (i));
162935731357SCaroline Tice                 }
163035731357SCaroline Tice             }
163110ad7993SCaroline Tice         }
163210ad7993SCaroline Tice         else
163310ad7993SCaroline Tice         {
163410ad7993SCaroline Tice             // No signal specified, if any command options were specified, update ALL signals.
163510ad7993SCaroline Tice             if ((notify_action != -1) || (stop_action != -1) || (pass_action != -1))
163610ad7993SCaroline Tice             {
163710ad7993SCaroline Tice                 if (m_interpreter.Confirm ("Do you really want to update all the signals?", false))
163810ad7993SCaroline Tice                 {
163910ad7993SCaroline Tice                     int32_t signo = signals.GetFirstSignalNumber();
164010ad7993SCaroline Tice                     while (signo != LLDB_INVALID_SIGNAL_NUMBER)
164110ad7993SCaroline Tice                     {
164210ad7993SCaroline Tice                         if (notify_action != -1)
164310ad7993SCaroline Tice                             signals.SetShouldNotify (signo, (bool) notify_action);
164410ad7993SCaroline Tice                         if (stop_action != -1)
164510ad7993SCaroline Tice                             signals.SetShouldStop (signo, (bool) stop_action);
164610ad7993SCaroline Tice                         if (pass_action != -1)
164710ad7993SCaroline Tice                         {
164810ad7993SCaroline Tice                             bool suppress = ! ((bool) pass_action);
164910ad7993SCaroline Tice                             signals.SetShouldSuppress (signo, suppress);
165010ad7993SCaroline Tice                         }
165110ad7993SCaroline Tice                         signo = signals.GetNextSignalNumber (signo);
165210ad7993SCaroline Tice                     }
165310ad7993SCaroline Tice                 }
165410ad7993SCaroline Tice             }
165510ad7993SCaroline Tice         }
165610ad7993SCaroline Tice 
165710ad7993SCaroline Tice         PrintSignalInformation (result.GetOutputStream(), signal_args, num_signals_set, signals);
165835731357SCaroline Tice 
165935731357SCaroline Tice         if (num_signals_set > 0)
166035731357SCaroline Tice             result.SetStatus (eReturnStatusSuccessFinishNoResult);
166135731357SCaroline Tice         else
166235731357SCaroline Tice             result.SetStatus (eReturnStatusFailed);
166335731357SCaroline Tice 
166435731357SCaroline Tice         return result.Succeeded();
166535731357SCaroline Tice     }
166635731357SCaroline Tice 
166735731357SCaroline Tice protected:
166835731357SCaroline Tice 
166935731357SCaroline Tice     CommandOptions m_options;
167035731357SCaroline Tice };
167135731357SCaroline Tice 
1672e0d378b3SGreg Clayton OptionDefinition
167335731357SCaroline Tice CommandObjectProcessHandle::CommandOptions::g_option_table[] =
167435731357SCaroline Tice {
167535731357SCaroline 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." },
167635731357SCaroline 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." },
167735731357SCaroline Tice { LLDB_OPT_SET_1, false, "pass",  'p', required_argument, NULL, 0, eArgTypeBoolean, "Whether or not the signal should be passed to the process." },
167835731357SCaroline Tice { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL }
167935731357SCaroline Tice };
168035731357SCaroline Tice 
168135731357SCaroline Tice //-------------------------------------------------------------------------
168230fdc8d8SChris Lattner // CommandObjectMultiwordProcess
168330fdc8d8SChris Lattner //-------------------------------------------------------------------------
168430fdc8d8SChris Lattner 
16856611103cSGreg Clayton CommandObjectMultiwordProcess::CommandObjectMultiwordProcess (CommandInterpreter &interpreter) :
1686a7015092SGreg Clayton     CommandObjectMultiword (interpreter,
1687a7015092SGreg Clayton                             "process",
168830fdc8d8SChris Lattner                             "A set of commands for operating on a process.",
168930fdc8d8SChris Lattner                             "process <subcommand> [<subcommand-options>]")
169030fdc8d8SChris Lattner {
1691a7015092SGreg Clayton     LoadSubCommand ("attach",      CommandObjectSP (new CommandObjectProcessAttach    (interpreter)));
1692a7015092SGreg Clayton     LoadSubCommand ("launch",      CommandObjectSP (new CommandObjectProcessLaunch    (interpreter)));
1693a7015092SGreg Clayton     LoadSubCommand ("continue",    CommandObjectSP (new CommandObjectProcessContinue  (interpreter)));
1694b766a73dSGreg Clayton     LoadSubCommand ("connect",     CommandObjectSP (new CommandObjectProcessConnect   (interpreter)));
1695a7015092SGreg Clayton     LoadSubCommand ("detach",      CommandObjectSP (new CommandObjectProcessDetach    (interpreter)));
16968f343b09SGreg Clayton     LoadSubCommand ("load",        CommandObjectSP (new CommandObjectProcessLoad      (interpreter)));
16978f343b09SGreg Clayton     LoadSubCommand ("unload",      CommandObjectSP (new CommandObjectProcessUnload    (interpreter)));
1698a7015092SGreg Clayton     LoadSubCommand ("signal",      CommandObjectSP (new CommandObjectProcessSignal    (interpreter)));
169935731357SCaroline Tice     LoadSubCommand ("handle",      CommandObjectSP (new CommandObjectProcessHandle    (interpreter)));
1700a7015092SGreg Clayton     LoadSubCommand ("status",      CommandObjectSP (new CommandObjectProcessStatus    (interpreter)));
1701a7015092SGreg Clayton     LoadSubCommand ("interrupt",   CommandObjectSP (new CommandObjectProcessInterrupt (interpreter)));
1702a7015092SGreg Clayton     LoadSubCommand ("kill",        CommandObjectSP (new CommandObjectProcessKill      (interpreter)));
170330fdc8d8SChris Lattner }
170430fdc8d8SChris Lattner 
170530fdc8d8SChris Lattner CommandObjectMultiwordProcess::~CommandObjectMultiwordProcess ()
170630fdc8d8SChris Lattner {
170730fdc8d8SChris Lattner }
170830fdc8d8SChris Lattner 
1709