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         {
223162b597cSGreg Clayton             // Save the arguments for subsequent runs in the current target.
224162b597cSGreg Clayton             target->SetRunArguments (launch_args);
225162b597cSGreg Clayton 
226982c9762SGreg Clayton             m_options.launch_info.GetArguments().AppendArguments (launch_args);
227982c9762SGreg Clayton         }
2281d885966SGreg Clayton 
229144f3a9cSGreg Clayton         if (target->GetDisableASLR())
230144f3a9cSGreg Clayton             m_options.launch_info.GetFlags().Set (eLaunchFlagDisableASLR);
231144f3a9cSGreg Clayton 
232144f3a9cSGreg Clayton         if (target->GetDisableSTDIO())
233144f3a9cSGreg Clayton             m_options.launch_info.GetFlags().Set (eLaunchFlagDisableSTDIO);
234144f3a9cSGreg Clayton 
235144f3a9cSGreg Clayton         m_options.launch_info.GetFlags().Set (eLaunchFlagDebug);
236144f3a9cSGreg Clayton 
237144f3a9cSGreg Clayton         Args environment;
238144f3a9cSGreg Clayton         target->GetEnvironmentAsArgs (environment);
239144f3a9cSGreg Clayton         if (environment.GetArgumentCount() > 0)
240144f3a9cSGreg Clayton             m_options.launch_info.GetEnvironmentEntries ().AppendArguments (environment);
241144f3a9cSGreg Clayton 
242ee95ed50SGreg Clayton         // Finalize the file actions, and if none were given, default to opening
243ee95ed50SGreg Clayton         // up a pseudo terminal
244ee95ed50SGreg Clayton         const bool default_to_use_pty = true;
245ee95ed50SGreg Clayton         m_options.launch_info.FinalizeFileActions (target, default_to_use_pty);
2461d885966SGreg Clayton 
2471d885966SGreg Clayton         if (state == eStateConnected)
2481d885966SGreg Clayton         {
2491d885966SGreg Clayton             if (m_options.launch_info.GetFlags().Test (eLaunchFlagLaunchInTTY))
2501d885966SGreg Clayton             {
2511d885966SGreg Clayton                 result.AppendWarning("can't launch in tty when launching through a remote connection");
2521d885966SGreg Clayton                 m_options.launch_info.GetFlags().Clear (eLaunchFlagLaunchInTTY);
2531d885966SGreg Clayton             }
2541d885966SGreg Clayton         }
255982c9762SGreg Clayton         else
256982c9762SGreg Clayton         {
257144f3a9cSGreg Clayton             if (!m_options.launch_info.GetArchitecture().IsValid())
258c9ed478aSGreg Clayton                 m_options.launch_info.GetArchitecture() = target->GetArchitecture();
259c9ed478aSGreg Clayton 
260*c982b3d6SGreg Clayton             PlatformSP platform_sp (target->GetPlatform());
261*c982b3d6SGreg Clayton 
262*c982b3d6SGreg Clayton             if (platform_sp && platform_sp->CanDebugProcess ())
263*c982b3d6SGreg Clayton             {
2641d885966SGreg Clayton                 process = target->GetPlatform()->DebugProcess (m_options.launch_info,
2651d885966SGreg Clayton                                                                debugger,
2661d885966SGreg Clayton                                                                target,
2671d885966SGreg Clayton                                                                debugger.GetListener(),
2681d885966SGreg Clayton                                                                error).get();
269*c982b3d6SGreg Clayton             }
270*c982b3d6SGreg Clayton             else
271*c982b3d6SGreg Clayton             {
272*c982b3d6SGreg Clayton                 const char *plugin_name = m_options.launch_info.GetProcessPluginName();
273*c982b3d6SGreg Clayton                 process = target->CreateProcess (debugger.GetListener(), plugin_name).get();
274*c982b3d6SGreg Clayton                 if (process)
275*c982b3d6SGreg Clayton                     error = process->Launch (m_options.launch_info);
276*c982b3d6SGreg Clayton             }
2771d885966SGreg Clayton 
2781d885966SGreg Clayton             if (process == NULL)
2791d885966SGreg Clayton             {
280144f3a9cSGreg Clayton                 result.SetError (error, "failed to launch or debug process");
2811d885966SGreg Clayton                 return false;
2821d885966SGreg Clayton             }
2831d885966SGreg Clayton         }
28430fdc8d8SChris Lattner 
28530fdc8d8SChris Lattner         if (error.Success())
28630fdc8d8SChris Lattner         {
28764195a2cSGreg Clayton             const char *archname = exe_module->GetArchitecture().GetArchitectureName();
28819388cfcSGreg Clayton 
28981c22f61SGreg Clayton             result.AppendMessageWithFormat ("Process %llu launched: '%s' (%s)\n", process->GetID(), filename, archname);
29005faeb71SGreg Clayton             result.SetDidChangeProcessState (true);
291982c9762SGreg Clayton             if (m_options.launch_info.GetFlags().Test(eLaunchFlagStopAtEntry) == false)
29230fdc8d8SChris Lattner             {
29305faeb71SGreg Clayton                 result.SetStatus (eReturnStatusSuccessContinuingNoResult);
29430fdc8d8SChris Lattner                 StateType state = process->WaitForProcessToStop (NULL);
29530fdc8d8SChris Lattner 
29630fdc8d8SChris Lattner                 if (state == eStateStopped)
29730fdc8d8SChris Lattner                 {
29805faeb71SGreg Clayton                     error = process->Resume();
29905faeb71SGreg Clayton                     if (error.Success())
30005faeb71SGreg Clayton                     {
30105faeb71SGreg Clayton                         bool synchronous_execution = m_interpreter.GetSynchronous ();
30230fdc8d8SChris Lattner                         if (synchronous_execution)
30330fdc8d8SChris Lattner                         {
30405faeb71SGreg Clayton                             state = process->WaitForProcessToStop (NULL);
3052637f825SGreg Clayton                             const bool must_be_alive = true;
3062637f825SGreg Clayton                             if (!StateIsStoppedState(state, must_be_alive))
307514487e8SGreg Clayton                             {
308144f3a9cSGreg Clayton                                 result.AppendErrorWithFormat ("process isn't stopped: %s", StateAsCString(state));
309514487e8SGreg Clayton                             }
31030fdc8d8SChris Lattner                             result.SetDidChangeProcessState (true);
31105faeb71SGreg Clayton                             result.SetStatus (eReturnStatusSuccessFinishResult);
31205faeb71SGreg Clayton                         }
31305faeb71SGreg Clayton                         else
31405faeb71SGreg Clayton                         {
31505faeb71SGreg Clayton                             result.SetStatus (eReturnStatusSuccessContinuingNoResult);
31605faeb71SGreg Clayton                         }
31705faeb71SGreg Clayton                     }
318514487e8SGreg Clayton                     else
319514487e8SGreg Clayton                     {
320144f3a9cSGreg Clayton                         result.AppendErrorWithFormat ("process resume at entry point failed: %s", error.AsCString());
321514487e8SGreg Clayton                         result.SetStatus (eReturnStatusFailed);
32230fdc8d8SChris Lattner                     }
32330fdc8d8SChris Lattner                 }
324514487e8SGreg Clayton                 else
325514487e8SGreg Clayton                 {
326144f3a9cSGreg Clayton                     result.AppendErrorWithFormat ("initial process state wasn't stopped: %s", StateAsCString(state));
327514487e8SGreg Clayton                     result.SetStatus (eReturnStatusFailed);
328514487e8SGreg Clayton                 }
329514487e8SGreg Clayton             }
330514487e8SGreg Clayton         }
331514487e8SGreg Clayton         else
332514487e8SGreg Clayton         {
333197bacffSGreg Clayton             result.AppendErrorWithFormat ("process launch failed: %s", error.AsCString());
334514487e8SGreg Clayton             result.SetStatus (eReturnStatusFailed);
33530fdc8d8SChris Lattner         }
33630fdc8d8SChris Lattner 
33730fdc8d8SChris Lattner         return result.Succeeded();
33830fdc8d8SChris Lattner     }
33930fdc8d8SChris Lattner 
340ebc09c36SJim Ingham     virtual const char *GetRepeatCommand (Args &current_command_args, uint32_t index)
341ebc09c36SJim Ingham     {
342ebc09c36SJim Ingham         // No repeat for "process launch"...
343ebc09c36SJim Ingham         return "";
344ebc09c36SJim Ingham     }
345ebc09c36SJim Ingham 
34630fdc8d8SChris Lattner protected:
347982c9762SGreg Clayton     ProcessLaunchCommandOptions m_options;
34830fdc8d8SChris Lattner };
34930fdc8d8SChris Lattner 
35030fdc8d8SChris Lattner 
351982c9762SGreg Clayton //#define SET1 LLDB_OPT_SET_1
352982c9762SGreg Clayton //#define SET2 LLDB_OPT_SET_2
353982c9762SGreg Clayton //#define SET3 LLDB_OPT_SET_3
354982c9762SGreg Clayton //
355982c9762SGreg Clayton //OptionDefinition
356982c9762SGreg Clayton //CommandObjectProcessLaunch::CommandOptions::g_option_table[] =
357982c9762SGreg Clayton //{
358982c9762SGreg 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."},
359982c9762SGreg Clayton //{ SET1              , false, "stdin",         'i', required_argument, NULL, 0, eArgTypePath,    "Redirect stdin for the process to <path>."},
360982c9762SGreg Clayton //{ SET1              , false, "stdout",        'o', required_argument, NULL, 0, eArgTypePath,    "Redirect stdout for the process to <path>."},
361982c9762SGreg Clayton //{ SET1              , false, "stderr",        'e', required_argument, NULL, 0, eArgTypePath,    "Redirect stderr for the process to <path>."},
362982c9762SGreg Clayton //{ SET1 | SET2 | SET3, false, "plugin",        'p', required_argument, NULL, 0, eArgTypePlugin,  "Name of the process plugin you want to use."},
363982c9762SGreg 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."},
364982c9762SGreg Clayton //{               SET3, false, "no-stdio",      'n', no_argument,       NULL, 0, eArgTypeNone,    "Do not set up for terminal I/O to go to running process."},
365982c9762SGreg Clayton //{ SET1 | SET2 | SET3, false, "working-dir",   'w', required_argument, NULL, 0, eArgTypePath,    "Set the current working directory to <path> when running the inferior."},
366982c9762SGreg Clayton //{ 0,                  false, NULL,             0,  0,                 NULL, 0, eArgTypeNone,    NULL }
367982c9762SGreg Clayton //};
368982c9762SGreg Clayton //
369982c9762SGreg Clayton //#undef SET1
370982c9762SGreg Clayton //#undef SET2
371982c9762SGreg Clayton //#undef SET3
37230fdc8d8SChris Lattner 
37330fdc8d8SChris Lattner //-------------------------------------------------------------------------
37430fdc8d8SChris Lattner // CommandObjectProcessAttach
37530fdc8d8SChris Lattner //-------------------------------------------------------------------------
376bb9caf73SJim Ingham #pragma mark CommandObjectProcessAttach
37730fdc8d8SChris Lattner class CommandObjectProcessAttach : public CommandObject
37830fdc8d8SChris Lattner {
37930fdc8d8SChris Lattner public:
38030fdc8d8SChris Lattner 
38130fdc8d8SChris Lattner     class CommandOptions : public Options
38230fdc8d8SChris Lattner     {
38330fdc8d8SChris Lattner     public:
38430fdc8d8SChris Lattner 
385eb0103f2SGreg Clayton         CommandOptions (CommandInterpreter &interpreter) :
386eb0103f2SGreg Clayton             Options(interpreter)
38730fdc8d8SChris Lattner         {
388f6b8b581SGreg Clayton             // Keep default values of all options in one place: OptionParsingStarting ()
389f6b8b581SGreg Clayton             OptionParsingStarting ();
39030fdc8d8SChris Lattner         }
39130fdc8d8SChris Lattner 
39230fdc8d8SChris Lattner         ~CommandOptions ()
39330fdc8d8SChris Lattner         {
39430fdc8d8SChris Lattner         }
39530fdc8d8SChris Lattner 
39630fdc8d8SChris Lattner         Error
397f6b8b581SGreg Clayton         SetOptionValue (uint32_t option_idx, const char *option_arg)
39830fdc8d8SChris Lattner         {
39930fdc8d8SChris Lattner             Error error;
40030fdc8d8SChris Lattner             char short_option = (char) m_getopt_table[option_idx].val;
40130fdc8d8SChris Lattner             bool success = false;
40230fdc8d8SChris Lattner             switch (short_option)
40330fdc8d8SChris Lattner             {
40430fdc8d8SChris Lattner                 case 'p':
405144f3a9cSGreg Clayton                     {
406144f3a9cSGreg Clayton                         lldb::pid_t pid = Args::StringToUInt32 (option_arg, LLDB_INVALID_PROCESS_ID, 0, &success);
40730fdc8d8SChris Lattner                         if (!success || pid == LLDB_INVALID_PROCESS_ID)
40830fdc8d8SChris Lattner                         {
40986edbf41SGreg Clayton                             error.SetErrorStringWithFormat("invalid process ID '%s'", option_arg);
41030fdc8d8SChris Lattner                         }
411144f3a9cSGreg Clayton                         else
412144f3a9cSGreg Clayton                         {
413144f3a9cSGreg Clayton                             attach_info.SetProcessID (pid);
414144f3a9cSGreg Clayton                         }
415144f3a9cSGreg Clayton                     }
41630fdc8d8SChris Lattner                     break;
41730fdc8d8SChris Lattner 
41830fdc8d8SChris Lattner                 case 'P':
419144f3a9cSGreg Clayton                     attach_info.SetProcessPluginName (option_arg);
42030fdc8d8SChris Lattner                     break;
42130fdc8d8SChris Lattner 
42230fdc8d8SChris Lattner                 case 'n':
423144f3a9cSGreg Clayton                     attach_info.GetExecutableFile().SetFile(option_arg, false);
42430fdc8d8SChris Lattner                     break;
42530fdc8d8SChris Lattner 
42630fdc8d8SChris Lattner                 case 'w':
427144f3a9cSGreg Clayton                     attach_info.SetWaitForLaunch(true);
42830fdc8d8SChris Lattner                     break;
42930fdc8d8SChris Lattner 
43030fdc8d8SChris Lattner                 default:
43186edbf41SGreg Clayton                     error.SetErrorStringWithFormat("invalid short option character '%c'", short_option);
43230fdc8d8SChris Lattner                     break;
43330fdc8d8SChris Lattner             }
43430fdc8d8SChris Lattner             return error;
43530fdc8d8SChris Lattner         }
43630fdc8d8SChris Lattner 
43730fdc8d8SChris Lattner         void
438f6b8b581SGreg Clayton         OptionParsingStarting ()
43930fdc8d8SChris Lattner         {
440144f3a9cSGreg Clayton             attach_info.Clear();
44130fdc8d8SChris Lattner         }
44230fdc8d8SChris Lattner 
443e0d378b3SGreg Clayton         const OptionDefinition*
44430fdc8d8SChris Lattner         GetDefinitions ()
44530fdc8d8SChris Lattner         {
44630fdc8d8SChris Lattner             return g_option_table;
44730fdc8d8SChris Lattner         }
44830fdc8d8SChris Lattner 
4495aee162fSJim Ingham         virtual bool
450eb0103f2SGreg Clayton         HandleOptionArgumentCompletion (Args &input,
4515aee162fSJim Ingham                                         int cursor_index,
4525aee162fSJim Ingham                                         int char_pos,
4535aee162fSJim Ingham                                         OptionElementVector &opt_element_vector,
4545aee162fSJim Ingham                                         int opt_element_index,
4555aee162fSJim Ingham                                         int match_start_point,
4565aee162fSJim Ingham                                         int max_return_elements,
4575aee162fSJim Ingham                                         bool &word_complete,
4585aee162fSJim Ingham                                         StringList &matches)
4595aee162fSJim Ingham         {
4605aee162fSJim Ingham             int opt_arg_pos = opt_element_vector[opt_element_index].opt_arg_pos;
4615aee162fSJim Ingham             int opt_defs_index = opt_element_vector[opt_element_index].opt_defs_index;
4625aee162fSJim Ingham 
4635aee162fSJim Ingham             // We are only completing the name option for now...
4645aee162fSJim Ingham 
465e0d378b3SGreg Clayton             const OptionDefinition *opt_defs = GetDefinitions();
4665aee162fSJim Ingham             if (opt_defs[opt_defs_index].short_option == 'n')
4675aee162fSJim Ingham             {
4685aee162fSJim Ingham                 // Are we in the name?
4695aee162fSJim Ingham 
4705aee162fSJim Ingham                 // Look to see if there is a -P argument provided, and if so use that plugin, otherwise
4715aee162fSJim Ingham                 // use the default plugin.
4725aee162fSJim Ingham 
4735aee162fSJim Ingham                 const char *partial_name = NULL;
4745aee162fSJim Ingham                 partial_name = input.GetArgumentAtIndex(opt_arg_pos);
4755aee162fSJim Ingham 
4768b82f087SGreg Clayton                 PlatformSP platform_sp (m_interpreter.GetPlatform (true));
477e996fd30SGreg Clayton                 if (platform_sp)
4785aee162fSJim Ingham                 {
4798b82f087SGreg Clayton                     ProcessInstanceInfoList process_infos;
4808b82f087SGreg Clayton                     ProcessInstanceInfoMatch match_info;
48132e0a750SGreg Clayton                     if (partial_name)
48232e0a750SGreg Clayton                     {
483144f3a9cSGreg Clayton                         match_info.GetProcessInfo().GetExecutableFile().SetFile(partial_name, false);
48432e0a750SGreg Clayton                         match_info.SetNameMatchType(eNameMatchStartsWith);
48532e0a750SGreg Clayton                     }
48632e0a750SGreg Clayton                     platform_sp->FindProcesses (match_info, process_infos);
487e996fd30SGreg Clayton                     const uint32_t num_matches = process_infos.GetSize();
488e996fd30SGreg Clayton                     if (num_matches > 0)
489e996fd30SGreg Clayton                     {
490e996fd30SGreg Clayton                         for (uint32_t i=0; i<num_matches; ++i)
491e996fd30SGreg Clayton                         {
492e996fd30SGreg Clayton                             matches.AppendString (process_infos.GetProcessNameAtIndex(i),
493e996fd30SGreg Clayton                                                   process_infos.GetProcessNameLengthAtIndex(i));
4945aee162fSJim Ingham                         }
4955aee162fSJim Ingham                     }
4965aee162fSJim Ingham                 }
4975aee162fSJim Ingham             }
4985aee162fSJim Ingham 
4995aee162fSJim Ingham             return false;
5005aee162fSJim Ingham         }
5015aee162fSJim Ingham 
50230fdc8d8SChris Lattner         // Options table: Required for subclasses of Options.
50330fdc8d8SChris Lattner 
504e0d378b3SGreg Clayton         static OptionDefinition g_option_table[];
50530fdc8d8SChris Lattner 
50630fdc8d8SChris Lattner         // Instance variables to hold the values for command options.
50730fdc8d8SChris Lattner 
508144f3a9cSGreg Clayton         ProcessAttachInfo attach_info;
50930fdc8d8SChris Lattner     };
51030fdc8d8SChris Lattner 
511a7015092SGreg Clayton     CommandObjectProcessAttach (CommandInterpreter &interpreter) :
512a7015092SGreg Clayton         CommandObject (interpreter,
513a7015092SGreg Clayton                        "process attach",
514e3d26315SCaroline Tice                        "Attach to a process.",
515eb0103f2SGreg Clayton                        "process attach <cmd-options>"),
516eb0103f2SGreg Clayton         m_options (interpreter)
5175aee162fSJim Ingham     {
5185aee162fSJim Ingham     }
5195aee162fSJim Ingham 
5205aee162fSJim Ingham     ~CommandObjectProcessAttach ()
5215aee162fSJim Ingham     {
5225aee162fSJim Ingham     }
5235aee162fSJim Ingham 
5245aee162fSJim Ingham     bool
525a7015092SGreg Clayton     Execute (Args& command,
5265aee162fSJim Ingham              CommandReturnObject &result)
5275aee162fSJim Ingham     {
528a7015092SGreg Clayton         Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get();
52931412642SJim Ingham         // N.B. The attach should be synchronous.  It doesn't help much to get the prompt back between initiating the attach
53031412642SJim Ingham         // and the target actually stopping.  So even if the interpreter is set to be asynchronous, we wait for the stop
53131412642SJim Ingham         // ourselves here.
5325aee162fSJim Ingham 
533c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
53471337622SGreg Clayton         StateType state = eStateInvalid;
5355aee162fSJim Ingham         if (process)
5365aee162fSJim Ingham         {
53771337622SGreg Clayton             state = process->GetState();
53871337622SGreg Clayton             if (process->IsAlive() && state != eStateConnected)
5395aee162fSJim Ingham             {
54081c22f61SGreg Clayton                 result.AppendErrorWithFormat ("Process %llu is currently being debugged, kill the process before attaching.\n",
5415aee162fSJim Ingham                                               process->GetID());
5425aee162fSJim Ingham                 result.SetStatus (eReturnStatusFailed);
5435aee162fSJim Ingham                 return false;
5445aee162fSJim Ingham             }
5455aee162fSJim Ingham         }
5465aee162fSJim Ingham 
5475aee162fSJim Ingham         if (target == NULL)
5485aee162fSJim Ingham         {
5495aee162fSJim Ingham             // If there isn't a current target create one.
5505aee162fSJim Ingham             TargetSP new_target_sp;
5515aee162fSJim Ingham             FileSpec emptyFileSpec;
5525aee162fSJim Ingham             Error error;
5535aee162fSJim Ingham 
554a7015092SGreg Clayton             error = m_interpreter.GetDebugger().GetTargetList().CreateTarget (m_interpreter.GetDebugger(),
5555aee162fSJim Ingham                                                                               emptyFileSpec,
556cac9c5f9SGreg Clayton                                                                               NULL,
5575aee162fSJim Ingham                                                                               false,
558cac9c5f9SGreg Clayton                                                                               NULL, // No platform options
5595aee162fSJim Ingham                                                                               new_target_sp);
5605aee162fSJim Ingham             target = new_target_sp.get();
5615aee162fSJim Ingham             if (target == NULL || error.Fail())
5625aee162fSJim Ingham             {
563b766a73dSGreg Clayton                 result.AppendError(error.AsCString("Error creating target"));
5645aee162fSJim Ingham                 return false;
5655aee162fSJim Ingham             }
566a7015092SGreg Clayton             m_interpreter.GetDebugger().GetTargetList().SetSelectedTarget(target);
5675aee162fSJim Ingham         }
5685aee162fSJim Ingham 
5695aee162fSJim Ingham         // Record the old executable module, we want to issue a warning if the process of attaching changed the
5705aee162fSJim Ingham         // current executable (like somebody said "file foo" then attached to a PID whose executable was bar.)
5715aee162fSJim Ingham 
5725aee162fSJim Ingham         ModuleSP old_exec_module_sp = target->GetExecutableModule();
5735aee162fSJim Ingham         ArchSpec old_arch_spec = target->GetArchitecture();
5745aee162fSJim Ingham 
5755aee162fSJim Ingham         if (command.GetArgumentCount())
5765aee162fSJim Ingham         {
577fd54b368SJason Molenda             result.AppendErrorWithFormat("Invalid arguments for '%s'.\nUsage: %s\n", m_cmd_name.c_str(), m_cmd_syntax.c_str());
5785aee162fSJim Ingham             result.SetStatus (eReturnStatusFailed);
5795aee162fSJim Ingham         }
5805aee162fSJim Ingham         else
5815aee162fSJim Ingham         {
58271337622SGreg Clayton             if (state != eStateConnected)
58371337622SGreg Clayton             {
584144f3a9cSGreg Clayton                 const char *plugin_name = m_options.attach_info.GetProcessPluginName();
585a7015092SGreg Clayton                 process = target->CreateProcess (m_interpreter.GetDebugger().GetListener(), plugin_name).get();
58671337622SGreg Clayton             }
5875aee162fSJim Ingham 
5885aee162fSJim Ingham             if (process)
5895aee162fSJim Ingham             {
5905aee162fSJim Ingham                 Error error;
591144f3a9cSGreg Clayton                 // If no process info was specified, then use the target executable
592144f3a9cSGreg Clayton                 // name as the process to attach to by default
593144f3a9cSGreg Clayton                 if (!m_options.attach_info.ProcessInfoSpecified ())
5945aee162fSJim Ingham                 {
5953a0b9cdfSJim Ingham                     if (old_exec_module_sp)
596144f3a9cSGreg Clayton                         m_options.attach_info.GetExecutableFile().GetFilename() = old_exec_module_sp->GetFileSpec().GetFilename();
597144f3a9cSGreg Clayton 
598144f3a9cSGreg Clayton                     if (!m_options.attach_info.ProcessInfoSpecified ())
5993a0b9cdfSJim Ingham                     {
600144f3a9cSGreg Clayton                         error.SetErrorString ("no process specified, create a target with a file, or specify the --pid or --name command option");
6013a0b9cdfSJim Ingham                     }
6025aee162fSJim Ingham                 }
6033a0b9cdfSJim Ingham 
604144f3a9cSGreg Clayton                 if (error.Success())
6053a0b9cdfSJim Ingham                 {
606144f3a9cSGreg Clayton                     error = process->Attach (m_options.attach_info);
6073a0b9cdfSJim Ingham 
6085aee162fSJim Ingham                     if (error.Success())
6095aee162fSJim Ingham                     {
6105aee162fSJim Ingham                         result.SetStatus (eReturnStatusSuccessContinuingNoResult);
6115aee162fSJim Ingham                     }
6125aee162fSJim Ingham                     else
6135aee162fSJim Ingham                     {
614144f3a9cSGreg Clayton                         result.AppendErrorWithFormat ("attach failed: %s\n", error.AsCString());
6155aee162fSJim Ingham                         result.SetStatus (eReturnStatusFailed);
6165aee162fSJim Ingham                         return false;
6175aee162fSJim Ingham                     }
618bb3a283bSJim Ingham                     // If we're synchronous, wait for the stopped event and report that.
619bb3a283bSJim Ingham                     // Otherwise just return.
620bb3a283bSJim Ingham                     // FIXME: in the async case it will now be possible to get to the command
621bb3a283bSJim Ingham                     // interpreter with a state eStateAttaching.  Make sure we handle that correctly.
622bb3a283bSJim Ingham                     StateType state = process->WaitForProcessToStop (NULL);
623bb3a283bSJim Ingham 
624bb3a283bSJim Ingham                     result.SetDidChangeProcessState (true);
62581c22f61SGreg Clayton                     result.AppendMessageWithFormat ("Process %llu %s\n", process->GetID(), StateAsCString (state));
626bb3a283bSJim Ingham                     result.SetStatus (eReturnStatusSuccessFinishNoResult);
627bb3a283bSJim Ingham                 }
6285aee162fSJim Ingham             }
6295aee162fSJim Ingham         }
6305aee162fSJim Ingham 
6315aee162fSJim Ingham         if (result.Succeeded())
6325aee162fSJim Ingham         {
6335aee162fSJim Ingham             // Okay, we're done.  Last step is to warn if the executable module has changed:
634513c26ceSGreg Clayton             char new_path[PATH_MAX];
635aa149cbdSGreg Clayton             ModuleSP new_exec_module_sp (target->GetExecutableModule());
6365aee162fSJim Ingham             if (!old_exec_module_sp)
6375aee162fSJim Ingham             {
638513c26ceSGreg Clayton                 // We might not have a module if we attached to a raw pid...
639aa149cbdSGreg Clayton                 if (new_exec_module_sp)
640513c26ceSGreg Clayton                 {
641aa149cbdSGreg Clayton                     new_exec_module_sp->GetFileSpec().GetPath(new_path, PATH_MAX);
642513c26ceSGreg Clayton                     result.AppendMessageWithFormat("Executable module set to \"%s\".\n", new_path);
643513c26ceSGreg Clayton                 }
6445aee162fSJim Ingham             }
645aa149cbdSGreg Clayton             else if (old_exec_module_sp->GetFileSpec() != new_exec_module_sp->GetFileSpec())
6465aee162fSJim Ingham             {
647513c26ceSGreg Clayton                 char old_path[PATH_MAX];
6485aee162fSJim Ingham 
6495aee162fSJim Ingham                 old_exec_module_sp->GetFileSpec().GetPath (old_path, PATH_MAX);
650aa149cbdSGreg Clayton                 new_exec_module_sp->GetFileSpec().GetPath (new_path, PATH_MAX);
6515aee162fSJim Ingham 
6525aee162fSJim Ingham                 result.AppendWarningWithFormat("Executable module changed from \"%s\" to \"%s\".\n",
6535aee162fSJim Ingham                                                     old_path, new_path);
6545aee162fSJim Ingham             }
6555aee162fSJim Ingham 
6565aee162fSJim Ingham             if (!old_arch_spec.IsValid())
6575aee162fSJim Ingham             {
65864195a2cSGreg Clayton                 result.AppendMessageWithFormat ("Architecture set to: %s.\n", target->GetArchitecture().GetArchitectureName());
6595aee162fSJim Ingham             }
6605aee162fSJim Ingham             else if (old_arch_spec != target->GetArchitecture())
6615aee162fSJim Ingham             {
6625aee162fSJim Ingham                 result.AppendWarningWithFormat("Architecture changed from %s to %s.\n",
66364195a2cSGreg Clayton                                                 old_arch_spec.GetArchitectureName(), target->GetArchitecture().GetArchitectureName());
6645aee162fSJim Ingham             }
6655aee162fSJim Ingham         }
6665aee162fSJim Ingham         return result.Succeeded();
6675aee162fSJim Ingham     }
6685aee162fSJim Ingham 
6695aee162fSJim Ingham     Options *
6705aee162fSJim Ingham     GetOptions ()
6715aee162fSJim Ingham     {
6725aee162fSJim Ingham         return &m_options;
6735aee162fSJim Ingham     }
6745aee162fSJim Ingham 
67530fdc8d8SChris Lattner protected:
67630fdc8d8SChris Lattner 
67730fdc8d8SChris Lattner     CommandOptions m_options;
67830fdc8d8SChris Lattner };
67930fdc8d8SChris Lattner 
68030fdc8d8SChris Lattner 
681e0d378b3SGreg Clayton OptionDefinition
68230fdc8d8SChris Lattner CommandObjectProcessAttach::CommandOptions::g_option_table[] =
68330fdc8d8SChris Lattner {
684deaab222SCaroline Tice { LLDB_OPT_SET_ALL, false, "plugin", 'P', required_argument, NULL, 0, eArgTypePlugin,        "Name of the process plugin you want to use."},
685deaab222SCaroline Tice { LLDB_OPT_SET_1,   false, "pid",    'p', required_argument, NULL, 0, eArgTypePid,           "The process ID of an existing process to attach to."},
686deaab222SCaroline Tice { LLDB_OPT_SET_2,   false, "name",   'n', required_argument, NULL, 0, eArgTypeProcessName,  "The name of the process to attach to."},
687deaab222SCaroline Tice { LLDB_OPT_SET_2,   false, "waitfor",'w', no_argument,       NULL, 0, eArgTypeNone,              "Wait for the the process with <process-name> to launch."},
688deaab222SCaroline Tice { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL }
68930fdc8d8SChris Lattner };
69030fdc8d8SChris Lattner 
69130fdc8d8SChris Lattner //-------------------------------------------------------------------------
69230fdc8d8SChris Lattner // CommandObjectProcessContinue
69330fdc8d8SChris Lattner //-------------------------------------------------------------------------
694bb9caf73SJim Ingham #pragma mark CommandObjectProcessContinue
69530fdc8d8SChris Lattner 
69630fdc8d8SChris Lattner class CommandObjectProcessContinue : public CommandObject
69730fdc8d8SChris Lattner {
69830fdc8d8SChris Lattner public:
69930fdc8d8SChris Lattner 
700a7015092SGreg Clayton     CommandObjectProcessContinue (CommandInterpreter &interpreter) :
701a7015092SGreg Clayton         CommandObject (interpreter,
702a7015092SGreg Clayton                        "process continue",
703e3d26315SCaroline Tice                        "Continue execution of all threads in the current process.",
70430fdc8d8SChris Lattner                        "process continue",
70530fdc8d8SChris Lattner                        eFlagProcessMustBeLaunched | eFlagProcessMustBePaused)
70630fdc8d8SChris Lattner     {
70730fdc8d8SChris Lattner     }
70830fdc8d8SChris Lattner 
70930fdc8d8SChris Lattner 
71030fdc8d8SChris Lattner     ~CommandObjectProcessContinue ()
71130fdc8d8SChris Lattner     {
71230fdc8d8SChris Lattner     }
71330fdc8d8SChris Lattner 
71430fdc8d8SChris Lattner     bool
715a7015092SGreg Clayton     Execute (Args& command,
71630fdc8d8SChris Lattner              CommandReturnObject &result)
71730fdc8d8SChris Lattner     {
718c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
719a7015092SGreg Clayton         bool synchronous_execution = m_interpreter.GetSynchronous ();
72030fdc8d8SChris Lattner 
72130fdc8d8SChris Lattner         if (process == NULL)
72230fdc8d8SChris Lattner         {
72330fdc8d8SChris Lattner             result.AppendError ("no process to continue");
72430fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
72530fdc8d8SChris Lattner             return false;
72630fdc8d8SChris Lattner          }
72730fdc8d8SChris Lattner 
72830fdc8d8SChris Lattner         StateType state = process->GetState();
72930fdc8d8SChris Lattner         if (state == eStateStopped)
73030fdc8d8SChris Lattner         {
73130fdc8d8SChris Lattner             if (command.GetArgumentCount() != 0)
73230fdc8d8SChris Lattner             {
73330fdc8d8SChris Lattner                 result.AppendErrorWithFormat ("The '%s' command does not take any arguments.\n", m_cmd_name.c_str());
73430fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusFailed);
73530fdc8d8SChris Lattner                 return false;
73630fdc8d8SChris Lattner             }
73730fdc8d8SChris Lattner 
73830fdc8d8SChris Lattner             const uint32_t num_threads = process->GetThreadList().GetSize();
73930fdc8d8SChris Lattner 
74030fdc8d8SChris Lattner             // Set the actions that the threads should each take when resuming
74130fdc8d8SChris Lattner             for (uint32_t idx=0; idx<num_threads; ++idx)
74230fdc8d8SChris Lattner             {
74330fdc8d8SChris Lattner                 process->GetThreadList().GetThreadAtIndex(idx)->SetResumeState (eStateRunning);
74430fdc8d8SChris Lattner             }
74530fdc8d8SChris Lattner 
74630fdc8d8SChris Lattner             Error error(process->Resume());
74730fdc8d8SChris Lattner             if (error.Success())
74830fdc8d8SChris Lattner             {
74981c22f61SGreg Clayton                 result.AppendMessageWithFormat ("Process %llu resuming\n", process->GetID());
75030fdc8d8SChris Lattner                 if (synchronous_execution)
75130fdc8d8SChris Lattner                 {
752b132097bSGreg Clayton                     state = process->WaitForProcessToStop (NULL);
75330fdc8d8SChris Lattner 
75430fdc8d8SChris Lattner                     result.SetDidChangeProcessState (true);
75581c22f61SGreg Clayton                     result.AppendMessageWithFormat ("Process %llu %s\n", process->GetID(), StateAsCString (state));
75630fdc8d8SChris Lattner                     result.SetStatus (eReturnStatusSuccessFinishNoResult);
75730fdc8d8SChris Lattner                 }
75830fdc8d8SChris Lattner                 else
75930fdc8d8SChris Lattner                 {
76030fdc8d8SChris Lattner                     result.SetStatus (eReturnStatusSuccessContinuingNoResult);
76130fdc8d8SChris Lattner                 }
76230fdc8d8SChris Lattner             }
76330fdc8d8SChris Lattner             else
76430fdc8d8SChris Lattner             {
76530fdc8d8SChris Lattner                 result.AppendErrorWithFormat("Failed to resume process: %s.\n", error.AsCString());
76630fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusFailed);
76730fdc8d8SChris Lattner             }
76830fdc8d8SChris Lattner         }
76930fdc8d8SChris Lattner         else
77030fdc8d8SChris Lattner         {
77130fdc8d8SChris Lattner             result.AppendErrorWithFormat ("Process cannot be continued from its current state (%s).\n",
77230fdc8d8SChris Lattner                                          StateAsCString(state));
77330fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
77430fdc8d8SChris Lattner         }
77530fdc8d8SChris Lattner         return result.Succeeded();
77630fdc8d8SChris Lattner     }
77730fdc8d8SChris Lattner };
77830fdc8d8SChris Lattner 
77930fdc8d8SChris Lattner //-------------------------------------------------------------------------
78030fdc8d8SChris Lattner // CommandObjectProcessDetach
78130fdc8d8SChris Lattner //-------------------------------------------------------------------------
782bb9caf73SJim Ingham #pragma mark CommandObjectProcessDetach
78330fdc8d8SChris Lattner 
78430fdc8d8SChris Lattner class CommandObjectProcessDetach : public CommandObject
78530fdc8d8SChris Lattner {
78630fdc8d8SChris Lattner public:
78730fdc8d8SChris Lattner 
788a7015092SGreg Clayton     CommandObjectProcessDetach (CommandInterpreter &interpreter) :
789a7015092SGreg Clayton         CommandObject (interpreter,
790a7015092SGreg Clayton                        "process detach",
791e3d26315SCaroline Tice                        "Detach from the current process being debugged.",
79230fdc8d8SChris Lattner                        "process detach",
79330fdc8d8SChris Lattner                        eFlagProcessMustBeLaunched)
79430fdc8d8SChris Lattner     {
79530fdc8d8SChris Lattner     }
79630fdc8d8SChris Lattner 
79730fdc8d8SChris Lattner     ~CommandObjectProcessDetach ()
79830fdc8d8SChris Lattner     {
79930fdc8d8SChris Lattner     }
80030fdc8d8SChris Lattner 
80130fdc8d8SChris Lattner     bool
802a7015092SGreg Clayton     Execute (Args& command,
80330fdc8d8SChris Lattner              CommandReturnObject &result)
80430fdc8d8SChris Lattner     {
805c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
80630fdc8d8SChris Lattner         if (process == NULL)
80730fdc8d8SChris Lattner         {
80830fdc8d8SChris Lattner             result.AppendError ("must have a valid process in order to detach");
80930fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
81030fdc8d8SChris Lattner             return false;
81130fdc8d8SChris Lattner         }
81230fdc8d8SChris Lattner 
81381c22f61SGreg Clayton         result.AppendMessageWithFormat ("Detaching from process %llu\n", process->GetID());
81430fdc8d8SChris Lattner         Error error (process->Detach());
81530fdc8d8SChris Lattner         if (error.Success())
81630fdc8d8SChris Lattner         {
81730fdc8d8SChris Lattner             result.SetStatus (eReturnStatusSuccessFinishResult);
81830fdc8d8SChris Lattner         }
81930fdc8d8SChris Lattner         else
82030fdc8d8SChris Lattner         {
82130fdc8d8SChris Lattner             result.AppendErrorWithFormat ("Detach failed: %s\n", error.AsCString());
82230fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
82330fdc8d8SChris Lattner             return false;
82430fdc8d8SChris Lattner         }
82530fdc8d8SChris Lattner         return result.Succeeded();
82630fdc8d8SChris Lattner     }
82730fdc8d8SChris Lattner };
82830fdc8d8SChris Lattner 
82930fdc8d8SChris Lattner //-------------------------------------------------------------------------
830b766a73dSGreg Clayton // CommandObjectProcessConnect
831b766a73dSGreg Clayton //-------------------------------------------------------------------------
832b766a73dSGreg Clayton #pragma mark CommandObjectProcessConnect
833b766a73dSGreg Clayton 
834b766a73dSGreg Clayton class CommandObjectProcessConnect : public CommandObject
835b766a73dSGreg Clayton {
836b766a73dSGreg Clayton public:
837b766a73dSGreg Clayton 
838b766a73dSGreg Clayton     class CommandOptions : public Options
839b766a73dSGreg Clayton     {
840b766a73dSGreg Clayton     public:
841b766a73dSGreg Clayton 
842eb0103f2SGreg Clayton         CommandOptions (CommandInterpreter &interpreter) :
843eb0103f2SGreg Clayton             Options(interpreter)
844b766a73dSGreg Clayton         {
845f6b8b581SGreg Clayton             // Keep default values of all options in one place: OptionParsingStarting ()
846f6b8b581SGreg Clayton             OptionParsingStarting ();
847b766a73dSGreg Clayton         }
848b766a73dSGreg Clayton 
849b766a73dSGreg Clayton         ~CommandOptions ()
850b766a73dSGreg Clayton         {
851b766a73dSGreg Clayton         }
852b766a73dSGreg Clayton 
853b766a73dSGreg Clayton         Error
854f6b8b581SGreg Clayton         SetOptionValue (uint32_t option_idx, const char *option_arg)
855b766a73dSGreg Clayton         {
856b766a73dSGreg Clayton             Error error;
857b766a73dSGreg Clayton             char short_option = (char) m_getopt_table[option_idx].val;
858b766a73dSGreg Clayton 
859b766a73dSGreg Clayton             switch (short_option)
860b766a73dSGreg Clayton             {
861b766a73dSGreg Clayton             case 'p':
862b766a73dSGreg Clayton                 plugin_name.assign (option_arg);
863b766a73dSGreg Clayton                 break;
864b766a73dSGreg Clayton 
865b766a73dSGreg Clayton             default:
86686edbf41SGreg Clayton                 error.SetErrorStringWithFormat("invalid short option character '%c'", short_option);
867b766a73dSGreg Clayton                 break;
868b766a73dSGreg Clayton             }
869b766a73dSGreg Clayton             return error;
870b766a73dSGreg Clayton         }
871b766a73dSGreg Clayton 
872b766a73dSGreg Clayton         void
873f6b8b581SGreg Clayton         OptionParsingStarting ()
874b766a73dSGreg Clayton         {
875b766a73dSGreg Clayton             plugin_name.clear();
876b766a73dSGreg Clayton         }
877b766a73dSGreg Clayton 
878e0d378b3SGreg Clayton         const OptionDefinition*
879b766a73dSGreg Clayton         GetDefinitions ()
880b766a73dSGreg Clayton         {
881b766a73dSGreg Clayton             return g_option_table;
882b766a73dSGreg Clayton         }
883b766a73dSGreg Clayton 
884b766a73dSGreg Clayton         // Options table: Required for subclasses of Options.
885b766a73dSGreg Clayton 
886e0d378b3SGreg Clayton         static OptionDefinition g_option_table[];
887b766a73dSGreg Clayton 
888b766a73dSGreg Clayton         // Instance variables to hold the values for command options.
889b766a73dSGreg Clayton 
890b766a73dSGreg Clayton         std::string plugin_name;
891b766a73dSGreg Clayton     };
892b766a73dSGreg Clayton 
893b766a73dSGreg Clayton     CommandObjectProcessConnect (CommandInterpreter &interpreter) :
894b766a73dSGreg Clayton         CommandObject (interpreter,
895b766a73dSGreg Clayton                        "process connect",
896b766a73dSGreg Clayton                        "Connect to a remote debug service.",
897b766a73dSGreg Clayton                        "process connect <remote-url>",
898eb0103f2SGreg Clayton                        0),
899eb0103f2SGreg Clayton         m_options (interpreter)
900b766a73dSGreg Clayton     {
901b766a73dSGreg Clayton     }
902b766a73dSGreg Clayton 
903b766a73dSGreg Clayton     ~CommandObjectProcessConnect ()
904b766a73dSGreg Clayton     {
905b766a73dSGreg Clayton     }
906b766a73dSGreg Clayton 
907b766a73dSGreg Clayton 
908b766a73dSGreg Clayton     bool
909b766a73dSGreg Clayton     Execute (Args& command,
910b766a73dSGreg Clayton              CommandReturnObject &result)
911b766a73dSGreg Clayton     {
912b766a73dSGreg Clayton 
913b766a73dSGreg Clayton         TargetSP target_sp (m_interpreter.GetDebugger().GetSelectedTarget());
914b766a73dSGreg Clayton         Error error;
915c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
916b766a73dSGreg Clayton         if (process)
917b766a73dSGreg Clayton         {
918b766a73dSGreg Clayton             if (process->IsAlive())
919b766a73dSGreg Clayton             {
92081c22f61SGreg Clayton                 result.AppendErrorWithFormat ("Process %llu is currently being debugged, kill the process before connecting.\n",
921b766a73dSGreg Clayton                                               process->GetID());
922b766a73dSGreg Clayton                 result.SetStatus (eReturnStatusFailed);
923b766a73dSGreg Clayton                 return false;
924b766a73dSGreg Clayton             }
925b766a73dSGreg Clayton         }
926b766a73dSGreg Clayton 
927b766a73dSGreg Clayton         if (!target_sp)
928b766a73dSGreg Clayton         {
929b766a73dSGreg Clayton             // If there isn't a current target create one.
930b766a73dSGreg Clayton             FileSpec emptyFileSpec;
931b766a73dSGreg Clayton 
932b766a73dSGreg Clayton             error = m_interpreter.GetDebugger().GetTargetList().CreateTarget (m_interpreter.GetDebugger(),
933b766a73dSGreg Clayton                                                                               emptyFileSpec,
934cac9c5f9SGreg Clayton                                                                               NULL,
935b766a73dSGreg Clayton                                                                               false,
936cac9c5f9SGreg Clayton                                                                               NULL, // No platform options
937b766a73dSGreg Clayton                                                                               target_sp);
938b766a73dSGreg Clayton             if (!target_sp || error.Fail())
939b766a73dSGreg Clayton             {
940b766a73dSGreg Clayton                 result.AppendError(error.AsCString("Error creating target"));
941b766a73dSGreg Clayton                 result.SetStatus (eReturnStatusFailed);
942b766a73dSGreg Clayton                 return false;
943b766a73dSGreg Clayton             }
944b766a73dSGreg Clayton             m_interpreter.GetDebugger().GetTargetList().SetSelectedTarget(target_sp.get());
945b766a73dSGreg Clayton         }
946b766a73dSGreg Clayton 
947b766a73dSGreg Clayton         if (command.GetArgumentCount() == 1)
948b766a73dSGreg Clayton         {
949b766a73dSGreg Clayton             const char *plugin_name = NULL;
950b766a73dSGreg Clayton             if (!m_options.plugin_name.empty())
951b766a73dSGreg Clayton                 plugin_name = m_options.plugin_name.c_str();
952b766a73dSGreg Clayton 
953b766a73dSGreg Clayton             const char *remote_url = command.GetArgumentAtIndex(0);
954b766a73dSGreg Clayton             process = target_sp->CreateProcess (m_interpreter.GetDebugger().GetListener(), plugin_name).get();
955b766a73dSGreg Clayton 
956b766a73dSGreg Clayton             if (process)
957b766a73dSGreg Clayton             {
958b766a73dSGreg Clayton                 error = process->ConnectRemote (remote_url);
959b766a73dSGreg Clayton 
960b766a73dSGreg Clayton                 if (error.Fail())
961b766a73dSGreg Clayton                 {
962b766a73dSGreg Clayton                     result.AppendError(error.AsCString("Remote connect failed"));
963b766a73dSGreg Clayton                     result.SetStatus (eReturnStatusFailed);
964b766a73dSGreg Clayton                     return false;
965b766a73dSGreg Clayton                 }
966b766a73dSGreg Clayton             }
967b766a73dSGreg Clayton             else
968b766a73dSGreg Clayton             {
969fd54b368SJason 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",
970fd54b368SJason Molenda                                               m_cmd_name.c_str());
971b766a73dSGreg Clayton                 result.SetStatus (eReturnStatusFailed);
972b766a73dSGreg Clayton             }
973b766a73dSGreg Clayton         }
974b766a73dSGreg Clayton         else
975b766a73dSGreg Clayton         {
976fd54b368SJason Molenda             result.AppendErrorWithFormat ("'%s' takes exactly one argument:\nUsage: %s\n",
977b766a73dSGreg Clayton                                           m_cmd_name.c_str(),
978b766a73dSGreg Clayton                                           m_cmd_syntax.c_str());
979b766a73dSGreg Clayton             result.SetStatus (eReturnStatusFailed);
980b766a73dSGreg Clayton         }
981b766a73dSGreg Clayton         return result.Succeeded();
982b766a73dSGreg Clayton     }
983b766a73dSGreg Clayton 
984b766a73dSGreg Clayton     Options *
985b766a73dSGreg Clayton     GetOptions ()
986b766a73dSGreg Clayton     {
987b766a73dSGreg Clayton         return &m_options;
988b766a73dSGreg Clayton     }
989b766a73dSGreg Clayton 
990b766a73dSGreg Clayton protected:
991b766a73dSGreg Clayton 
992b766a73dSGreg Clayton     CommandOptions m_options;
993b766a73dSGreg Clayton };
994b766a73dSGreg Clayton 
995b766a73dSGreg Clayton 
996e0d378b3SGreg Clayton OptionDefinition
997b766a73dSGreg Clayton CommandObjectProcessConnect::CommandOptions::g_option_table[] =
998b766a73dSGreg Clayton {
999b766a73dSGreg Clayton     { LLDB_OPT_SET_ALL, false, "plugin", 'p', required_argument, NULL, 0, eArgTypePlugin, "Name of the process plugin you want to use."},
1000b766a73dSGreg Clayton     { 0,                false, NULL,      0 , 0,                 NULL, 0, eArgTypeNone,   NULL }
1001b766a73dSGreg Clayton };
1002b766a73dSGreg Clayton 
1003b766a73dSGreg Clayton //-------------------------------------------------------------------------
10048f343b09SGreg Clayton // CommandObjectProcessLoad
10058f343b09SGreg Clayton //-------------------------------------------------------------------------
1006bb9caf73SJim Ingham #pragma mark CommandObjectProcessLoad
10078f343b09SGreg Clayton 
10088f343b09SGreg Clayton class CommandObjectProcessLoad : public CommandObject
10098f343b09SGreg Clayton {
10108f343b09SGreg Clayton public:
10118f343b09SGreg Clayton 
10128f343b09SGreg Clayton     CommandObjectProcessLoad (CommandInterpreter &interpreter) :
10138f343b09SGreg Clayton         CommandObject (interpreter,
10148f343b09SGreg Clayton                        "process load",
10158f343b09SGreg Clayton                        "Load a shared library into the current process.",
10168f343b09SGreg Clayton                        "process load <filename> [<filename> ...]",
10178f343b09SGreg Clayton                        eFlagProcessMustBeLaunched | eFlagProcessMustBePaused)
10188f343b09SGreg Clayton     {
10198f343b09SGreg Clayton     }
10208f343b09SGreg Clayton 
10218f343b09SGreg Clayton     ~CommandObjectProcessLoad ()
10228f343b09SGreg Clayton     {
10238f343b09SGreg Clayton     }
10248f343b09SGreg Clayton 
10258f343b09SGreg Clayton     bool
10268f343b09SGreg Clayton     Execute (Args& command,
10278f343b09SGreg Clayton              CommandReturnObject &result)
10288f343b09SGreg Clayton     {
1029c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
10308f343b09SGreg Clayton         if (process == NULL)
10318f343b09SGreg Clayton         {
10328f343b09SGreg Clayton             result.AppendError ("must have a valid process in order to load a shared library");
10338f343b09SGreg Clayton             result.SetStatus (eReturnStatusFailed);
10348f343b09SGreg Clayton             return false;
10358f343b09SGreg Clayton         }
10368f343b09SGreg Clayton 
10378f343b09SGreg Clayton         const uint32_t argc = command.GetArgumentCount();
10388f343b09SGreg Clayton 
10398f343b09SGreg Clayton         for (uint32_t i=0; i<argc; ++i)
10408f343b09SGreg Clayton         {
10418f343b09SGreg Clayton             Error error;
10428f343b09SGreg Clayton             const char *image_path = command.GetArgumentAtIndex(i);
10438f343b09SGreg Clayton             FileSpec image_spec (image_path, false);
1044aa516843SGreg Clayton             process->GetTarget().GetPlatform()->ResolveRemotePath(image_spec, image_spec);
10458f343b09SGreg Clayton             uint32_t image_token = process->LoadImage(image_spec, error);
10468f343b09SGreg Clayton             if (image_token != LLDB_INVALID_IMAGE_TOKEN)
10478f343b09SGreg Clayton             {
10488f343b09SGreg Clayton                 result.AppendMessageWithFormat ("Loading \"%s\"...ok\nImage %u loaded.\n", image_path, image_token);
10498f343b09SGreg Clayton                 result.SetStatus (eReturnStatusSuccessFinishResult);
10508f343b09SGreg Clayton             }
10518f343b09SGreg Clayton             else
10528f343b09SGreg Clayton             {
10538f343b09SGreg Clayton                 result.AppendErrorWithFormat ("failed to load '%s': %s", image_path, error.AsCString());
10548f343b09SGreg Clayton                 result.SetStatus (eReturnStatusFailed);
10558f343b09SGreg Clayton             }
10568f343b09SGreg Clayton         }
10578f343b09SGreg Clayton         return result.Succeeded();
10588f343b09SGreg Clayton     }
10598f343b09SGreg Clayton };
10608f343b09SGreg Clayton 
10618f343b09SGreg Clayton 
10628f343b09SGreg Clayton //-------------------------------------------------------------------------
10638f343b09SGreg Clayton // CommandObjectProcessUnload
10648f343b09SGreg Clayton //-------------------------------------------------------------------------
1065bb9caf73SJim Ingham #pragma mark CommandObjectProcessUnload
10668f343b09SGreg Clayton 
10678f343b09SGreg Clayton class CommandObjectProcessUnload : public CommandObject
10688f343b09SGreg Clayton {
10698f343b09SGreg Clayton public:
10708f343b09SGreg Clayton 
10718f343b09SGreg Clayton     CommandObjectProcessUnload (CommandInterpreter &interpreter) :
10728f343b09SGreg Clayton         CommandObject (interpreter,
10738f343b09SGreg Clayton                        "process unload",
10748f343b09SGreg Clayton                        "Unload a shared library from the current process using the index returned by a previous call to \"process load\".",
10758f343b09SGreg Clayton                        "process unload <index>",
10768f343b09SGreg Clayton                        eFlagProcessMustBeLaunched | eFlagProcessMustBePaused)
10778f343b09SGreg Clayton     {
10788f343b09SGreg Clayton     }
10798f343b09SGreg Clayton 
10808f343b09SGreg Clayton     ~CommandObjectProcessUnload ()
10818f343b09SGreg Clayton     {
10828f343b09SGreg Clayton     }
10838f343b09SGreg Clayton 
10848f343b09SGreg Clayton     bool
10858f343b09SGreg Clayton     Execute (Args& command,
10868f343b09SGreg Clayton              CommandReturnObject &result)
10878f343b09SGreg Clayton     {
1088c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
10898f343b09SGreg Clayton         if (process == NULL)
10908f343b09SGreg Clayton         {
10918f343b09SGreg Clayton             result.AppendError ("must have a valid process in order to load a shared library");
10928f343b09SGreg Clayton             result.SetStatus (eReturnStatusFailed);
10938f343b09SGreg Clayton             return false;
10948f343b09SGreg Clayton         }
10958f343b09SGreg Clayton 
10968f343b09SGreg Clayton         const uint32_t argc = command.GetArgumentCount();
10978f343b09SGreg Clayton 
10988f343b09SGreg Clayton         for (uint32_t i=0; i<argc; ++i)
10998f343b09SGreg Clayton         {
11008f343b09SGreg Clayton             const char *image_token_cstr = command.GetArgumentAtIndex(i);
11018f343b09SGreg Clayton             uint32_t image_token = Args::StringToUInt32(image_token_cstr, LLDB_INVALID_IMAGE_TOKEN, 0);
11028f343b09SGreg Clayton             if (image_token == LLDB_INVALID_IMAGE_TOKEN)
11038f343b09SGreg Clayton             {
11048f343b09SGreg Clayton                 result.AppendErrorWithFormat ("invalid image index argument '%s'", image_token_cstr);
11058f343b09SGreg Clayton                 result.SetStatus (eReturnStatusFailed);
11068f343b09SGreg Clayton                 break;
11078f343b09SGreg Clayton             }
11088f343b09SGreg Clayton             else
11098f343b09SGreg Clayton             {
11108f343b09SGreg Clayton                 Error error (process->UnloadImage(image_token));
11118f343b09SGreg Clayton                 if (error.Success())
11128f343b09SGreg Clayton                 {
11138f343b09SGreg Clayton                     result.AppendMessageWithFormat ("Unloading shared library with index %u...ok\n", image_token);
11148f343b09SGreg Clayton                     result.SetStatus (eReturnStatusSuccessFinishResult);
11158f343b09SGreg Clayton                 }
11168f343b09SGreg Clayton                 else
11178f343b09SGreg Clayton                 {
11188f343b09SGreg Clayton                     result.AppendErrorWithFormat ("failed to unload image: %s", error.AsCString());
11198f343b09SGreg Clayton                     result.SetStatus (eReturnStatusFailed);
11208f343b09SGreg Clayton                     break;
11218f343b09SGreg Clayton                 }
11228f343b09SGreg Clayton             }
11238f343b09SGreg Clayton         }
11248f343b09SGreg Clayton         return result.Succeeded();
11258f343b09SGreg Clayton     }
11268f343b09SGreg Clayton };
11278f343b09SGreg Clayton 
11288f343b09SGreg Clayton //-------------------------------------------------------------------------
112930fdc8d8SChris Lattner // CommandObjectProcessSignal
113030fdc8d8SChris Lattner //-------------------------------------------------------------------------
1131bb9caf73SJim Ingham #pragma mark CommandObjectProcessSignal
113230fdc8d8SChris Lattner 
113330fdc8d8SChris Lattner class CommandObjectProcessSignal : public CommandObject
113430fdc8d8SChris Lattner {
113530fdc8d8SChris Lattner public:
113630fdc8d8SChris Lattner 
1137a7015092SGreg Clayton     CommandObjectProcessSignal (CommandInterpreter &interpreter) :
1138a7015092SGreg Clayton         CommandObject (interpreter,
1139a7015092SGreg Clayton                        "process signal",
1140e3d26315SCaroline Tice                        "Send a UNIX signal to the current process being debugged.",
1141405fe67fSCaroline Tice                        NULL)
114230fdc8d8SChris Lattner     {
1143405fe67fSCaroline Tice         CommandArgumentEntry arg;
1144405fe67fSCaroline Tice         CommandArgumentData signal_arg;
1145405fe67fSCaroline Tice 
1146405fe67fSCaroline Tice         // Define the first (and only) variant of this arg.
1147c0dbdfb6SCaroline Tice         signal_arg.arg_type = eArgTypeUnixSignal;
1148405fe67fSCaroline Tice         signal_arg.arg_repetition = eArgRepeatPlain;
1149405fe67fSCaroline Tice 
1150405fe67fSCaroline Tice         // There is only one variant this argument could be; put it into the argument entry.
1151405fe67fSCaroline Tice         arg.push_back (signal_arg);
1152405fe67fSCaroline Tice 
1153405fe67fSCaroline Tice         // Push the data for the first argument into the m_arguments vector.
1154405fe67fSCaroline Tice         m_arguments.push_back (arg);
115530fdc8d8SChris Lattner     }
115630fdc8d8SChris Lattner 
115730fdc8d8SChris Lattner     ~CommandObjectProcessSignal ()
115830fdc8d8SChris Lattner     {
115930fdc8d8SChris Lattner     }
116030fdc8d8SChris Lattner 
116130fdc8d8SChris Lattner     bool
1162a7015092SGreg Clayton     Execute (Args& command,
116330fdc8d8SChris Lattner              CommandReturnObject &result)
116430fdc8d8SChris Lattner     {
1165c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
116630fdc8d8SChris Lattner         if (process == NULL)
116730fdc8d8SChris Lattner         {
116830fdc8d8SChris Lattner             result.AppendError ("no process to signal");
116930fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
117030fdc8d8SChris Lattner             return false;
117130fdc8d8SChris Lattner         }
117230fdc8d8SChris Lattner 
117330fdc8d8SChris Lattner         if (command.GetArgumentCount() == 1)
117430fdc8d8SChris Lattner         {
1175237cd906SGreg Clayton             int signo = LLDB_INVALID_SIGNAL_NUMBER;
1176237cd906SGreg Clayton 
1177237cd906SGreg Clayton             const char *signal_name = command.GetArgumentAtIndex(0);
1178237cd906SGreg Clayton             if (::isxdigit (signal_name[0]))
1179237cd906SGreg Clayton                 signo = Args::StringToSInt32(signal_name, LLDB_INVALID_SIGNAL_NUMBER, 0);
1180237cd906SGreg Clayton             else
1181237cd906SGreg Clayton                 signo = process->GetUnixSignals().GetSignalNumberFromName (signal_name);
1182237cd906SGreg Clayton 
1183237cd906SGreg Clayton             if (signo == LLDB_INVALID_SIGNAL_NUMBER)
118430fdc8d8SChris Lattner             {
118530fdc8d8SChris Lattner                 result.AppendErrorWithFormat ("Invalid signal argument '%s'.\n", command.GetArgumentAtIndex(0));
118630fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusFailed);
118730fdc8d8SChris Lattner             }
118830fdc8d8SChris Lattner             else
118930fdc8d8SChris Lattner             {
119030fdc8d8SChris Lattner                 Error error (process->Signal (signo));
119130fdc8d8SChris Lattner                 if (error.Success())
119230fdc8d8SChris Lattner                 {
119330fdc8d8SChris Lattner                     result.SetStatus (eReturnStatusSuccessFinishResult);
119430fdc8d8SChris Lattner                 }
119530fdc8d8SChris Lattner                 else
119630fdc8d8SChris Lattner                 {
119730fdc8d8SChris Lattner                     result.AppendErrorWithFormat ("Failed to send signal %i: %s\n", signo, error.AsCString());
119830fdc8d8SChris Lattner                     result.SetStatus (eReturnStatusFailed);
119930fdc8d8SChris Lattner                 }
120030fdc8d8SChris Lattner             }
120130fdc8d8SChris Lattner         }
120230fdc8d8SChris Lattner         else
120330fdc8d8SChris Lattner         {
1204fd54b368SJason Molenda             result.AppendErrorWithFormat("'%s' takes exactly one signal number argument:\nUsage: %s\n", m_cmd_name.c_str(),
120530fdc8d8SChris Lattner                                         m_cmd_syntax.c_str());
120630fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
120730fdc8d8SChris Lattner         }
120830fdc8d8SChris Lattner         return result.Succeeded();
120930fdc8d8SChris Lattner     }
121030fdc8d8SChris Lattner };
121130fdc8d8SChris Lattner 
121230fdc8d8SChris Lattner 
121330fdc8d8SChris Lattner //-------------------------------------------------------------------------
121430fdc8d8SChris Lattner // CommandObjectProcessInterrupt
121530fdc8d8SChris Lattner //-------------------------------------------------------------------------
1216bb9caf73SJim Ingham #pragma mark CommandObjectProcessInterrupt
121730fdc8d8SChris Lattner 
121830fdc8d8SChris Lattner class CommandObjectProcessInterrupt : public CommandObject
121930fdc8d8SChris Lattner {
122030fdc8d8SChris Lattner public:
122130fdc8d8SChris Lattner 
122230fdc8d8SChris Lattner 
1223a7015092SGreg Clayton     CommandObjectProcessInterrupt (CommandInterpreter &interpreter) :
1224a7015092SGreg Clayton     CommandObject (interpreter,
1225a7015092SGreg Clayton                    "process interrupt",
1226e3d26315SCaroline Tice                    "Interrupt the current process being debugged.",
122730fdc8d8SChris Lattner                    "process interrupt",
122830fdc8d8SChris Lattner                    eFlagProcessMustBeLaunched)
122930fdc8d8SChris Lattner     {
123030fdc8d8SChris Lattner     }
123130fdc8d8SChris Lattner 
123230fdc8d8SChris Lattner     ~CommandObjectProcessInterrupt ()
123330fdc8d8SChris Lattner     {
123430fdc8d8SChris Lattner     }
123530fdc8d8SChris Lattner 
123630fdc8d8SChris Lattner     bool
1237a7015092SGreg Clayton     Execute (Args& command,
123830fdc8d8SChris Lattner              CommandReturnObject &result)
123930fdc8d8SChris Lattner     {
1240c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
124130fdc8d8SChris Lattner         if (process == NULL)
124230fdc8d8SChris Lattner         {
124330fdc8d8SChris Lattner             result.AppendError ("no process to halt");
124430fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
124530fdc8d8SChris Lattner             return false;
124630fdc8d8SChris Lattner         }
124730fdc8d8SChris Lattner 
124830fdc8d8SChris Lattner         if (command.GetArgumentCount() == 0)
124930fdc8d8SChris Lattner         {
125030fdc8d8SChris Lattner             Error error(process->Halt ());
125130fdc8d8SChris Lattner             if (error.Success())
125230fdc8d8SChris Lattner             {
125330fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusSuccessFinishResult);
125430fdc8d8SChris Lattner 
125530fdc8d8SChris Lattner                 // Maybe we should add a "SuspendThreadPlans so we
125630fdc8d8SChris Lattner                 // can halt, and keep in place all the current thread plans.
125730fdc8d8SChris Lattner                 process->GetThreadList().DiscardThreadPlans();
125830fdc8d8SChris Lattner             }
125930fdc8d8SChris Lattner             else
126030fdc8d8SChris Lattner             {
126130fdc8d8SChris Lattner                 result.AppendErrorWithFormat ("Failed to halt process: %s\n", error.AsCString());
126230fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusFailed);
126330fdc8d8SChris Lattner             }
126430fdc8d8SChris Lattner         }
126530fdc8d8SChris Lattner         else
126630fdc8d8SChris Lattner         {
1267fd54b368SJason Molenda             result.AppendErrorWithFormat("'%s' takes no arguments:\nUsage: %s\n",
126830fdc8d8SChris Lattner                                         m_cmd_name.c_str(),
126930fdc8d8SChris Lattner                                         m_cmd_syntax.c_str());
127030fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
127130fdc8d8SChris Lattner         }
127230fdc8d8SChris Lattner         return result.Succeeded();
127330fdc8d8SChris Lattner     }
127430fdc8d8SChris Lattner };
127530fdc8d8SChris Lattner 
127630fdc8d8SChris Lattner //-------------------------------------------------------------------------
127730fdc8d8SChris Lattner // CommandObjectProcessKill
127830fdc8d8SChris Lattner //-------------------------------------------------------------------------
1279bb9caf73SJim Ingham #pragma mark CommandObjectProcessKill
128030fdc8d8SChris Lattner 
128130fdc8d8SChris Lattner class CommandObjectProcessKill : public CommandObject
128230fdc8d8SChris Lattner {
128330fdc8d8SChris Lattner public:
128430fdc8d8SChris Lattner 
1285a7015092SGreg Clayton     CommandObjectProcessKill (CommandInterpreter &interpreter) :
1286a7015092SGreg Clayton     CommandObject (interpreter,
1287a7015092SGreg Clayton                    "process kill",
1288e3d26315SCaroline Tice                    "Terminate the current process being debugged.",
128930fdc8d8SChris Lattner                    "process kill",
129030fdc8d8SChris Lattner                    eFlagProcessMustBeLaunched)
129130fdc8d8SChris Lattner     {
129230fdc8d8SChris Lattner     }
129330fdc8d8SChris Lattner 
129430fdc8d8SChris Lattner     ~CommandObjectProcessKill ()
129530fdc8d8SChris Lattner     {
129630fdc8d8SChris Lattner     }
129730fdc8d8SChris Lattner 
129830fdc8d8SChris Lattner     bool
1299a7015092SGreg Clayton     Execute (Args& command,
130030fdc8d8SChris Lattner              CommandReturnObject &result)
130130fdc8d8SChris Lattner     {
1302c14ee32dSGreg Clayton         Process *process = m_interpreter.GetExecutionContext().GetProcessPtr();
130330fdc8d8SChris Lattner         if (process == NULL)
130430fdc8d8SChris Lattner         {
130530fdc8d8SChris Lattner             result.AppendError ("no process to kill");
130630fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
130730fdc8d8SChris Lattner             return false;
130830fdc8d8SChris Lattner         }
130930fdc8d8SChris Lattner 
131030fdc8d8SChris Lattner         if (command.GetArgumentCount() == 0)
131130fdc8d8SChris Lattner         {
131230fdc8d8SChris Lattner             Error error (process->Destroy());
131330fdc8d8SChris Lattner             if (error.Success())
131430fdc8d8SChris Lattner             {
131530fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusSuccessFinishResult);
131630fdc8d8SChris Lattner             }
131730fdc8d8SChris Lattner             else
131830fdc8d8SChris Lattner             {
131930fdc8d8SChris Lattner                 result.AppendErrorWithFormat ("Failed to kill process: %s\n", error.AsCString());
132030fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusFailed);
132130fdc8d8SChris Lattner             }
132230fdc8d8SChris Lattner         }
132330fdc8d8SChris Lattner         else
132430fdc8d8SChris Lattner         {
1325fd54b368SJason Molenda             result.AppendErrorWithFormat("'%s' takes no arguments:\nUsage: %s\n",
132630fdc8d8SChris Lattner                                         m_cmd_name.c_str(),
132730fdc8d8SChris Lattner                                         m_cmd_syntax.c_str());
132830fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
132930fdc8d8SChris Lattner         }
133030fdc8d8SChris Lattner         return result.Succeeded();
133130fdc8d8SChris Lattner     }
133230fdc8d8SChris Lattner };
133330fdc8d8SChris Lattner 
133430fdc8d8SChris Lattner //-------------------------------------------------------------------------
13354b9bea87SJim Ingham // CommandObjectProcessStatus
13364b9bea87SJim Ingham //-------------------------------------------------------------------------
1337bb9caf73SJim Ingham #pragma mark CommandObjectProcessStatus
1338bb9caf73SJim Ingham 
13394b9bea87SJim Ingham class CommandObjectProcessStatus : public CommandObject
13404b9bea87SJim Ingham {
13414b9bea87SJim Ingham public:
1342a7015092SGreg Clayton     CommandObjectProcessStatus (CommandInterpreter &interpreter) :
1343a7015092SGreg Clayton     CommandObject (interpreter,
1344a7015092SGreg Clayton                    "process status",
1345e3d26315SCaroline Tice                    "Show the current status and location of executing process.",
1346e3d26315SCaroline Tice                    "process status",
13474b9bea87SJim Ingham                    0)
13484b9bea87SJim Ingham     {
13494b9bea87SJim Ingham     }
13504b9bea87SJim Ingham 
13514b9bea87SJim Ingham     ~CommandObjectProcessStatus()
13524b9bea87SJim Ingham     {
13534b9bea87SJim Ingham     }
13544b9bea87SJim Ingham 
13554b9bea87SJim Ingham 
13564b9bea87SJim Ingham     bool
13574b9bea87SJim Ingham     Execute
13584b9bea87SJim Ingham     (
13594b9bea87SJim Ingham         Args& command,
13604b9bea87SJim Ingham         CommandReturnObject &result
13614b9bea87SJim Ingham     )
13624b9bea87SJim Ingham     {
13637260f620SGreg Clayton         Stream &strm = result.GetOutputStream();
13644b9bea87SJim Ingham         result.SetStatus (eReturnStatusSuccessFinishNoResult);
13658b82f087SGreg Clayton         ExecutionContext exe_ctx(m_interpreter.GetExecutionContext());
1366c14ee32dSGreg Clayton         Process *process = exe_ctx.GetProcessPtr();
1367c14ee32dSGreg Clayton         if (process)
13684b9bea87SJim Ingham         {
13697260f620SGreg Clayton             const bool only_threads_with_stop_reason = true;
13707260f620SGreg Clayton             const uint32_t start_frame = 0;
13717260f620SGreg Clayton             const uint32_t num_frames = 1;
13727260f620SGreg Clayton             const uint32_t num_frames_with_source = 1;
1373c14ee32dSGreg Clayton             process->GetStatus(strm);
1374c14ee32dSGreg Clayton             process->GetThreadStatus (strm,
13757260f620SGreg Clayton                                       only_threads_with_stop_reason,
13767260f620SGreg Clayton                                       start_frame,
13777260f620SGreg Clayton                                       num_frames,
13787260f620SGreg Clayton                                       num_frames_with_source);
13797260f620SGreg Clayton 
13804b9bea87SJim Ingham         }
13814b9bea87SJim Ingham         else
13824b9bea87SJim Ingham         {
13837260f620SGreg Clayton             result.AppendError ("No process.");
13844b9bea87SJim Ingham             result.SetStatus (eReturnStatusFailed);
13854b9bea87SJim Ingham         }
13864b9bea87SJim Ingham         return result.Succeeded();
13874b9bea87SJim Ingham     }
13884b9bea87SJim Ingham };
13894b9bea87SJim Ingham 
13904b9bea87SJim Ingham //-------------------------------------------------------------------------
139135731357SCaroline Tice // CommandObjectProcessHandle
139235731357SCaroline Tice //-------------------------------------------------------------------------
1393bb9caf73SJim Ingham #pragma mark CommandObjectProcessHandle
139435731357SCaroline Tice 
139535731357SCaroline Tice class CommandObjectProcessHandle : public CommandObject
139635731357SCaroline Tice {
139735731357SCaroline Tice public:
139835731357SCaroline Tice 
139935731357SCaroline Tice     class CommandOptions : public Options
140035731357SCaroline Tice     {
140135731357SCaroline Tice     public:
140235731357SCaroline Tice 
1403eb0103f2SGreg Clayton         CommandOptions (CommandInterpreter &interpreter) :
1404eb0103f2SGreg Clayton             Options (interpreter)
140535731357SCaroline Tice         {
1406f6b8b581SGreg Clayton             OptionParsingStarting ();
140735731357SCaroline Tice         }
140835731357SCaroline Tice 
140935731357SCaroline Tice         ~CommandOptions ()
141035731357SCaroline Tice         {
141135731357SCaroline Tice         }
141235731357SCaroline Tice 
141335731357SCaroline Tice         Error
1414f6b8b581SGreg Clayton         SetOptionValue (uint32_t option_idx, const char *option_arg)
141535731357SCaroline Tice         {
141635731357SCaroline Tice             Error error;
141735731357SCaroline Tice             char short_option = (char) m_getopt_table[option_idx].val;
141835731357SCaroline Tice 
141935731357SCaroline Tice             switch (short_option)
142035731357SCaroline Tice             {
142135731357SCaroline Tice                 case 's':
142235731357SCaroline Tice                     stop = option_arg;
142335731357SCaroline Tice                     break;
142435731357SCaroline Tice                 case 'n':
142535731357SCaroline Tice                     notify = option_arg;
142635731357SCaroline Tice                     break;
142735731357SCaroline Tice                 case 'p':
142835731357SCaroline Tice                     pass = option_arg;
142935731357SCaroline Tice                     break;
143035731357SCaroline Tice                 default:
143186edbf41SGreg Clayton                     error.SetErrorStringWithFormat("invalid short option character '%c'", short_option);
143235731357SCaroline Tice                     break;
143335731357SCaroline Tice             }
143435731357SCaroline Tice             return error;
143535731357SCaroline Tice         }
143635731357SCaroline Tice 
143735731357SCaroline Tice         void
1438f6b8b581SGreg Clayton         OptionParsingStarting ()
143935731357SCaroline Tice         {
144035731357SCaroline Tice             stop.clear();
144135731357SCaroline Tice             notify.clear();
144235731357SCaroline Tice             pass.clear();
144335731357SCaroline Tice         }
144435731357SCaroline Tice 
1445e0d378b3SGreg Clayton         const OptionDefinition*
144635731357SCaroline Tice         GetDefinitions ()
144735731357SCaroline Tice         {
144835731357SCaroline Tice             return g_option_table;
144935731357SCaroline Tice         }
145035731357SCaroline Tice 
145135731357SCaroline Tice         // Options table: Required for subclasses of Options.
145235731357SCaroline Tice 
1453e0d378b3SGreg Clayton         static OptionDefinition g_option_table[];
145435731357SCaroline Tice 
145535731357SCaroline Tice         // Instance variables to hold the values for command options.
145635731357SCaroline Tice 
145735731357SCaroline Tice         std::string stop;
145835731357SCaroline Tice         std::string notify;
145935731357SCaroline Tice         std::string pass;
146035731357SCaroline Tice     };
146135731357SCaroline Tice 
146235731357SCaroline Tice 
146335731357SCaroline Tice     CommandObjectProcessHandle (CommandInterpreter &interpreter) :
146435731357SCaroline Tice         CommandObject (interpreter,
146535731357SCaroline Tice                        "process handle",
146610ad7993SCaroline Tice                        "Show or update what the process and debugger should do with various signals received from the OS.",
1467eb0103f2SGreg Clayton                        NULL),
1468eb0103f2SGreg Clayton         m_options (interpreter)
146935731357SCaroline Tice     {
147010ad7993SCaroline Tice         SetHelpLong ("If no signals are specified, update them all.  If no update option is specified, list the current values.\n");
147135731357SCaroline Tice         CommandArgumentEntry arg;
1472c0dbdfb6SCaroline Tice         CommandArgumentData signal_arg;
147335731357SCaroline Tice 
1474c0dbdfb6SCaroline Tice         signal_arg.arg_type = eArgTypeUnixSignal;
1475c0dbdfb6SCaroline Tice         signal_arg.arg_repetition = eArgRepeatStar;
147635731357SCaroline Tice 
1477c0dbdfb6SCaroline Tice         arg.push_back (signal_arg);
147835731357SCaroline Tice 
147935731357SCaroline Tice         m_arguments.push_back (arg);
148035731357SCaroline Tice     }
148135731357SCaroline Tice 
148235731357SCaroline Tice     ~CommandObjectProcessHandle ()
148335731357SCaroline Tice     {
148435731357SCaroline Tice     }
148535731357SCaroline Tice 
148635731357SCaroline Tice     Options *
148735731357SCaroline Tice     GetOptions ()
148835731357SCaroline Tice     {
148935731357SCaroline Tice         return &m_options;
149035731357SCaroline Tice     }
149135731357SCaroline Tice 
149235731357SCaroline Tice     bool
149310ad7993SCaroline Tice     VerifyCommandOptionValue (const std::string &option, int &real_value)
149435731357SCaroline Tice     {
149535731357SCaroline Tice         bool okay = true;
149635731357SCaroline Tice 
149710ad7993SCaroline Tice         bool success = false;
149810ad7993SCaroline Tice         bool tmp_value = Args::StringToBoolean (option.c_str(), false, &success);
149910ad7993SCaroline Tice 
150010ad7993SCaroline Tice         if (success && tmp_value)
150110ad7993SCaroline Tice             real_value = 1;
150210ad7993SCaroline Tice         else if (success && !tmp_value)
150310ad7993SCaroline Tice             real_value = 0;
150435731357SCaroline Tice         else
150535731357SCaroline Tice         {
150635731357SCaroline Tice             // If the value isn't 'true' or 'false', it had better be 0 or 1.
150710ad7993SCaroline Tice             real_value = Args::StringToUInt32 (option.c_str(), 3);
150810ad7993SCaroline Tice             if (real_value != 0 && real_value != 1)
150935731357SCaroline Tice                 okay = false;
151035731357SCaroline Tice         }
151135731357SCaroline Tice 
151235731357SCaroline Tice         return okay;
151335731357SCaroline Tice     }
151435731357SCaroline Tice 
151510ad7993SCaroline Tice     void
151610ad7993SCaroline Tice     PrintSignalHeader (Stream &str)
151710ad7993SCaroline Tice     {
151810ad7993SCaroline Tice         str.Printf ("NAME        PASS   STOP   NOTIFY\n");
151910ad7993SCaroline Tice         str.Printf ("==========  =====  =====  ======\n");
152010ad7993SCaroline Tice     }
152110ad7993SCaroline Tice 
152210ad7993SCaroline Tice     void
152310ad7993SCaroline Tice     PrintSignal (Stream &str, int32_t signo, const char *sig_name, UnixSignals &signals)
152410ad7993SCaroline Tice     {
152510ad7993SCaroline Tice         bool stop;
152610ad7993SCaroline Tice         bool suppress;
152710ad7993SCaroline Tice         bool notify;
152810ad7993SCaroline Tice 
152910ad7993SCaroline Tice         str.Printf ("%-10s  ", sig_name);
153010ad7993SCaroline Tice         if (signals.GetSignalInfo (signo, suppress, stop, notify))
153110ad7993SCaroline Tice         {
153210ad7993SCaroline Tice             bool pass = !suppress;
153310ad7993SCaroline Tice             str.Printf ("%s  %s  %s",
153410ad7993SCaroline Tice                         (pass ? "true " : "false"),
153510ad7993SCaroline Tice                         (stop ? "true " : "false"),
153610ad7993SCaroline Tice                         (notify ? "true " : "false"));
153710ad7993SCaroline Tice         }
153810ad7993SCaroline Tice         str.Printf ("\n");
153910ad7993SCaroline Tice     }
154010ad7993SCaroline Tice 
154110ad7993SCaroline Tice     void
154210ad7993SCaroline Tice     PrintSignalInformation (Stream &str, Args &signal_args, int num_valid_signals, UnixSignals &signals)
154310ad7993SCaroline Tice     {
154410ad7993SCaroline Tice         PrintSignalHeader (str);
154510ad7993SCaroline Tice 
154610ad7993SCaroline Tice         if (num_valid_signals > 0)
154710ad7993SCaroline Tice         {
154810ad7993SCaroline Tice             size_t num_args = signal_args.GetArgumentCount();
154910ad7993SCaroline Tice             for (size_t i = 0; i < num_args; ++i)
155010ad7993SCaroline Tice             {
155110ad7993SCaroline Tice                 int32_t signo = signals.GetSignalNumberFromName (signal_args.GetArgumentAtIndex (i));
155210ad7993SCaroline Tice                 if (signo != LLDB_INVALID_SIGNAL_NUMBER)
155310ad7993SCaroline Tice                     PrintSignal (str, signo, signal_args.GetArgumentAtIndex (i), signals);
155410ad7993SCaroline Tice             }
155510ad7993SCaroline Tice         }
155610ad7993SCaroline Tice         else // Print info for ALL signals
155710ad7993SCaroline Tice         {
155810ad7993SCaroline Tice             int32_t signo = signals.GetFirstSignalNumber();
155910ad7993SCaroline Tice             while (signo != LLDB_INVALID_SIGNAL_NUMBER)
156010ad7993SCaroline Tice             {
156110ad7993SCaroline Tice                 PrintSignal (str, signo, signals.GetSignalAsCString (signo), signals);
156210ad7993SCaroline Tice                 signo = signals.GetNextSignalNumber (signo);
156310ad7993SCaroline Tice             }
156410ad7993SCaroline Tice         }
156510ad7993SCaroline Tice     }
156610ad7993SCaroline Tice 
156735731357SCaroline Tice     bool
156835731357SCaroline Tice     Execute (Args &signal_args, CommandReturnObject &result)
156935731357SCaroline Tice     {
157035731357SCaroline Tice         TargetSP target_sp = m_interpreter.GetDebugger().GetSelectedTarget();
157135731357SCaroline Tice 
157235731357SCaroline Tice         if (!target_sp)
157335731357SCaroline Tice         {
157435731357SCaroline Tice             result.AppendError ("No current target;"
157535731357SCaroline Tice                                 " cannot handle signals until you have a valid target and process.\n");
157635731357SCaroline Tice             result.SetStatus (eReturnStatusFailed);
157735731357SCaroline Tice             return false;
157835731357SCaroline Tice         }
157935731357SCaroline Tice 
158035731357SCaroline Tice         ProcessSP process_sp = target_sp->GetProcessSP();
158135731357SCaroline Tice 
158235731357SCaroline Tice         if (!process_sp)
158335731357SCaroline Tice         {
158435731357SCaroline Tice             result.AppendError ("No current process; cannot handle signals until you have a valid process.\n");
158535731357SCaroline Tice             result.SetStatus (eReturnStatusFailed);
158635731357SCaroline Tice             return false;
158735731357SCaroline Tice         }
158835731357SCaroline Tice 
158935731357SCaroline Tice         int stop_action = -1;   // -1 means leave the current setting alone
159035731357SCaroline Tice         int pass_action = -1;   // -1 means leave the current setting alone
159135731357SCaroline Tice         int notify_action = -1; // -1 means leave the current setting alone
159235731357SCaroline Tice 
159335731357SCaroline Tice         if (! m_options.stop.empty()
159410ad7993SCaroline Tice             && ! VerifyCommandOptionValue (m_options.stop, stop_action))
159535731357SCaroline Tice         {
159635731357SCaroline Tice             result.AppendError ("Invalid argument for command option --stop; must be true or false.\n");
159735731357SCaroline Tice             result.SetStatus (eReturnStatusFailed);
159835731357SCaroline Tice             return false;
159935731357SCaroline Tice         }
160035731357SCaroline Tice 
160135731357SCaroline Tice         if (! m_options.notify.empty()
160210ad7993SCaroline Tice             && ! VerifyCommandOptionValue (m_options.notify, notify_action))
160335731357SCaroline Tice         {
160435731357SCaroline Tice             result.AppendError ("Invalid argument for command option --notify; must be true or false.\n");
160535731357SCaroline Tice             result.SetStatus (eReturnStatusFailed);
160635731357SCaroline Tice             return false;
160735731357SCaroline Tice         }
160835731357SCaroline Tice 
160935731357SCaroline Tice         if (! m_options.pass.empty()
161010ad7993SCaroline Tice             && ! VerifyCommandOptionValue (m_options.pass, pass_action))
161135731357SCaroline Tice         {
161235731357SCaroline Tice             result.AppendError ("Invalid argument for command option --pass; must be true or false.\n");
161335731357SCaroline Tice             result.SetStatus (eReturnStatusFailed);
161435731357SCaroline Tice             return false;
161535731357SCaroline Tice         }
161635731357SCaroline Tice 
161735731357SCaroline Tice         size_t num_args = signal_args.GetArgumentCount();
161835731357SCaroline Tice         UnixSignals &signals = process_sp->GetUnixSignals();
161935731357SCaroline Tice         int num_signals_set = 0;
162035731357SCaroline Tice 
162110ad7993SCaroline Tice         if (num_args > 0)
162210ad7993SCaroline Tice         {
162335731357SCaroline Tice             for (size_t i = 0; i < num_args; ++i)
162435731357SCaroline Tice             {
162535731357SCaroline Tice                 int32_t signo = signals.GetSignalNumberFromName (signal_args.GetArgumentAtIndex (i));
162635731357SCaroline Tice                 if (signo != LLDB_INVALID_SIGNAL_NUMBER)
162735731357SCaroline Tice                 {
162810ad7993SCaroline Tice                     // Casting the actions as bools here should be okay, because VerifyCommandOptionValue guarantees
162935731357SCaroline Tice                     // the value is either 0 or 1.
163035731357SCaroline Tice                     if (stop_action != -1)
163135731357SCaroline Tice                         signals.SetShouldStop (signo, (bool) stop_action);
163235731357SCaroline Tice                     if (pass_action != -1)
163335731357SCaroline Tice                     {
163410ad7993SCaroline Tice                         bool suppress = ! ((bool) pass_action);
163510ad7993SCaroline Tice                         signals.SetShouldSuppress (signo, suppress);
163635731357SCaroline Tice                     }
163735731357SCaroline Tice                     if (notify_action != -1)
163835731357SCaroline Tice                         signals.SetShouldNotify (signo, (bool) notify_action);
163935731357SCaroline Tice                     ++num_signals_set;
164035731357SCaroline Tice                 }
164135731357SCaroline Tice                 else
164235731357SCaroline Tice                 {
164335731357SCaroline Tice                     result.AppendErrorWithFormat ("Invalid signal name '%s'\n", signal_args.GetArgumentAtIndex (i));
164435731357SCaroline Tice                 }
164535731357SCaroline Tice             }
164610ad7993SCaroline Tice         }
164710ad7993SCaroline Tice         else
164810ad7993SCaroline Tice         {
164910ad7993SCaroline Tice             // No signal specified, if any command options were specified, update ALL signals.
165010ad7993SCaroline Tice             if ((notify_action != -1) || (stop_action != -1) || (pass_action != -1))
165110ad7993SCaroline Tice             {
165210ad7993SCaroline Tice                 if (m_interpreter.Confirm ("Do you really want to update all the signals?", false))
165310ad7993SCaroline Tice                 {
165410ad7993SCaroline Tice                     int32_t signo = signals.GetFirstSignalNumber();
165510ad7993SCaroline Tice                     while (signo != LLDB_INVALID_SIGNAL_NUMBER)
165610ad7993SCaroline Tice                     {
165710ad7993SCaroline Tice                         if (notify_action != -1)
165810ad7993SCaroline Tice                             signals.SetShouldNotify (signo, (bool) notify_action);
165910ad7993SCaroline Tice                         if (stop_action != -1)
166010ad7993SCaroline Tice                             signals.SetShouldStop (signo, (bool) stop_action);
166110ad7993SCaroline Tice                         if (pass_action != -1)
166210ad7993SCaroline Tice                         {
166310ad7993SCaroline Tice                             bool suppress = ! ((bool) pass_action);
166410ad7993SCaroline Tice                             signals.SetShouldSuppress (signo, suppress);
166510ad7993SCaroline Tice                         }
166610ad7993SCaroline Tice                         signo = signals.GetNextSignalNumber (signo);
166710ad7993SCaroline Tice                     }
166810ad7993SCaroline Tice                 }
166910ad7993SCaroline Tice             }
167010ad7993SCaroline Tice         }
167110ad7993SCaroline Tice 
167210ad7993SCaroline Tice         PrintSignalInformation (result.GetOutputStream(), signal_args, num_signals_set, signals);
167335731357SCaroline Tice 
167435731357SCaroline Tice         if (num_signals_set > 0)
167535731357SCaroline Tice             result.SetStatus (eReturnStatusSuccessFinishNoResult);
167635731357SCaroline Tice         else
167735731357SCaroline Tice             result.SetStatus (eReturnStatusFailed);
167835731357SCaroline Tice 
167935731357SCaroline Tice         return result.Succeeded();
168035731357SCaroline Tice     }
168135731357SCaroline Tice 
168235731357SCaroline Tice protected:
168335731357SCaroline Tice 
168435731357SCaroline Tice     CommandOptions m_options;
168535731357SCaroline Tice };
168635731357SCaroline Tice 
1687e0d378b3SGreg Clayton OptionDefinition
168835731357SCaroline Tice CommandObjectProcessHandle::CommandOptions::g_option_table[] =
168935731357SCaroline Tice {
169035731357SCaroline 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." },
169135731357SCaroline 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." },
169235731357SCaroline Tice { LLDB_OPT_SET_1, false, "pass",  'p', required_argument, NULL, 0, eArgTypeBoolean, "Whether or not the signal should be passed to the process." },
169335731357SCaroline Tice { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL }
169435731357SCaroline Tice };
169535731357SCaroline Tice 
169635731357SCaroline Tice //-------------------------------------------------------------------------
169730fdc8d8SChris Lattner // CommandObjectMultiwordProcess
169830fdc8d8SChris Lattner //-------------------------------------------------------------------------
169930fdc8d8SChris Lattner 
17006611103cSGreg Clayton CommandObjectMultiwordProcess::CommandObjectMultiwordProcess (CommandInterpreter &interpreter) :
1701a7015092SGreg Clayton     CommandObjectMultiword (interpreter,
1702a7015092SGreg Clayton                             "process",
170330fdc8d8SChris Lattner                             "A set of commands for operating on a process.",
170430fdc8d8SChris Lattner                             "process <subcommand> [<subcommand-options>]")
170530fdc8d8SChris Lattner {
1706a7015092SGreg Clayton     LoadSubCommand ("attach",      CommandObjectSP (new CommandObjectProcessAttach    (interpreter)));
1707a7015092SGreg Clayton     LoadSubCommand ("launch",      CommandObjectSP (new CommandObjectProcessLaunch    (interpreter)));
1708a7015092SGreg Clayton     LoadSubCommand ("continue",    CommandObjectSP (new CommandObjectProcessContinue  (interpreter)));
1709b766a73dSGreg Clayton     LoadSubCommand ("connect",     CommandObjectSP (new CommandObjectProcessConnect   (interpreter)));
1710a7015092SGreg Clayton     LoadSubCommand ("detach",      CommandObjectSP (new CommandObjectProcessDetach    (interpreter)));
17118f343b09SGreg Clayton     LoadSubCommand ("load",        CommandObjectSP (new CommandObjectProcessLoad      (interpreter)));
17128f343b09SGreg Clayton     LoadSubCommand ("unload",      CommandObjectSP (new CommandObjectProcessUnload    (interpreter)));
1713a7015092SGreg Clayton     LoadSubCommand ("signal",      CommandObjectSP (new CommandObjectProcessSignal    (interpreter)));
171435731357SCaroline Tice     LoadSubCommand ("handle",      CommandObjectSP (new CommandObjectProcessHandle    (interpreter)));
1715a7015092SGreg Clayton     LoadSubCommand ("status",      CommandObjectSP (new CommandObjectProcessStatus    (interpreter)));
1716a7015092SGreg Clayton     LoadSubCommand ("interrupt",   CommandObjectSP (new CommandObjectProcessInterrupt (interpreter)));
1717a7015092SGreg Clayton     LoadSubCommand ("kill",        CommandObjectSP (new CommandObjectProcessKill      (interpreter)));
171830fdc8d8SChris Lattner }
171930fdc8d8SChris Lattner 
172030fdc8d8SChris Lattner CommandObjectMultiwordProcess::~CommandObjectMultiwordProcess ()
172130fdc8d8SChris Lattner {
172230fdc8d8SChris Lattner }
172330fdc8d8SChris Lattner 
1724