130fdc8d8SChris Lattner //===-- CommandObjectBreakpoint.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 "CommandObjectBreakpoint.h"
1130fdc8d8SChris Lattner #include "CommandObjectBreakpointCommand.h"
1230fdc8d8SChris Lattner 
1330fdc8d8SChris Lattner // C Includes
1430fdc8d8SChris Lattner // C++ Includes
1530fdc8d8SChris Lattner // Other libraries and framework includes
1630fdc8d8SChris Lattner // Project includes
1730fdc8d8SChris Lattner #include "lldb/Breakpoint/Breakpoint.h"
1830fdc8d8SChris Lattner #include "lldb/Breakpoint/BreakpointIDList.h"
1930fdc8d8SChris Lattner #include "lldb/Breakpoint/BreakpointLocation.h"
2040af72e1SJim Ingham #include "lldb/Interpreter/Options.h"
2130fdc8d8SChris Lattner #include "lldb/Core/RegularExpression.h"
2230fdc8d8SChris Lattner #include "lldb/Core/StreamString.h"
2330fdc8d8SChris Lattner #include "lldb/Interpreter/CommandInterpreter.h"
2430fdc8d8SChris Lattner #include "lldb/Interpreter/CommandReturnObject.h"
2530fdc8d8SChris Lattner #include "lldb/Target/Target.h"
2630fdc8d8SChris Lattner #include "lldb/Interpreter/CommandCompletions.h"
2730fdc8d8SChris Lattner #include "lldb/Target/StackFrame.h"
281b54c88cSJim Ingham #include "lldb/Target/Thread.h"
291b54c88cSJim Ingham #include "lldb/Target/ThreadSpec.h"
3030fdc8d8SChris Lattner 
31b7234e40SJohnny Chen #include <vector>
32b7234e40SJohnny Chen 
3330fdc8d8SChris Lattner using namespace lldb;
3430fdc8d8SChris Lattner using namespace lldb_private;
3530fdc8d8SChris Lattner 
3630fdc8d8SChris Lattner static void
3785e8b814SJim Ingham AddBreakpointDescription (Stream *s, Breakpoint *bp, lldb::DescriptionLevel level)
3830fdc8d8SChris Lattner {
3930fdc8d8SChris Lattner     s->IndentMore();
4030fdc8d8SChris Lattner     bp->GetDescription (s, level, true);
4130fdc8d8SChris Lattner     s->IndentLess();
4230fdc8d8SChris Lattner     s->EOL();
4330fdc8d8SChris Lattner }
4430fdc8d8SChris Lattner 
4530fdc8d8SChris Lattner //-------------------------------------------------------------------------
4630fdc8d8SChris Lattner // CommandObjectBreakpointSet::CommandOptions
4730fdc8d8SChris Lattner //-------------------------------------------------------------------------
48ae1c4cf5SJim Ingham #pragma mark Set::CommandOptions
4930fdc8d8SChris Lattner 
50eb0103f2SGreg Clayton CommandObjectBreakpointSet::CommandOptions::CommandOptions(CommandInterpreter &interpreter) :
51eb0103f2SGreg Clayton     Options (interpreter),
5230fdc8d8SChris Lattner     m_filename (),
5330fdc8d8SChris Lattner     m_line_num (0),
5430fdc8d8SChris Lattner     m_column (0),
552856d462SGreg Clayton     m_check_inlines (true),
5630fdc8d8SChris Lattner     m_func_name (),
570c5cd90dSGreg Clayton     m_func_name_type_mask (0),
5830fdc8d8SChris Lattner     m_func_regexp (),
5930fdc8d8SChris Lattner     m_modules (),
601b54c88cSJim Ingham     m_load_addr(),
61c982c768SGreg Clayton     m_ignore_count (0),
621b54c88cSJim Ingham     m_thread_id(LLDB_INVALID_THREAD_ID),
63c982c768SGreg Clayton     m_thread_index (UINT32_MAX),
641b54c88cSJim Ingham     m_thread_name(),
65c982c768SGreg Clayton     m_queue_name()
6630fdc8d8SChris Lattner {
6730fdc8d8SChris Lattner }
6830fdc8d8SChris Lattner 
6930fdc8d8SChris Lattner CommandObjectBreakpointSet::CommandOptions::~CommandOptions ()
7030fdc8d8SChris Lattner {
7130fdc8d8SChris Lattner }
7230fdc8d8SChris Lattner 
73e0d378b3SGreg Clayton OptionDefinition
7430fdc8d8SChris Lattner CommandObjectBreakpointSet::CommandOptions::g_option_table[] =
7530fdc8d8SChris Lattner {
76deaab222SCaroline Tice     { LLDB_OPT_SET_ALL, false, "shlib", 's', required_argument, NULL, CommandCompletions::eModuleCompletion, eArgTypeShlibName,
778651121cSJim Ingham         "Set the breakpoint only in this shared library (can use this option multiple times for multiple shlibs)."},
788651121cSJim Ingham 
79deaab222SCaroline Tice     { LLDB_OPT_SET_ALL, false, "ignore-count", 'i', required_argument,   NULL, 0, eArgTypeCount,
80deaab222SCaroline Tice         "Set the number of times this breakpoint is skipped before stopping." },
811b54c88cSJim Ingham 
82deaab222SCaroline Tice     { LLDB_OPT_SET_ALL, false, "thread-index", 'x', required_argument, NULL, NULL, eArgTypeThreadIndex,
83ed8a705cSGreg Clayton         "The breakpoint stops only for the thread whose index matches this argument."},
841b54c88cSJim Ingham 
85deaab222SCaroline Tice     { LLDB_OPT_SET_ALL, false, "thread-id", 't', required_argument, NULL, NULL, eArgTypeThreadID,
861b54c88cSJim Ingham         "The breakpoint stops only for the thread whose TID matches this argument."},
871b54c88cSJim Ingham 
88deaab222SCaroline Tice     { LLDB_OPT_SET_ALL, false, "thread-name", 'T', required_argument, NULL, NULL, eArgTypeThreadName,
891b54c88cSJim Ingham         "The breakpoint stops only for the thread whose thread name matches this argument."},
901b54c88cSJim Ingham 
91deaab222SCaroline Tice     { LLDB_OPT_SET_ALL, false, "queue-name", 'q', required_argument, NULL, NULL, eArgTypeQueueName,
921b54c88cSJim Ingham         "The breakpoint stops only for threads in the queue whose name is given by this argument."},
931b54c88cSJim Ingham 
94deaab222SCaroline Tice     { LLDB_OPT_SET_1, false, "file", 'f', required_argument, NULL, CommandCompletions::eSourceFileCompletion, eArgTypeFilename,
9530fdc8d8SChris Lattner         "Set the breakpoint by source location in this particular file."},
9630fdc8d8SChris Lattner 
97deaab222SCaroline Tice     { LLDB_OPT_SET_1, true, "line", 'l', required_argument, NULL, 0, eArgTypeLineNum,
9830fdc8d8SChris Lattner         "Set the breakpoint by source location at this particular line."},
9930fdc8d8SChris Lattner 
10030fdc8d8SChris Lattner     // Comment out this option for the moment, as we don't actually use it, but will in the future.
10130fdc8d8SChris Lattner     // This way users won't see it, but the infrastructure is left in place.
10230fdc8d8SChris Lattner     //    { 0, false, "column",     'c', required_argument, NULL, "<column>",
10330fdc8d8SChris Lattner     //    "Set the breakpoint by source location at this particular column."},
10430fdc8d8SChris Lattner 
105deaab222SCaroline Tice     { LLDB_OPT_SET_2, true, "address", 'a', required_argument, NULL, 0, eArgTypeAddress,
10630fdc8d8SChris Lattner         "Set the breakpoint by address, at the specified address."},
10730fdc8d8SChris Lattner 
108deaab222SCaroline Tice     { LLDB_OPT_SET_3, true, "name", 'n', required_argument, NULL, CommandCompletions::eSymbolCompletion, eArgTypeFunctionName,
109e02b8504SGreg Clayton         "Set the breakpoint by function name." },
11030fdc8d8SChris Lattner 
111deaab222SCaroline Tice     { LLDB_OPT_SET_4, true, "fullname", 'F', required_argument, NULL, CommandCompletions::eSymbolCompletion, eArgTypeFullName,
1122561aa61SJim Ingham         "Set the breakpoint by fully qualified function names. For C++ this means namespaces and all arguemnts, and "
1132561aa61SJim Ingham         "for Objective C this means a full function prototype with class and selector." },
1140c5cd90dSGreg Clayton 
115deaab222SCaroline Tice     { LLDB_OPT_SET_5, true, "selector", 'S', required_argument, NULL, 0, eArgTypeSelector,
1162561aa61SJim Ingham         "Set the breakpoint by ObjC selector name." },
1170c5cd90dSGreg Clayton 
118deaab222SCaroline Tice     { LLDB_OPT_SET_6, true, "method", 'M', required_argument, NULL, 0, eArgTypeMethod,
1192561aa61SJim Ingham         "Set the breakpoint by C++ method names." },
1200c5cd90dSGreg Clayton 
121deaab222SCaroline Tice     { LLDB_OPT_SET_7, true, "func-regex", 'r', required_argument, NULL, 0, eArgTypeRegularExpression,
12230fdc8d8SChris Lattner         "Set the breakpoint by function name, evaluating a regular-expression to find the function name(s)." },
12330fdc8d8SChris Lattner 
124e02b8504SGreg Clayton     { LLDB_OPT_SET_8, true, "basename", 'b', required_argument, NULL, CommandCompletions::eSymbolCompletion, eArgTypeFunctionName,
125e02b8504SGreg Clayton         "Set the breakpoint by function basename (C++ namespaces and arguments will be ignored)." },
126e02b8504SGreg Clayton 
127deaab222SCaroline Tice     { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL }
12830fdc8d8SChris Lattner };
12930fdc8d8SChris Lattner 
130e0d378b3SGreg Clayton const OptionDefinition*
13130fdc8d8SChris Lattner CommandObjectBreakpointSet::CommandOptions::GetDefinitions ()
13230fdc8d8SChris Lattner {
13330fdc8d8SChris Lattner     return g_option_table;
13430fdc8d8SChris Lattner }
13530fdc8d8SChris Lattner 
13630fdc8d8SChris Lattner Error
137f6b8b581SGreg Clayton CommandObjectBreakpointSet::CommandOptions::SetOptionValue (uint32_t option_idx, const char *option_arg)
13830fdc8d8SChris Lattner {
13930fdc8d8SChris Lattner     Error error;
14030fdc8d8SChris Lattner     char short_option = (char) m_getopt_table[option_idx].val;
14130fdc8d8SChris Lattner 
14230fdc8d8SChris Lattner     switch (short_option)
14330fdc8d8SChris Lattner     {
14430fdc8d8SChris Lattner         case 'a':
1450292f4a5SJim Ingham             m_load_addr = Args::StringToUInt64(option_arg, LLDB_INVALID_ADDRESS, 0);
14630fdc8d8SChris Lattner             if (m_load_addr == LLDB_INVALID_ADDRESS)
1470292f4a5SJim Ingham                 m_load_addr = Args::StringToUInt64(option_arg, LLDB_INVALID_ADDRESS, 16);
14830fdc8d8SChris Lattner 
14930fdc8d8SChris Lattner             if (m_load_addr == LLDB_INVALID_ADDRESS)
1500292f4a5SJim Ingham                 error.SetErrorStringWithFormat ("Invalid address string '%s'.\n", option_arg);
15130fdc8d8SChris Lattner             break;
15230fdc8d8SChris Lattner 
15330fdc8d8SChris Lattner         case 'c':
15430fdc8d8SChris Lattner             m_column = Args::StringToUInt32 (option_arg, 0);
15530fdc8d8SChris Lattner             break;
1560c5cd90dSGreg Clayton 
15730fdc8d8SChris Lattner         case 'f':
158357132ebSGreg Clayton             m_filename.assign (option_arg);
15930fdc8d8SChris Lattner             break;
1600c5cd90dSGreg Clayton 
16130fdc8d8SChris Lattner         case 'l':
16230fdc8d8SChris Lattner             m_line_num = Args::StringToUInt32 (option_arg, 0);
16330fdc8d8SChris Lattner             break;
1640c5cd90dSGreg Clayton 
165e02b8504SGreg Clayton         case 'b':
166357132ebSGreg Clayton             m_func_name.assign (option_arg);
1670c5cd90dSGreg Clayton             m_func_name_type_mask |= eFunctionNameTypeBase;
1680c5cd90dSGreg Clayton             break;
1690c5cd90dSGreg Clayton 
170e02b8504SGreg Clayton         case 'n':
171357132ebSGreg Clayton             m_func_name.assign (option_arg);
172e02b8504SGreg Clayton             m_func_name_type_mask |= eFunctionNameTypeAuto;
173e02b8504SGreg Clayton             break;
174e02b8504SGreg Clayton 
1750c5cd90dSGreg Clayton         case 'F':
176357132ebSGreg Clayton             m_func_name.assign (option_arg);
1770c5cd90dSGreg Clayton             m_func_name_type_mask |= eFunctionNameTypeFull;
1780c5cd90dSGreg Clayton             break;
1790c5cd90dSGreg Clayton 
1800c5cd90dSGreg Clayton         case 'S':
181357132ebSGreg Clayton             m_func_name.assign (option_arg);
1820c5cd90dSGreg Clayton             m_func_name_type_mask |= eFunctionNameTypeSelector;
1830c5cd90dSGreg Clayton             break;
1840c5cd90dSGreg Clayton 
1852561aa61SJim Ingham         case 'M':
186357132ebSGreg Clayton             m_func_name.assign (option_arg);
1870c5cd90dSGreg Clayton             m_func_name_type_mask |= eFunctionNameTypeMethod;
1880c5cd90dSGreg Clayton             break;
1890c5cd90dSGreg Clayton 
19030fdc8d8SChris Lattner         case 'r':
191357132ebSGreg Clayton             m_func_regexp.assign (option_arg);
19230fdc8d8SChris Lattner             break;
1930c5cd90dSGreg Clayton 
19430fdc8d8SChris Lattner         case 's':
19530fdc8d8SChris Lattner             {
19630fdc8d8SChris Lattner                 m_modules.push_back (std::string (option_arg));
19730fdc8d8SChris Lattner                 break;
19830fdc8d8SChris Lattner             }
199ed8a705cSGreg Clayton         case 'i':
2001b54c88cSJim Ingham         {
2010292f4a5SJim Ingham             m_ignore_count = Args::StringToUInt32(option_arg, UINT32_MAX, 0);
202c982c768SGreg Clayton             if (m_ignore_count == UINT32_MAX)
2030292f4a5SJim Ingham                error.SetErrorStringWithFormat ("Invalid ignore count '%s'.\n", option_arg);
2041b54c88cSJim Ingham         }
205ae1c4cf5SJim Ingham         break;
2061b54c88cSJim Ingham         case 't' :
2071b54c88cSJim Ingham         {
2080292f4a5SJim Ingham             m_thread_id = Args::StringToUInt64(option_arg, LLDB_INVALID_THREAD_ID, 0);
2091b54c88cSJim Ingham             if (m_thread_id == LLDB_INVALID_THREAD_ID)
2100292f4a5SJim Ingham                error.SetErrorStringWithFormat ("Invalid thread id string '%s'.\n", option_arg);
2111b54c88cSJim Ingham         }
2121b54c88cSJim Ingham         break;
2131b54c88cSJim Ingham         case 'T':
214357132ebSGreg Clayton             m_thread_name.assign (option_arg);
2151b54c88cSJim Ingham             break;
2161b54c88cSJim Ingham         case 'q':
217357132ebSGreg Clayton             m_queue_name.assign (option_arg);
2181b54c88cSJim Ingham             break;
2191b54c88cSJim Ingham         case 'x':
2201b54c88cSJim Ingham         {
2210292f4a5SJim Ingham             m_thread_index = Args::StringToUInt32(option_arg, UINT32_MAX, 0);
222c982c768SGreg Clayton             if (m_thread_id == UINT32_MAX)
2230292f4a5SJim Ingham                error.SetErrorStringWithFormat ("Invalid thread index string '%s'.\n", option_arg);
2241b54c88cSJim Ingham 
2251b54c88cSJim Ingham         }
2261b54c88cSJim Ingham         break;
22730fdc8d8SChris Lattner         default:
22830fdc8d8SChris Lattner             error.SetErrorStringWithFormat ("Unrecognized option '%c'.\n", short_option);
22930fdc8d8SChris Lattner             break;
23030fdc8d8SChris Lattner     }
23130fdc8d8SChris Lattner 
23230fdc8d8SChris Lattner     return error;
23330fdc8d8SChris Lattner }
23430fdc8d8SChris Lattner 
23530fdc8d8SChris Lattner void
236f6b8b581SGreg Clayton CommandObjectBreakpointSet::CommandOptions::OptionParsingStarting ()
23730fdc8d8SChris Lattner {
23830fdc8d8SChris Lattner     m_filename.clear();
23930fdc8d8SChris Lattner     m_line_num = 0;
24030fdc8d8SChris Lattner     m_column = 0;
24130fdc8d8SChris Lattner     m_func_name.clear();
2420c5cd90dSGreg Clayton     m_func_name_type_mask = 0;
24330fdc8d8SChris Lattner     m_func_regexp.clear();
24430fdc8d8SChris Lattner     m_load_addr = LLDB_INVALID_ADDRESS;
24530fdc8d8SChris Lattner     m_modules.clear();
246c982c768SGreg Clayton     m_ignore_count = 0;
2471b54c88cSJim Ingham     m_thread_id = LLDB_INVALID_THREAD_ID;
248c982c768SGreg Clayton     m_thread_index = UINT32_MAX;
2491b54c88cSJim Ingham     m_thread_name.clear();
2501b54c88cSJim Ingham     m_queue_name.clear();
25130fdc8d8SChris Lattner }
25230fdc8d8SChris Lattner 
25330fdc8d8SChris Lattner //-------------------------------------------------------------------------
25430fdc8d8SChris Lattner // CommandObjectBreakpointSet
25530fdc8d8SChris Lattner //-------------------------------------------------------------------------
256ae1c4cf5SJim Ingham #pragma mark Set
25730fdc8d8SChris Lattner 
258a7015092SGreg Clayton CommandObjectBreakpointSet::CommandObjectBreakpointSet (CommandInterpreter &interpreter) :
259a7015092SGreg Clayton     CommandObject (interpreter,
260a7015092SGreg Clayton                    "breakpoint set",
261a7015092SGreg Clayton                    "Sets a breakpoint or set of breakpoints in the executable.",
262eb0103f2SGreg Clayton                    "breakpoint set <cmd-options>"),
263eb0103f2SGreg Clayton     m_options (interpreter)
26430fdc8d8SChris Lattner {
26530fdc8d8SChris Lattner }
26630fdc8d8SChris Lattner 
26730fdc8d8SChris Lattner CommandObjectBreakpointSet::~CommandObjectBreakpointSet ()
26830fdc8d8SChris Lattner {
26930fdc8d8SChris Lattner }
27030fdc8d8SChris Lattner 
27130fdc8d8SChris Lattner Options *
27230fdc8d8SChris Lattner CommandObjectBreakpointSet::GetOptions ()
27330fdc8d8SChris Lattner {
27430fdc8d8SChris Lattner     return &m_options;
27530fdc8d8SChris Lattner }
27630fdc8d8SChris Lattner 
27730fdc8d8SChris Lattner bool
27830fdc8d8SChris Lattner CommandObjectBreakpointSet::Execute
27930fdc8d8SChris Lattner (
28030fdc8d8SChris Lattner     Args& command,
28130fdc8d8SChris Lattner     CommandReturnObject &result
28230fdc8d8SChris Lattner )
28330fdc8d8SChris Lattner {
284a7015092SGreg Clayton     Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get();
28530fdc8d8SChris Lattner     if (target == NULL)
28630fdc8d8SChris Lattner     {
287effe5c95SGreg Clayton         result.AppendError ("Invalid target.  Must set target before setting breakpoints (see 'target create' command).");
28830fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
28930fdc8d8SChris Lattner         return false;
29030fdc8d8SChris Lattner     }
29130fdc8d8SChris Lattner 
29230fdc8d8SChris Lattner     // The following are the various types of breakpoints that could be set:
29330fdc8d8SChris Lattner     //   1).  -f -l -p  [-s -g]   (setting breakpoint by source location)
29430fdc8d8SChris Lattner     //   2).  -a  [-s -g]         (setting breakpoint by address)
29530fdc8d8SChris Lattner     //   3).  -n  [-s -g]         (setting breakpoint by function name)
29630fdc8d8SChris Lattner     //   4).  -r  [-s -g]         (setting breakpoint by function name regular expression)
29730fdc8d8SChris Lattner 
29830fdc8d8SChris Lattner     BreakpointSetType break_type = eSetTypeInvalid;
29930fdc8d8SChris Lattner 
30030fdc8d8SChris Lattner     if (m_options.m_line_num != 0)
30130fdc8d8SChris Lattner         break_type = eSetTypeFileAndLine;
30230fdc8d8SChris Lattner     else if (m_options.m_load_addr != LLDB_INVALID_ADDRESS)
30330fdc8d8SChris Lattner         break_type = eSetTypeAddress;
30430fdc8d8SChris Lattner     else if (!m_options.m_func_name.empty())
30530fdc8d8SChris Lattner         break_type = eSetTypeFunctionName;
30630fdc8d8SChris Lattner     else if  (!m_options.m_func_regexp.empty())
30730fdc8d8SChris Lattner         break_type = eSetTypeFunctionRegexp;
30830fdc8d8SChris Lattner 
30930fdc8d8SChris Lattner     ModuleSP module_sp = target->GetExecutableModule();
31030fdc8d8SChris Lattner     Breakpoint *bp = NULL;
311274060b6SGreg Clayton     FileSpec module_spec;
31230fdc8d8SChris Lattner     bool use_module = false;
31330fdc8d8SChris Lattner     int num_modules = m_options.m_modules.size();
31430fdc8d8SChris Lattner 
31530fdc8d8SChris Lattner     if ((num_modules > 0) && (break_type != eSetTypeAddress))
31630fdc8d8SChris Lattner         use_module = true;
31730fdc8d8SChris Lattner 
31830fdc8d8SChris Lattner     switch (break_type)
31930fdc8d8SChris Lattner     {
32030fdc8d8SChris Lattner         case eSetTypeFileAndLine: // Breakpoint by source position
32130fdc8d8SChris Lattner             {
32230fdc8d8SChris Lattner                 FileSpec file;
32330fdc8d8SChris Lattner                 if (m_options.m_filename.empty())
32430fdc8d8SChris Lattner                 {
3258b82f087SGreg Clayton                     StackFrame *cur_frame = m_interpreter.GetExecutionContext().frame;
32630fdc8d8SChris Lattner                     if (cur_frame == NULL)
32730fdc8d8SChris Lattner                     {
32830fdc8d8SChris Lattner                         result.AppendError ("Attempting to set breakpoint by line number alone with no selected frame.");
32930fdc8d8SChris Lattner                         result.SetStatus (eReturnStatusFailed);
33030fdc8d8SChris Lattner                         break;
33130fdc8d8SChris Lattner                     }
33230fdc8d8SChris Lattner                     else if (!cur_frame->HasDebugInformation())
33330fdc8d8SChris Lattner                     {
33430fdc8d8SChris Lattner                         result.AppendError ("Attempting to set breakpoint by line number alone but selected frame has no debug info.");
33530fdc8d8SChris Lattner                         result.SetStatus (eReturnStatusFailed);
33630fdc8d8SChris Lattner                         break;
33730fdc8d8SChris Lattner                     }
33830fdc8d8SChris Lattner                     else
33930fdc8d8SChris Lattner                     {
340176761e5SGreg Clayton                         const SymbolContext &sc = cur_frame->GetSymbolContext (eSymbolContextLineEntry);
341176761e5SGreg Clayton                         if (sc.line_entry.file)
34230fdc8d8SChris Lattner                         {
343176761e5SGreg Clayton                             file = sc.line_entry.file;
34430fdc8d8SChris Lattner                         }
34530fdc8d8SChris Lattner                         else
34630fdc8d8SChris Lattner                         {
34730fdc8d8SChris Lattner                             result.AppendError ("Attempting to set breakpoint by line number alone but can't find the file for the selected frame.");
34830fdc8d8SChris Lattner                             result.SetStatus (eReturnStatusFailed);
34930fdc8d8SChris Lattner                             break;
35030fdc8d8SChris Lattner                         }
35130fdc8d8SChris Lattner                     }
35230fdc8d8SChris Lattner                 }
35330fdc8d8SChris Lattner                 else
35430fdc8d8SChris Lattner                 {
355274060b6SGreg Clayton                     file.SetFile(m_options.m_filename.c_str(), false);
35630fdc8d8SChris Lattner                 }
35730fdc8d8SChris Lattner 
35830fdc8d8SChris Lattner                 if (use_module)
35930fdc8d8SChris Lattner                 {
36030fdc8d8SChris Lattner                     for (int i = 0; i < num_modules; ++i)
36130fdc8d8SChris Lattner                     {
362274060b6SGreg Clayton                         module_spec.SetFile(m_options.m_modules[i].c_str(), false);
363274060b6SGreg Clayton                         bp = target->CreateBreakpoint (&module_spec,
36430fdc8d8SChris Lattner                                                        file,
36530fdc8d8SChris Lattner                                                        m_options.m_line_num,
3662856d462SGreg Clayton                                                        m_options.m_check_inlines).get();
36730fdc8d8SChris Lattner                         if (bp)
36830fdc8d8SChris Lattner                         {
36985e8b814SJim Ingham                             Stream &output_stream = result.GetOutputStream();
37085e8b814SJim Ingham                             result.AppendMessage ("Breakpoint created: ");
37130fdc8d8SChris Lattner                             bp->GetDescription(&output_stream, lldb::eDescriptionLevelBrief);
37230fdc8d8SChris Lattner                             output_stream.EOL();
373be484f41SCaroline Tice                             if (bp->GetNumLocations() == 0)
374be484f41SCaroline Tice                                 output_stream.Printf ("WARNING:  Unable to resolve breakpoint to any actual"
375be484f41SCaroline Tice                                                       " locations.\n");
37630fdc8d8SChris Lattner                             result.SetStatus (eReturnStatusSuccessFinishResult);
37730fdc8d8SChris Lattner                         }
37830fdc8d8SChris Lattner                         else
37930fdc8d8SChris Lattner                         {
38030fdc8d8SChris Lattner                             result.AppendErrorWithFormat("Breakpoint creation failed: No breakpoint created in module '%s'.\n",
38130fdc8d8SChris Lattner                                                         m_options.m_modules[i].c_str());
38230fdc8d8SChris Lattner                             result.SetStatus (eReturnStatusFailed);
38330fdc8d8SChris Lattner                         }
38430fdc8d8SChris Lattner                     }
38530fdc8d8SChris Lattner                 }
38630fdc8d8SChris Lattner                 else
38730fdc8d8SChris Lattner                     bp = target->CreateBreakpoint (NULL,
38830fdc8d8SChris Lattner                                                    file,
38930fdc8d8SChris Lattner                                                    m_options.m_line_num,
3902856d462SGreg Clayton                                                    m_options.m_check_inlines).get();
39130fdc8d8SChris Lattner             }
39230fdc8d8SChris Lattner             break;
3936eee5aa0SGreg Clayton 
39430fdc8d8SChris Lattner         case eSetTypeAddress: // Breakpoint by address
39530fdc8d8SChris Lattner             bp = target->CreateBreakpoint (m_options.m_load_addr, false).get();
39630fdc8d8SChris Lattner             break;
3970c5cd90dSGreg Clayton 
39830fdc8d8SChris Lattner         case eSetTypeFunctionName: // Breakpoint by function name
3990c5cd90dSGreg Clayton             {
4000c5cd90dSGreg Clayton                 uint32_t name_type_mask = m_options.m_func_name_type_mask;
4010c5cd90dSGreg Clayton 
4020c5cd90dSGreg Clayton                 if (name_type_mask == 0)
403e02b8504SGreg Clayton                     name_type_mask = eFunctionNameTypeAuto;
4040c5cd90dSGreg Clayton 
40530fdc8d8SChris Lattner                 if (use_module)
40630fdc8d8SChris Lattner                 {
40730fdc8d8SChris Lattner                     for (int i = 0; i < num_modules; ++i)
40830fdc8d8SChris Lattner                     {
409274060b6SGreg Clayton                         module_spec.SetFile(m_options.m_modules[i].c_str(), false);
410274060b6SGreg Clayton                         bp = target->CreateBreakpoint (&module_spec,
411274060b6SGreg Clayton                                                        m_options.m_func_name.c_str(),
412274060b6SGreg Clayton                                                        name_type_mask,
413274060b6SGreg Clayton                                                        Breakpoint::Exact).get();
41430fdc8d8SChris Lattner                         if (bp)
41530fdc8d8SChris Lattner                         {
41685e8b814SJim Ingham                             Stream &output_stream = result.GetOutputStream();
41730fdc8d8SChris Lattner                             output_stream.Printf ("Breakpoint created: ");
41830fdc8d8SChris Lattner                             bp->GetDescription(&output_stream, lldb::eDescriptionLevelBrief);
41930fdc8d8SChris Lattner                             output_stream.EOL();
420be484f41SCaroline Tice                             if (bp->GetNumLocations() == 0)
421be484f41SCaroline Tice                                 output_stream.Printf ("WARNING:  Unable to resolve breakpoint to any actual"
422be484f41SCaroline Tice                                                       " locations.\n");
42330fdc8d8SChris Lattner                             result.SetStatus (eReturnStatusSuccessFinishResult);
42430fdc8d8SChris Lattner                         }
42530fdc8d8SChris Lattner                         else
42630fdc8d8SChris Lattner                         {
42730fdc8d8SChris Lattner                             result.AppendErrorWithFormat("Breakpoint creation failed: No breakpoint created in module '%s'.\n",
42830fdc8d8SChris Lattner                                                         m_options.m_modules[i].c_str());
42930fdc8d8SChris Lattner                             result.SetStatus (eReturnStatusFailed);
43030fdc8d8SChris Lattner                         }
43130fdc8d8SChris Lattner                     }
43230fdc8d8SChris Lattner                 }
43330fdc8d8SChris Lattner                 else
4340c5cd90dSGreg Clayton                     bp = target->CreateBreakpoint (NULL, m_options.m_func_name.c_str(), name_type_mask, Breakpoint::Exact).get();
4350c5cd90dSGreg Clayton             }
43630fdc8d8SChris Lattner             break;
4370c5cd90dSGreg Clayton 
43830fdc8d8SChris Lattner         case eSetTypeFunctionRegexp: // Breakpoint by regular expression function name
43930fdc8d8SChris Lattner             {
44030fdc8d8SChris Lattner                 RegularExpression regexp(m_options.m_func_regexp.c_str());
44130fdc8d8SChris Lattner                 if (use_module)
44230fdc8d8SChris Lattner                 {
44330fdc8d8SChris Lattner                     for (int i = 0; i < num_modules; ++i)
44430fdc8d8SChris Lattner                     {
445274060b6SGreg Clayton                         module_spec.SetFile(m_options.m_modules[i].c_str(), false);
446274060b6SGreg Clayton                         bp = target->CreateBreakpoint (&module_spec, regexp).get();
44730fdc8d8SChris Lattner                         if (bp)
44830fdc8d8SChris Lattner                         {
44985e8b814SJim Ingham                             Stream &output_stream = result.GetOutputStream();
45030fdc8d8SChris Lattner                             output_stream.Printf ("Breakpoint created: ");
45130fdc8d8SChris Lattner                             bp->GetDescription(&output_stream, lldb::eDescriptionLevelBrief);
45230fdc8d8SChris Lattner                             output_stream.EOL();
453be484f41SCaroline Tice                             if (bp->GetNumLocations() == 0)
454be484f41SCaroline Tice                                 output_stream.Printf ("WARNING:  Unable to resolve breakpoint to any actual"
455be484f41SCaroline Tice                                                       " locations.\n");
45630fdc8d8SChris Lattner                             result.SetStatus (eReturnStatusSuccessFinishResult);
45730fdc8d8SChris Lattner                         }
45830fdc8d8SChris Lattner                         else
45930fdc8d8SChris Lattner                         {
46030fdc8d8SChris Lattner                             result.AppendErrorWithFormat("Breakpoint creation failed: No breakpoint created in module '%s'.\n",
46130fdc8d8SChris Lattner                                                         m_options.m_modules[i].c_str());
46230fdc8d8SChris Lattner                             result.SetStatus (eReturnStatusFailed);
46330fdc8d8SChris Lattner                         }
46430fdc8d8SChris Lattner                     }
46530fdc8d8SChris Lattner                 }
46630fdc8d8SChris Lattner                 else
46730fdc8d8SChris Lattner                     bp = target->CreateBreakpoint (NULL, regexp).get();
46830fdc8d8SChris Lattner             }
46930fdc8d8SChris Lattner             break;
4700c5cd90dSGreg Clayton 
47130fdc8d8SChris Lattner         default:
47230fdc8d8SChris Lattner             break;
47330fdc8d8SChris Lattner     }
47430fdc8d8SChris Lattner 
4751b54c88cSJim Ingham     // Now set the various options that were passed in:
4761b54c88cSJim Ingham     if (bp)
4771b54c88cSJim Ingham     {
4781b54c88cSJim Ingham         if (m_options.m_thread_id != LLDB_INVALID_THREAD_ID)
4791b54c88cSJim Ingham             bp->SetThreadID (m_options.m_thread_id);
4801b54c88cSJim Ingham 
481c982c768SGreg Clayton         if (m_options.m_thread_index != UINT32_MAX)
4821b54c88cSJim Ingham             bp->GetOptions()->GetThreadSpec()->SetIndex(m_options.m_thread_index);
4831b54c88cSJim Ingham 
4841b54c88cSJim Ingham         if (!m_options.m_thread_name.empty())
4851b54c88cSJim Ingham             bp->GetOptions()->GetThreadSpec()->SetName(m_options.m_thread_name.c_str());
4861b54c88cSJim Ingham 
4871b54c88cSJim Ingham         if (!m_options.m_queue_name.empty())
4881b54c88cSJim Ingham             bp->GetOptions()->GetThreadSpec()->SetQueueName(m_options.m_queue_name.c_str());
4891b54c88cSJim Ingham 
490c982c768SGreg Clayton         if (m_options.m_ignore_count != 0)
4911b54c88cSJim Ingham             bp->GetOptions()->SetIgnoreCount(m_options.m_ignore_count);
4921b54c88cSJim Ingham     }
4931b54c88cSJim Ingham 
49430fdc8d8SChris Lattner     if (bp && !use_module)
49530fdc8d8SChris Lattner     {
49685e8b814SJim Ingham         Stream &output_stream = result.GetOutputStream();
49730fdc8d8SChris Lattner         output_stream.Printf ("Breakpoint created: ");
49830fdc8d8SChris Lattner         bp->GetDescription(&output_stream, lldb::eDescriptionLevelBrief);
49930fdc8d8SChris Lattner         output_stream.EOL();
500be484f41SCaroline Tice         if (bp->GetNumLocations() == 0)
501be484f41SCaroline Tice             output_stream.Printf ("WARNING:  Unable to resolve breakpoint to any actual locations.\n");
50230fdc8d8SChris Lattner         result.SetStatus (eReturnStatusSuccessFinishResult);
50330fdc8d8SChris Lattner     }
50430fdc8d8SChris Lattner     else if (!bp)
50530fdc8d8SChris Lattner     {
50630fdc8d8SChris Lattner         result.AppendError ("Breakpoint creation failed: No breakpoint created.");
50730fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
50830fdc8d8SChris Lattner     }
50930fdc8d8SChris Lattner 
51030fdc8d8SChris Lattner     return result.Succeeded();
51130fdc8d8SChris Lattner }
51230fdc8d8SChris Lattner 
51330fdc8d8SChris Lattner //-------------------------------------------------------------------------
51430fdc8d8SChris Lattner // CommandObjectMultiwordBreakpoint
51530fdc8d8SChris Lattner //-------------------------------------------------------------------------
516ae1c4cf5SJim Ingham #pragma mark MultiwordBreakpoint
51730fdc8d8SChris Lattner 
5186611103cSGreg Clayton CommandObjectMultiwordBreakpoint::CommandObjectMultiwordBreakpoint (CommandInterpreter &interpreter) :
519a7015092SGreg Clayton     CommandObjectMultiword (interpreter,
520a7015092SGreg Clayton                             "breakpoint",
521*46fbc60fSJim Ingham                             "A set of commands for operating on breakpoints. Also see _regexp-break.",
52230fdc8d8SChris Lattner                             "breakpoint <command> [<command-options>]")
52330fdc8d8SChris Lattner {
52430fdc8d8SChris Lattner     bool status;
52530fdc8d8SChris Lattner 
526a7015092SGreg Clayton     CommandObjectSP list_command_object (new CommandObjectBreakpointList (interpreter));
527a7015092SGreg Clayton     CommandObjectSP enable_command_object (new CommandObjectBreakpointEnable (interpreter));
528a7015092SGreg Clayton     CommandObjectSP disable_command_object (new CommandObjectBreakpointDisable (interpreter));
529b7234e40SJohnny Chen     CommandObjectSP clear_command_object (new CommandObjectBreakpointClear (interpreter));
530b7234e40SJohnny Chen     CommandObjectSP delete_command_object (new CommandObjectBreakpointDelete (interpreter));
531a7015092SGreg Clayton     CommandObjectSP set_command_object (new CommandObjectBreakpointSet (interpreter));
53230fdc8d8SChris Lattner     CommandObjectSP command_command_object (new CommandObjectBreakpointCommand (interpreter));
533a7015092SGreg Clayton     CommandObjectSP modify_command_object (new CommandObjectBreakpointModify(interpreter));
53430fdc8d8SChris Lattner 
535b7234e40SJohnny Chen     list_command_object->SetCommandName ("breakpoint list");
53630fdc8d8SChris Lattner     enable_command_object->SetCommandName("breakpoint enable");
53730fdc8d8SChris Lattner     disable_command_object->SetCommandName("breakpoint disable");
538b7234e40SJohnny Chen     clear_command_object->SetCommandName("breakpoint clear");
539b7234e40SJohnny Chen     delete_command_object->SetCommandName("breakpoint delete");
540ae1c4cf5SJim Ingham     set_command_object->SetCommandName("breakpoint set");
541b7234e40SJohnny Chen     command_command_object->SetCommandName ("breakpoint command");
542b7234e40SJohnny Chen     modify_command_object->SetCommandName ("breakpoint modify");
54330fdc8d8SChris Lattner 
544a7015092SGreg Clayton     status = LoadSubCommand ("list",       list_command_object);
545a7015092SGreg Clayton     status = LoadSubCommand ("enable",     enable_command_object);
546a7015092SGreg Clayton     status = LoadSubCommand ("disable",    disable_command_object);
547b7234e40SJohnny Chen     status = LoadSubCommand ("clear",      clear_command_object);
548a7015092SGreg Clayton     status = LoadSubCommand ("delete",     delete_command_object);
549a7015092SGreg Clayton     status = LoadSubCommand ("set",        set_command_object);
550a7015092SGreg Clayton     status = LoadSubCommand ("command",    command_command_object);
551a7015092SGreg Clayton     status = LoadSubCommand ("modify",     modify_command_object);
55230fdc8d8SChris Lattner }
55330fdc8d8SChris Lattner 
55430fdc8d8SChris Lattner CommandObjectMultiwordBreakpoint::~CommandObjectMultiwordBreakpoint ()
55530fdc8d8SChris Lattner {
55630fdc8d8SChris Lattner }
55730fdc8d8SChris Lattner 
55830fdc8d8SChris Lattner void
55930fdc8d8SChris Lattner CommandObjectMultiwordBreakpoint::VerifyBreakpointIDs (Args &args, Target *target, CommandReturnObject &result,
56030fdc8d8SChris Lattner                                                          BreakpointIDList *valid_ids)
56130fdc8d8SChris Lattner {
56230fdc8d8SChris Lattner     // args can be strings representing 1). integers (for breakpoint ids)
56330fdc8d8SChris Lattner     //                                  2). the full breakpoint & location canonical representation
56430fdc8d8SChris Lattner     //                                  3). the word "to" or a hyphen, representing a range (in which case there
56530fdc8d8SChris Lattner     //                                      had *better* be an entry both before & after of one of the first two types.
56636f3b369SJim Ingham     // If args is empty, we will use the last created breakpoint (if there is one.)
56730fdc8d8SChris Lattner 
56830fdc8d8SChris Lattner     Args temp_args;
56930fdc8d8SChris Lattner 
57036f3b369SJim Ingham     if (args.GetArgumentCount() == 0)
57136f3b369SJim Ingham     {
57236f3b369SJim Ingham         if (target->GetLastCreatedBreakpoint() != NULL)
57336f3b369SJim Ingham         {
57436f3b369SJim Ingham             valid_ids->AddBreakpointID (BreakpointID(target->GetLastCreatedBreakpoint()->GetID(), LLDB_INVALID_BREAK_ID));
57536f3b369SJim Ingham             result.SetStatus (eReturnStatusSuccessFinishNoResult);
57636f3b369SJim Ingham         }
57736f3b369SJim Ingham         else
57836f3b369SJim Ingham         {
57936f3b369SJim Ingham             result.AppendError("No breakpoint specified and no last created breakpoint.");
58036f3b369SJim Ingham             result.SetStatus (eReturnStatusFailed);
58136f3b369SJim Ingham         }
58236f3b369SJim Ingham         return;
58336f3b369SJim Ingham     }
58436f3b369SJim Ingham 
58530fdc8d8SChris Lattner     // Create a new Args variable to use; copy any non-breakpoint-id-ranges stuff directly from the old ARGS to
58630fdc8d8SChris Lattner     // the new TEMP_ARGS.  Do not copy breakpoint id range strings over; instead generate a list of strings for
58730fdc8d8SChris Lattner     // all the breakpoint ids in the range, and shove all of those breakpoint id strings into TEMP_ARGS.
58830fdc8d8SChris Lattner 
58930fdc8d8SChris Lattner     BreakpointIDList::FindAndReplaceIDRanges (args, target, result, temp_args);
59030fdc8d8SChris Lattner 
59130fdc8d8SChris Lattner     // NOW, convert the list of breakpoint id strings in TEMP_ARGS into an actual BreakpointIDList:
59230fdc8d8SChris Lattner 
593c982c768SGreg Clayton     valid_ids->InsertStringArray (temp_args.GetConstArgumentVector(), temp_args.GetArgumentCount(), result);
59430fdc8d8SChris Lattner 
59530fdc8d8SChris Lattner     // At this point,  all of the breakpoint ids that the user passed in have been converted to breakpoint IDs
59630fdc8d8SChris Lattner     // and put into valid_ids.
59730fdc8d8SChris Lattner 
59830fdc8d8SChris Lattner     if (result.Succeeded())
59930fdc8d8SChris Lattner     {
60030fdc8d8SChris Lattner         // Now that we've converted everything from args into a list of breakpoint ids, go through our tentative list
60130fdc8d8SChris Lattner         // of breakpoint id's and verify that they correspond to valid/currently set breakpoints.
60230fdc8d8SChris Lattner 
603c982c768SGreg Clayton         const size_t count = valid_ids->GetSize();
604c982c768SGreg Clayton         for (size_t i = 0; i < count; ++i)
60530fdc8d8SChris Lattner         {
60630fdc8d8SChris Lattner             BreakpointID cur_bp_id = valid_ids->GetBreakpointIDAtIndex (i);
60730fdc8d8SChris Lattner             Breakpoint *breakpoint = target->GetBreakpointByID (cur_bp_id.GetBreakpointID()).get();
60830fdc8d8SChris Lattner             if (breakpoint != NULL)
60930fdc8d8SChris Lattner             {
61030fdc8d8SChris Lattner                 int num_locations = breakpoint->GetNumLocations();
61130fdc8d8SChris Lattner                 if (cur_bp_id.GetLocationID() > num_locations)
61230fdc8d8SChris Lattner                 {
61330fdc8d8SChris Lattner                     StreamString id_str;
614c982c768SGreg Clayton                     BreakpointID::GetCanonicalReference (&id_str,
615c982c768SGreg Clayton                                                          cur_bp_id.GetBreakpointID(),
61630fdc8d8SChris Lattner                                                          cur_bp_id.GetLocationID());
617c982c768SGreg Clayton                     i = valid_ids->GetSize() + 1;
61830fdc8d8SChris Lattner                     result.AppendErrorWithFormat ("'%s' is not a currently valid breakpoint/location id.\n",
61930fdc8d8SChris Lattner                                                  id_str.GetData());
62030fdc8d8SChris Lattner                     result.SetStatus (eReturnStatusFailed);
62130fdc8d8SChris Lattner                 }
62230fdc8d8SChris Lattner             }
62330fdc8d8SChris Lattner             else
62430fdc8d8SChris Lattner             {
625c982c768SGreg Clayton                 i = valid_ids->GetSize() + 1;
62630fdc8d8SChris Lattner                 result.AppendErrorWithFormat ("'%d' is not a currently valid breakpoint id.\n", cur_bp_id.GetBreakpointID());
62730fdc8d8SChris Lattner                 result.SetStatus (eReturnStatusFailed);
62830fdc8d8SChris Lattner             }
62930fdc8d8SChris Lattner         }
63030fdc8d8SChris Lattner     }
63130fdc8d8SChris Lattner }
63230fdc8d8SChris Lattner 
63330fdc8d8SChris Lattner //-------------------------------------------------------------------------
63430fdc8d8SChris Lattner // CommandObjectBreakpointList::Options
63530fdc8d8SChris Lattner //-------------------------------------------------------------------------
636ae1c4cf5SJim Ingham #pragma mark List::CommandOptions
63730fdc8d8SChris Lattner 
638eb0103f2SGreg Clayton CommandObjectBreakpointList::CommandOptions::CommandOptions(CommandInterpreter &interpreter) :
639eb0103f2SGreg Clayton     Options (interpreter),
64079042b3eSCaroline Tice     m_level (lldb::eDescriptionLevelBrief)  // Breakpoint List defaults to brief descriptions
64130fdc8d8SChris Lattner {
64230fdc8d8SChris Lattner }
64330fdc8d8SChris Lattner 
64430fdc8d8SChris Lattner CommandObjectBreakpointList::CommandOptions::~CommandOptions ()
64530fdc8d8SChris Lattner {
64630fdc8d8SChris Lattner }
64730fdc8d8SChris Lattner 
648e0d378b3SGreg Clayton OptionDefinition
64930fdc8d8SChris Lattner CommandObjectBreakpointList::CommandOptions::g_option_table[] =
65030fdc8d8SChris Lattner {
651deaab222SCaroline Tice     { LLDB_OPT_SET_ALL, false, "internal", 'i', no_argument, NULL, 0, eArgTypeNone,
6528651121cSJim Ingham         "Show debugger internal breakpoints" },
6538651121cSJim Ingham 
654deaab222SCaroline Tice     { LLDB_OPT_SET_1, false, "brief",    'b', no_argument, NULL, 0, eArgTypeNone,
65530fdc8d8SChris Lattner         "Give a brief description of the breakpoint (no location info)."},
65630fdc8d8SChris Lattner 
65730fdc8d8SChris Lattner     // FIXME: We need to add an "internal" command, and then add this sort of thing to it.
65830fdc8d8SChris Lattner     // But I need to see it for now, and don't want to wait.
659deaab222SCaroline Tice     { LLDB_OPT_SET_2, false, "full",    'f', no_argument, NULL, 0, eArgTypeNone,
66030fdc8d8SChris Lattner         "Give a full description of the breakpoint and its locations."},
66130fdc8d8SChris Lattner 
662deaab222SCaroline Tice     { LLDB_OPT_SET_3, false, "verbose", 'v', no_argument, NULL, 0, eArgTypeNone,
66330fdc8d8SChris Lattner         "Explain everything we know about the breakpoint (for debugging debugger bugs)." },
66430fdc8d8SChris Lattner 
665deaab222SCaroline Tice     { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL }
66630fdc8d8SChris Lattner };
66730fdc8d8SChris Lattner 
668e0d378b3SGreg Clayton const OptionDefinition*
66930fdc8d8SChris Lattner CommandObjectBreakpointList::CommandOptions::GetDefinitions ()
67030fdc8d8SChris Lattner {
67130fdc8d8SChris Lattner     return g_option_table;
67230fdc8d8SChris Lattner }
67330fdc8d8SChris Lattner 
67430fdc8d8SChris Lattner Error
675f6b8b581SGreg Clayton CommandObjectBreakpointList::CommandOptions::SetOptionValue (uint32_t option_idx, const char *option_arg)
67630fdc8d8SChris Lattner {
67730fdc8d8SChris Lattner     Error error;
67830fdc8d8SChris Lattner     char short_option = (char) m_getopt_table[option_idx].val;
67930fdc8d8SChris Lattner 
68030fdc8d8SChris Lattner     switch (short_option)
68130fdc8d8SChris Lattner     {
68230fdc8d8SChris Lattner         case 'b':
68330fdc8d8SChris Lattner             m_level = lldb::eDescriptionLevelBrief;
68430fdc8d8SChris Lattner             break;
68530fdc8d8SChris Lattner         case 'f':
68630fdc8d8SChris Lattner             m_level = lldb::eDescriptionLevelFull;
68730fdc8d8SChris Lattner             break;
68830fdc8d8SChris Lattner         case 'v':
68930fdc8d8SChris Lattner             m_level = lldb::eDescriptionLevelVerbose;
69030fdc8d8SChris Lattner             break;
69130fdc8d8SChris Lattner         case 'i':
69230fdc8d8SChris Lattner             m_internal = true;
69330fdc8d8SChris Lattner             break;
69430fdc8d8SChris Lattner         default:
69530fdc8d8SChris Lattner             error.SetErrorStringWithFormat ("Unrecognized option '%c'.\n", short_option);
69630fdc8d8SChris Lattner             break;
69730fdc8d8SChris Lattner     }
69830fdc8d8SChris Lattner 
69930fdc8d8SChris Lattner     return error;
70030fdc8d8SChris Lattner }
70130fdc8d8SChris Lattner 
70230fdc8d8SChris Lattner void
703f6b8b581SGreg Clayton CommandObjectBreakpointList::CommandOptions::OptionParsingStarting ()
70430fdc8d8SChris Lattner {
705d915e16fSJim Ingham     m_level = lldb::eDescriptionLevelFull;
70630fdc8d8SChris Lattner     m_internal = false;
70730fdc8d8SChris Lattner }
70830fdc8d8SChris Lattner 
70930fdc8d8SChris Lattner //-------------------------------------------------------------------------
71030fdc8d8SChris Lattner // CommandObjectBreakpointList
71130fdc8d8SChris Lattner //-------------------------------------------------------------------------
712ae1c4cf5SJim Ingham #pragma mark List
71330fdc8d8SChris Lattner 
714a7015092SGreg Clayton CommandObjectBreakpointList::CommandObjectBreakpointList (CommandInterpreter &interpreter) :
715a7015092SGreg Clayton     CommandObject (interpreter,
716a7015092SGreg Clayton                    "breakpoint list",
71730fdc8d8SChris Lattner                    "List some or all breakpoints at configurable levels of detail.",
718eb0103f2SGreg Clayton                    NULL),
719eb0103f2SGreg Clayton     m_options (interpreter)
72030fdc8d8SChris Lattner {
721e139cf23SCaroline Tice     CommandArgumentEntry arg;
722e139cf23SCaroline Tice     CommandArgumentData bp_id_arg;
723e139cf23SCaroline Tice 
724e139cf23SCaroline Tice     // Define the first (and only) variant of this arg.
725e139cf23SCaroline Tice     bp_id_arg.arg_type = eArgTypeBreakpointID;
726405fe67fSCaroline Tice     bp_id_arg.arg_repetition = eArgRepeatOptional;
727e139cf23SCaroline Tice 
728e139cf23SCaroline Tice     // There is only one variant this argument could be; put it into the argument entry.
729e139cf23SCaroline Tice     arg.push_back (bp_id_arg);
730e139cf23SCaroline Tice 
731e139cf23SCaroline Tice     // Push the data for the first argument into the m_arguments vector.
732e139cf23SCaroline Tice     m_arguments.push_back (arg);
73330fdc8d8SChris Lattner }
73430fdc8d8SChris Lattner 
73530fdc8d8SChris Lattner CommandObjectBreakpointList::~CommandObjectBreakpointList ()
73630fdc8d8SChris Lattner {
73730fdc8d8SChris Lattner }
73830fdc8d8SChris Lattner 
73930fdc8d8SChris Lattner Options *
74030fdc8d8SChris Lattner CommandObjectBreakpointList::GetOptions ()
74130fdc8d8SChris Lattner {
74230fdc8d8SChris Lattner     return &m_options;
74330fdc8d8SChris Lattner }
74430fdc8d8SChris Lattner 
74530fdc8d8SChris Lattner bool
74630fdc8d8SChris Lattner CommandObjectBreakpointList::Execute
74730fdc8d8SChris Lattner (
74830fdc8d8SChris Lattner     Args& args,
74930fdc8d8SChris Lattner     CommandReturnObject &result
75030fdc8d8SChris Lattner )
75130fdc8d8SChris Lattner {
752a7015092SGreg Clayton     Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get();
75330fdc8d8SChris Lattner     if (target == NULL)
75430fdc8d8SChris Lattner     {
7559068d794SCaroline Tice         result.AppendError ("Invalid target. No current target or breakpoints.");
75630fdc8d8SChris Lattner         result.SetStatus (eReturnStatusSuccessFinishNoResult);
75730fdc8d8SChris Lattner         return true;
75830fdc8d8SChris Lattner     }
75930fdc8d8SChris Lattner 
76030fdc8d8SChris Lattner     const BreakpointList &breakpoints = target->GetBreakpointList(m_options.m_internal);
7611b54c88cSJim Ingham     Mutex::Locker locker;
7621b54c88cSJim Ingham     target->GetBreakpointList(m_options.m_internal).GetListMutex(locker);
7631b54c88cSJim Ingham 
76430fdc8d8SChris Lattner     size_t num_breakpoints = breakpoints.GetSize();
76530fdc8d8SChris Lattner 
76630fdc8d8SChris Lattner     if (num_breakpoints == 0)
76730fdc8d8SChris Lattner     {
76830fdc8d8SChris Lattner         result.AppendMessage ("No breakpoints currently set.");
76930fdc8d8SChris Lattner         result.SetStatus (eReturnStatusSuccessFinishNoResult);
77030fdc8d8SChris Lattner         return true;
77130fdc8d8SChris Lattner     }
77230fdc8d8SChris Lattner 
77385e8b814SJim Ingham     Stream &output_stream = result.GetOutputStream();
77430fdc8d8SChris Lattner 
77530fdc8d8SChris Lattner     if (args.GetArgumentCount() == 0)
77630fdc8d8SChris Lattner     {
77730fdc8d8SChris Lattner         // No breakpoint selected; show info about all currently set breakpoints.
77830fdc8d8SChris Lattner         result.AppendMessage ("Current breakpoints:");
779c982c768SGreg Clayton         for (size_t i = 0; i < num_breakpoints; ++i)
78030fdc8d8SChris Lattner         {
7819fed0d85SGreg Clayton             Breakpoint *breakpoint = breakpoints.GetBreakpointAtIndex (i).get();
7826611103cSGreg Clayton             AddBreakpointDescription (&output_stream, breakpoint, m_options.m_level);
78330fdc8d8SChris Lattner         }
78430fdc8d8SChris Lattner         result.SetStatus (eReturnStatusSuccessFinishNoResult);
78530fdc8d8SChris Lattner     }
78630fdc8d8SChris Lattner     else
78730fdc8d8SChris Lattner     {
78830fdc8d8SChris Lattner         // Particular breakpoints selected; show info about that breakpoint.
78930fdc8d8SChris Lattner         BreakpointIDList valid_bp_ids;
79030fdc8d8SChris Lattner         CommandObjectMultiwordBreakpoint::VerifyBreakpointIDs (args, target, result, &valid_bp_ids);
79130fdc8d8SChris Lattner 
79230fdc8d8SChris Lattner         if (result.Succeeded())
79330fdc8d8SChris Lattner         {
794c982c768SGreg Clayton             for (size_t i = 0; i < valid_bp_ids.GetSize(); ++i)
79530fdc8d8SChris Lattner             {
79630fdc8d8SChris Lattner                 BreakpointID cur_bp_id = valid_bp_ids.GetBreakpointIDAtIndex (i);
79730fdc8d8SChris Lattner                 Breakpoint *breakpoint = target->GetBreakpointByID (cur_bp_id.GetBreakpointID()).get();
7986611103cSGreg Clayton                 AddBreakpointDescription (&output_stream, breakpoint, m_options.m_level);
79930fdc8d8SChris Lattner             }
80030fdc8d8SChris Lattner             result.SetStatus (eReturnStatusSuccessFinishNoResult);
80130fdc8d8SChris Lattner         }
80230fdc8d8SChris Lattner         else
80330fdc8d8SChris Lattner         {
80430fdc8d8SChris Lattner             result.AppendError ("Invalid breakpoint id.");
80530fdc8d8SChris Lattner             result.SetStatus (eReturnStatusFailed);
80630fdc8d8SChris Lattner         }
80730fdc8d8SChris Lattner     }
80830fdc8d8SChris Lattner 
80930fdc8d8SChris Lattner     return result.Succeeded();
81030fdc8d8SChris Lattner }
81130fdc8d8SChris Lattner 
81230fdc8d8SChris Lattner //-------------------------------------------------------------------------
81330fdc8d8SChris Lattner // CommandObjectBreakpointEnable
81430fdc8d8SChris Lattner //-------------------------------------------------------------------------
815ae1c4cf5SJim Ingham #pragma mark Enable
81630fdc8d8SChris Lattner 
817a7015092SGreg Clayton CommandObjectBreakpointEnable::CommandObjectBreakpointEnable (CommandInterpreter &interpreter) :
818a7015092SGreg Clayton     CommandObject (interpreter,
819a7015092SGreg Clayton                    "enable",
820e3d26315SCaroline Tice                    "Enable the specified disabled breakpoint(s). If no breakpoints are specified, enable all of them.",
821e139cf23SCaroline Tice                    NULL)
82230fdc8d8SChris Lattner {
823e139cf23SCaroline Tice     CommandArgumentEntry arg;
824e139cf23SCaroline Tice     CommandArgumentData bp_id_arg;
825e139cf23SCaroline Tice     CommandArgumentData bp_id_range_arg;
826e139cf23SCaroline Tice 
827e139cf23SCaroline Tice     // Create the first variant for the first (and only) argument for this command.
828e139cf23SCaroline Tice     bp_id_arg.arg_type = eArgTypeBreakpointID;
829405fe67fSCaroline Tice     bp_id_arg.arg_repetition = eArgRepeatOptional;
830e139cf23SCaroline Tice 
831e139cf23SCaroline Tice     // Create the second variant for the first (and only) argument for this command.
832e139cf23SCaroline Tice     bp_id_range_arg.arg_type = eArgTypeBreakpointIDRange;
833405fe67fSCaroline Tice     bp_id_range_arg.arg_repetition = eArgRepeatOptional;
834e139cf23SCaroline Tice 
835e139cf23SCaroline Tice     // The first (and only) argument for this command could be either a bp_id or a bp_id_range.
836e139cf23SCaroline Tice     // Push both variants into the entry for the first argument for this command.
837e139cf23SCaroline Tice     arg.push_back (bp_id_arg);
838e139cf23SCaroline Tice     arg.push_back (bp_id_range_arg);
839e139cf23SCaroline Tice 
840e139cf23SCaroline Tice     // Add the entry for the first argument for this command to the object's arguments vector.
841e139cf23SCaroline Tice     m_arguments.push_back (arg);
84230fdc8d8SChris Lattner }
84330fdc8d8SChris Lattner 
84430fdc8d8SChris Lattner 
84530fdc8d8SChris Lattner CommandObjectBreakpointEnable::~CommandObjectBreakpointEnable ()
84630fdc8d8SChris Lattner {
84730fdc8d8SChris Lattner }
84830fdc8d8SChris Lattner 
84930fdc8d8SChris Lattner 
85030fdc8d8SChris Lattner bool
8516611103cSGreg Clayton CommandObjectBreakpointEnable::Execute
8526611103cSGreg Clayton (
8536611103cSGreg Clayton     Args& args,
8546611103cSGreg Clayton     CommandReturnObject &result
8556611103cSGreg Clayton )
85630fdc8d8SChris Lattner {
857a7015092SGreg Clayton     Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get();
85830fdc8d8SChris Lattner     if (target == NULL)
85930fdc8d8SChris Lattner     {
8609068d794SCaroline Tice         result.AppendError ("Invalid target.  No existing target or breakpoints.");
86130fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
86230fdc8d8SChris Lattner         return false;
86330fdc8d8SChris Lattner     }
86430fdc8d8SChris Lattner 
8651b54c88cSJim Ingham     Mutex::Locker locker;
8661b54c88cSJim Ingham     target->GetBreakpointList().GetListMutex(locker);
8671b54c88cSJim Ingham 
86830fdc8d8SChris Lattner     const BreakpointList &breakpoints = target->GetBreakpointList();
8691b54c88cSJim Ingham 
87030fdc8d8SChris Lattner     size_t num_breakpoints = breakpoints.GetSize();
87130fdc8d8SChris Lattner 
87230fdc8d8SChris Lattner     if (num_breakpoints == 0)
87330fdc8d8SChris Lattner     {
87430fdc8d8SChris Lattner         result.AppendError ("No breakpoints exist to be enabled.");
87530fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
87630fdc8d8SChris Lattner         return false;
87730fdc8d8SChris Lattner     }
87830fdc8d8SChris Lattner 
87930fdc8d8SChris Lattner     if (args.GetArgumentCount() == 0)
88030fdc8d8SChris Lattner     {
88130fdc8d8SChris Lattner         // No breakpoint selected; enable all currently set breakpoints.
88230fdc8d8SChris Lattner         target->EnableAllBreakpoints ();
88330fdc8d8SChris Lattner         result.AppendMessageWithFormat ("All breakpoints enabled. (%d breakpoints)\n", num_breakpoints);
88430fdc8d8SChris Lattner         result.SetStatus (eReturnStatusSuccessFinishNoResult);
88530fdc8d8SChris Lattner     }
88630fdc8d8SChris Lattner     else
88730fdc8d8SChris Lattner     {
88830fdc8d8SChris Lattner         // Particular breakpoint selected; enable that breakpoint.
88930fdc8d8SChris Lattner         BreakpointIDList valid_bp_ids;
89030fdc8d8SChris Lattner         CommandObjectMultiwordBreakpoint::VerifyBreakpointIDs (args, target, result, &valid_bp_ids);
89130fdc8d8SChris Lattner 
89230fdc8d8SChris Lattner         if (result.Succeeded())
89330fdc8d8SChris Lattner         {
89430fdc8d8SChris Lattner             int enable_count = 0;
89530fdc8d8SChris Lattner             int loc_count = 0;
896c982c768SGreg Clayton             const size_t count = valid_bp_ids.GetSize();
897c982c768SGreg Clayton             for (size_t i = 0; i < count; ++i)
89830fdc8d8SChris Lattner             {
89930fdc8d8SChris Lattner                 BreakpointID cur_bp_id = valid_bp_ids.GetBreakpointIDAtIndex (i);
90030fdc8d8SChris Lattner 
90130fdc8d8SChris Lattner                 if (cur_bp_id.GetBreakpointID() != LLDB_INVALID_BREAK_ID)
90230fdc8d8SChris Lattner                 {
90330fdc8d8SChris Lattner                     Breakpoint *breakpoint = target->GetBreakpointByID (cur_bp_id.GetBreakpointID()).get();
90430fdc8d8SChris Lattner                     if (cur_bp_id.GetLocationID() != LLDB_INVALID_BREAK_ID)
90530fdc8d8SChris Lattner                     {
90630fdc8d8SChris Lattner                         BreakpointLocation *location = breakpoint->FindLocationByID (cur_bp_id.GetLocationID()).get();
90730fdc8d8SChris Lattner                         if (location)
90830fdc8d8SChris Lattner                         {
90930fdc8d8SChris Lattner                             location->SetEnabled (true);
91030fdc8d8SChris Lattner                             ++loc_count;
91130fdc8d8SChris Lattner                         }
91230fdc8d8SChris Lattner                     }
91330fdc8d8SChris Lattner                     else
91430fdc8d8SChris Lattner                     {
915ae1c4cf5SJim Ingham                         breakpoint->SetEnabled (true);
91630fdc8d8SChris Lattner                         ++enable_count;
91730fdc8d8SChris Lattner                     }
91830fdc8d8SChris Lattner                 }
91930fdc8d8SChris Lattner             }
92030fdc8d8SChris Lattner             result.AppendMessageWithFormat ("%d breakpoints enabled.\n", enable_count + loc_count);
92130fdc8d8SChris Lattner             result.SetStatus (eReturnStatusSuccessFinishNoResult);
92230fdc8d8SChris Lattner         }
92330fdc8d8SChris Lattner     }
92430fdc8d8SChris Lattner 
92530fdc8d8SChris Lattner     return result.Succeeded();
92630fdc8d8SChris Lattner }
92730fdc8d8SChris Lattner 
92830fdc8d8SChris Lattner //-------------------------------------------------------------------------
92930fdc8d8SChris Lattner // CommandObjectBreakpointDisable
93030fdc8d8SChris Lattner //-------------------------------------------------------------------------
931ae1c4cf5SJim Ingham #pragma mark Disable
93230fdc8d8SChris Lattner 
933a7015092SGreg Clayton CommandObjectBreakpointDisable::CommandObjectBreakpointDisable (CommandInterpreter &interpreter) :
934a7015092SGreg Clayton     CommandObject (interpreter,
935e139cf23SCaroline Tice                    "breakpoint disable",
936e3d26315SCaroline Tice                    "Disable the specified breakpoint(s) without removing it/them.  If no breakpoints are specified, disable them all.",
937e139cf23SCaroline Tice                    NULL)
93830fdc8d8SChris Lattner {
939e139cf23SCaroline Tice     CommandArgumentEntry arg;
940e139cf23SCaroline Tice     CommandArgumentData bp_id_arg;
941e139cf23SCaroline Tice     CommandArgumentData bp_id_range_arg;
942e139cf23SCaroline Tice 
943e139cf23SCaroline Tice     // Create the first variant for the first (and only) argument for this command.
944e139cf23SCaroline Tice     bp_id_arg.arg_type = eArgTypeBreakpointID;
945405fe67fSCaroline Tice     bp_id_arg.arg_repetition = eArgRepeatOptional;
946e139cf23SCaroline Tice 
947e139cf23SCaroline Tice     // Create the second variant for the first (and only) argument for this command.
948e139cf23SCaroline Tice     bp_id_range_arg.arg_type = eArgTypeBreakpointIDRange;
949405fe67fSCaroline Tice     bp_id_range_arg.arg_repetition = eArgRepeatOptional;
950e139cf23SCaroline Tice 
951e139cf23SCaroline Tice     // The first (and only) argument for this command could be either a bp_id or a bp_id_range.
952e139cf23SCaroline Tice     // Push both variants into the entry for the first argument for this command.
953e139cf23SCaroline Tice     arg.push_back (bp_id_arg);
954e139cf23SCaroline Tice     arg.push_back (bp_id_range_arg);
955e139cf23SCaroline Tice 
956e139cf23SCaroline Tice     // Add the entry for the first argument for this command to the object's arguments vector.
957e139cf23SCaroline Tice     m_arguments.push_back (arg);
95830fdc8d8SChris Lattner }
95930fdc8d8SChris Lattner 
96030fdc8d8SChris Lattner CommandObjectBreakpointDisable::~CommandObjectBreakpointDisable ()
96130fdc8d8SChris Lattner {
96230fdc8d8SChris Lattner }
96330fdc8d8SChris Lattner 
96430fdc8d8SChris Lattner bool
9656611103cSGreg Clayton CommandObjectBreakpointDisable::Execute
9666611103cSGreg Clayton (
9676611103cSGreg Clayton     Args& args,
9686611103cSGreg Clayton     CommandReturnObject &result
9696611103cSGreg Clayton )
97030fdc8d8SChris Lattner {
971a7015092SGreg Clayton     Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get();
97230fdc8d8SChris Lattner     if (target == NULL)
97330fdc8d8SChris Lattner     {
9749068d794SCaroline Tice         result.AppendError ("Invalid target.  No existing target or breakpoints.");
97530fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
97630fdc8d8SChris Lattner         return false;
97730fdc8d8SChris Lattner     }
97830fdc8d8SChris Lattner 
9791b54c88cSJim Ingham     Mutex::Locker locker;
9801b54c88cSJim Ingham     target->GetBreakpointList().GetListMutex(locker);
9811b54c88cSJim Ingham 
98230fdc8d8SChris Lattner     const BreakpointList &breakpoints = target->GetBreakpointList();
98330fdc8d8SChris Lattner     size_t num_breakpoints = breakpoints.GetSize();
98430fdc8d8SChris Lattner 
98530fdc8d8SChris Lattner     if (num_breakpoints == 0)
98630fdc8d8SChris Lattner     {
98730fdc8d8SChris Lattner         result.AppendError ("No breakpoints exist to be disabled.");
98830fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
98930fdc8d8SChris Lattner         return false;
99030fdc8d8SChris Lattner     }
99130fdc8d8SChris Lattner 
99230fdc8d8SChris Lattner     if (args.GetArgumentCount() == 0)
99330fdc8d8SChris Lattner     {
99430fdc8d8SChris Lattner         // No breakpoint selected; disable all currently set breakpoints.
99530fdc8d8SChris Lattner         target->DisableAllBreakpoints ();
99630fdc8d8SChris Lattner         result.AppendMessageWithFormat ("All breakpoints disabled. (%d breakpoints)\n", num_breakpoints);
99730fdc8d8SChris Lattner         result.SetStatus (eReturnStatusSuccessFinishNoResult);
99830fdc8d8SChris Lattner     }
99930fdc8d8SChris Lattner     else
100030fdc8d8SChris Lattner     {
100130fdc8d8SChris Lattner         // Particular breakpoint selected; disable that breakpoint.
100230fdc8d8SChris Lattner         BreakpointIDList valid_bp_ids;
100330fdc8d8SChris Lattner 
100430fdc8d8SChris Lattner         CommandObjectMultiwordBreakpoint::VerifyBreakpointIDs (args, target, result, &valid_bp_ids);
100530fdc8d8SChris Lattner 
100630fdc8d8SChris Lattner         if (result.Succeeded())
100730fdc8d8SChris Lattner         {
100830fdc8d8SChris Lattner             int disable_count = 0;
100930fdc8d8SChris Lattner             int loc_count = 0;
1010c982c768SGreg Clayton             const size_t count = valid_bp_ids.GetSize();
1011c982c768SGreg Clayton             for (size_t i = 0; i < count; ++i)
101230fdc8d8SChris Lattner             {
101330fdc8d8SChris Lattner                 BreakpointID cur_bp_id = valid_bp_ids.GetBreakpointIDAtIndex (i);
101430fdc8d8SChris Lattner 
101530fdc8d8SChris Lattner                 if (cur_bp_id.GetBreakpointID() != LLDB_INVALID_BREAK_ID)
101630fdc8d8SChris Lattner                 {
101730fdc8d8SChris Lattner                     Breakpoint *breakpoint = target->GetBreakpointByID (cur_bp_id.GetBreakpointID()).get();
101830fdc8d8SChris Lattner                     if (cur_bp_id.GetLocationID() != LLDB_INVALID_BREAK_ID)
101930fdc8d8SChris Lattner                     {
102030fdc8d8SChris Lattner                         BreakpointLocation *location = breakpoint->FindLocationByID (cur_bp_id.GetLocationID()).get();
102130fdc8d8SChris Lattner                         if (location)
102230fdc8d8SChris Lattner                         {
102330fdc8d8SChris Lattner                             location->SetEnabled (false);
102430fdc8d8SChris Lattner                             ++loc_count;
102530fdc8d8SChris Lattner                         }
102630fdc8d8SChris Lattner                     }
102730fdc8d8SChris Lattner                     else
102830fdc8d8SChris Lattner                     {
1029ae1c4cf5SJim Ingham                         breakpoint->SetEnabled (false);
103030fdc8d8SChris Lattner                         ++disable_count;
103130fdc8d8SChris Lattner                     }
103230fdc8d8SChris Lattner                 }
103330fdc8d8SChris Lattner             }
103430fdc8d8SChris Lattner             result.AppendMessageWithFormat ("%d breakpoints disabled.\n", disable_count + loc_count);
103530fdc8d8SChris Lattner             result.SetStatus (eReturnStatusSuccessFinishNoResult);
103630fdc8d8SChris Lattner         }
103730fdc8d8SChris Lattner     }
103830fdc8d8SChris Lattner 
103930fdc8d8SChris Lattner     return result.Succeeded();
104030fdc8d8SChris Lattner }
104130fdc8d8SChris Lattner 
104230fdc8d8SChris Lattner //-------------------------------------------------------------------------
1043b7234e40SJohnny Chen // CommandObjectBreakpointClear::CommandOptions
1044b7234e40SJohnny Chen //-------------------------------------------------------------------------
1045b7234e40SJohnny Chen #pragma mark Clear::CommandOptions
1046b7234e40SJohnny Chen 
1047eb0103f2SGreg Clayton CommandObjectBreakpointClear::CommandOptions::CommandOptions(CommandInterpreter &interpreter) :
1048eb0103f2SGreg Clayton     Options (interpreter),
1049b7234e40SJohnny Chen     m_filename (),
1050b7234e40SJohnny Chen     m_line_num (0)
1051b7234e40SJohnny Chen {
1052b7234e40SJohnny Chen }
1053b7234e40SJohnny Chen 
1054b7234e40SJohnny Chen CommandObjectBreakpointClear::CommandOptions::~CommandOptions ()
1055b7234e40SJohnny Chen {
1056b7234e40SJohnny Chen }
1057b7234e40SJohnny Chen 
1058e0d378b3SGreg Clayton OptionDefinition
1059b7234e40SJohnny Chen CommandObjectBreakpointClear::CommandOptions::g_option_table[] =
1060b7234e40SJohnny Chen {
1061b7234e40SJohnny Chen     { LLDB_OPT_SET_1, false, "file", 'f', required_argument, NULL, CommandCompletions::eSourceFileCompletion, eArgTypeFilename,
1062b7234e40SJohnny Chen         "Specify the breakpoint by source location in this particular file."},
1063b7234e40SJohnny Chen 
1064b7234e40SJohnny Chen     { LLDB_OPT_SET_1, true, "line", 'l', required_argument, NULL, 0, eArgTypeLineNum,
1065b7234e40SJohnny Chen         "Specify the breakpoint by source location at this particular line."},
1066b7234e40SJohnny Chen 
1067b7234e40SJohnny Chen     { 0, false, NULL, 0, 0, NULL, 0, eArgTypeNone, NULL }
1068b7234e40SJohnny Chen };
1069b7234e40SJohnny Chen 
1070e0d378b3SGreg Clayton const OptionDefinition*
1071b7234e40SJohnny Chen CommandObjectBreakpointClear::CommandOptions::GetDefinitions ()
1072b7234e40SJohnny Chen {
1073b7234e40SJohnny Chen     return g_option_table;
1074b7234e40SJohnny Chen }
1075b7234e40SJohnny Chen 
1076b7234e40SJohnny Chen Error
1077f6b8b581SGreg Clayton CommandObjectBreakpointClear::CommandOptions::SetOptionValue (uint32_t option_idx, const char *option_arg)
1078b7234e40SJohnny Chen {
1079b7234e40SJohnny Chen     Error error;
1080b7234e40SJohnny Chen     char short_option = (char) m_getopt_table[option_idx].val;
1081b7234e40SJohnny Chen 
1082b7234e40SJohnny Chen     switch (short_option)
1083b7234e40SJohnny Chen     {
1084b7234e40SJohnny Chen         case 'f':
1085357132ebSGreg Clayton             m_filename.assign (option_arg);
1086b7234e40SJohnny Chen             break;
1087b7234e40SJohnny Chen 
1088b7234e40SJohnny Chen         case 'l':
1089b7234e40SJohnny Chen             m_line_num = Args::StringToUInt32 (option_arg, 0);
1090b7234e40SJohnny Chen             break;
1091b7234e40SJohnny Chen 
1092b7234e40SJohnny Chen         default:
1093b7234e40SJohnny Chen             error.SetErrorStringWithFormat ("Unrecognized option '%c'.\n", short_option);
1094b7234e40SJohnny Chen             break;
1095b7234e40SJohnny Chen     }
1096b7234e40SJohnny Chen 
1097b7234e40SJohnny Chen     return error;
1098b7234e40SJohnny Chen }
1099b7234e40SJohnny Chen 
1100b7234e40SJohnny Chen void
1101f6b8b581SGreg Clayton CommandObjectBreakpointClear::CommandOptions::OptionParsingStarting ()
1102b7234e40SJohnny Chen {
1103b7234e40SJohnny Chen     m_filename.clear();
1104b7234e40SJohnny Chen     m_line_num = 0;
1105b7234e40SJohnny Chen }
1106b7234e40SJohnny Chen 
1107b7234e40SJohnny Chen //-------------------------------------------------------------------------
1108b7234e40SJohnny Chen // CommandObjectBreakpointClear
1109b7234e40SJohnny Chen //-------------------------------------------------------------------------
1110b7234e40SJohnny Chen #pragma mark Clear
1111b7234e40SJohnny Chen 
1112b7234e40SJohnny Chen CommandObjectBreakpointClear::CommandObjectBreakpointClear (CommandInterpreter &interpreter) :
1113b7234e40SJohnny Chen     CommandObject (interpreter,
1114b7234e40SJohnny Chen                    "breakpoint clear",
1115b7234e40SJohnny Chen                    "Clears a breakpoint or set of breakpoints in the executable.",
1116eb0103f2SGreg Clayton                    "breakpoint clear <cmd-options>"),
1117eb0103f2SGreg Clayton     m_options (interpreter)
1118b7234e40SJohnny Chen {
1119b7234e40SJohnny Chen }
1120b7234e40SJohnny Chen 
1121b7234e40SJohnny Chen CommandObjectBreakpointClear::~CommandObjectBreakpointClear ()
1122b7234e40SJohnny Chen {
1123b7234e40SJohnny Chen }
1124b7234e40SJohnny Chen 
1125b7234e40SJohnny Chen Options *
1126b7234e40SJohnny Chen CommandObjectBreakpointClear::GetOptions ()
1127b7234e40SJohnny Chen {
1128b7234e40SJohnny Chen     return &m_options;
1129b7234e40SJohnny Chen }
1130b7234e40SJohnny Chen 
1131b7234e40SJohnny Chen bool
1132b7234e40SJohnny Chen CommandObjectBreakpointClear::Execute
1133b7234e40SJohnny Chen (
1134b7234e40SJohnny Chen     Args& command,
1135b7234e40SJohnny Chen     CommandReturnObject &result
1136b7234e40SJohnny Chen )
1137b7234e40SJohnny Chen {
1138b7234e40SJohnny Chen     Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get();
1139b7234e40SJohnny Chen     if (target == NULL)
1140b7234e40SJohnny Chen     {
1141b7234e40SJohnny Chen         result.AppendError ("Invalid target. No existing target or breakpoints.");
1142b7234e40SJohnny Chen         result.SetStatus (eReturnStatusFailed);
1143b7234e40SJohnny Chen         return false;
1144b7234e40SJohnny Chen     }
1145b7234e40SJohnny Chen 
1146b7234e40SJohnny Chen     // The following are the various types of breakpoints that could be cleared:
1147b7234e40SJohnny Chen     //   1). -f -l (clearing breakpoint by source location)
1148b7234e40SJohnny Chen 
1149b7234e40SJohnny Chen     BreakpointClearType break_type = eClearTypeInvalid;
1150b7234e40SJohnny Chen 
1151b7234e40SJohnny Chen     if (m_options.m_line_num != 0)
1152b7234e40SJohnny Chen         break_type = eClearTypeFileAndLine;
1153b7234e40SJohnny Chen 
1154b7234e40SJohnny Chen     Mutex::Locker locker;
1155b7234e40SJohnny Chen     target->GetBreakpointList().GetListMutex(locker);
1156b7234e40SJohnny Chen 
1157b7234e40SJohnny Chen     BreakpointList &breakpoints = target->GetBreakpointList();
1158b7234e40SJohnny Chen     size_t num_breakpoints = breakpoints.GetSize();
1159b7234e40SJohnny Chen 
1160b7234e40SJohnny Chen     // Early return if there's no breakpoint at all.
1161b7234e40SJohnny Chen     if (num_breakpoints == 0)
1162b7234e40SJohnny Chen     {
1163b7234e40SJohnny Chen         result.AppendError ("Breakpoint clear: No breakpoint cleared.");
1164b7234e40SJohnny Chen         result.SetStatus (eReturnStatusFailed);
1165b7234e40SJohnny Chen         return result.Succeeded();
1166b7234e40SJohnny Chen     }
1167b7234e40SJohnny Chen 
1168b7234e40SJohnny Chen     // Find matching breakpoints and delete them.
1169b7234e40SJohnny Chen 
1170b7234e40SJohnny Chen     // First create a copy of all the IDs.
1171b7234e40SJohnny Chen     std::vector<break_id_t> BreakIDs;
1172b7234e40SJohnny Chen     for (size_t i = 0; i < num_breakpoints; ++i)
1173b7234e40SJohnny Chen         BreakIDs.push_back(breakpoints.GetBreakpointAtIndex(i).get()->GetID());
1174b7234e40SJohnny Chen 
1175b7234e40SJohnny Chen     int num_cleared = 0;
1176b7234e40SJohnny Chen     StreamString ss;
1177b7234e40SJohnny Chen     switch (break_type)
1178b7234e40SJohnny Chen     {
1179b7234e40SJohnny Chen         case eClearTypeFileAndLine: // Breakpoint by source position
1180b7234e40SJohnny Chen             {
1181b7234e40SJohnny Chen                 const ConstString filename(m_options.m_filename.c_str());
1182b7234e40SJohnny Chen                 BreakpointLocationCollection loc_coll;
1183b7234e40SJohnny Chen 
1184b7234e40SJohnny Chen                 for (size_t i = 0; i < num_breakpoints; ++i)
1185b7234e40SJohnny Chen                 {
1186b7234e40SJohnny Chen                     Breakpoint *bp = breakpoints.FindBreakpointByID(BreakIDs[i]).get();
1187b7234e40SJohnny Chen 
1188b7234e40SJohnny Chen                     if (bp->GetMatchingFileLine(filename, m_options.m_line_num, loc_coll))
1189b7234e40SJohnny Chen                     {
1190b7234e40SJohnny Chen                         // If the collection size is 0, it's a full match and we can just remove the breakpoint.
1191b7234e40SJohnny Chen                         if (loc_coll.GetSize() == 0)
1192b7234e40SJohnny Chen                         {
1193b7234e40SJohnny Chen                             bp->GetDescription(&ss, lldb::eDescriptionLevelBrief);
1194b7234e40SJohnny Chen                             ss.EOL();
1195b7234e40SJohnny Chen                             target->RemoveBreakpointByID (bp->GetID());
1196b7234e40SJohnny Chen                             ++num_cleared;
1197b7234e40SJohnny Chen                         }
1198b7234e40SJohnny Chen                     }
1199b7234e40SJohnny Chen                 }
1200b7234e40SJohnny Chen             }
1201b7234e40SJohnny Chen             break;
1202b7234e40SJohnny Chen 
1203b7234e40SJohnny Chen         default:
1204b7234e40SJohnny Chen             break;
1205b7234e40SJohnny Chen     }
1206b7234e40SJohnny Chen 
1207b7234e40SJohnny Chen     if (num_cleared > 0)
1208b7234e40SJohnny Chen     {
120985e8b814SJim Ingham         Stream &output_stream = result.GetOutputStream();
1210b7234e40SJohnny Chen         output_stream.Printf ("%d breakpoints cleared:\n", num_cleared);
1211b7234e40SJohnny Chen         output_stream << ss.GetData();
1212b7234e40SJohnny Chen         output_stream.EOL();
1213b7234e40SJohnny Chen         result.SetStatus (eReturnStatusSuccessFinishNoResult);
1214b7234e40SJohnny Chen     }
1215b7234e40SJohnny Chen     else
1216b7234e40SJohnny Chen     {
1217b7234e40SJohnny Chen         result.AppendError ("Breakpoint clear: No breakpoint cleared.");
1218b7234e40SJohnny Chen         result.SetStatus (eReturnStatusFailed);
1219b7234e40SJohnny Chen     }
1220b7234e40SJohnny Chen 
1221b7234e40SJohnny Chen     return result.Succeeded();
1222b7234e40SJohnny Chen }
1223b7234e40SJohnny Chen 
1224b7234e40SJohnny Chen //-------------------------------------------------------------------------
122530fdc8d8SChris Lattner // CommandObjectBreakpointDelete
122630fdc8d8SChris Lattner //-------------------------------------------------------------------------
1227ae1c4cf5SJim Ingham #pragma mark Delete
122830fdc8d8SChris Lattner 
1229a7015092SGreg Clayton CommandObjectBreakpointDelete::CommandObjectBreakpointDelete(CommandInterpreter &interpreter) :
1230a7015092SGreg Clayton     CommandObject (interpreter,
1231a7015092SGreg Clayton                    "breakpoint delete",
1232e3d26315SCaroline Tice                    "Delete the specified breakpoint(s).  If no breakpoints are specified, delete them all.",
1233e139cf23SCaroline Tice                    NULL)
123430fdc8d8SChris Lattner {
1235e139cf23SCaroline Tice     CommandArgumentEntry arg;
1236e139cf23SCaroline Tice     CommandArgumentData bp_id_arg;
1237e139cf23SCaroline Tice     CommandArgumentData bp_id_range_arg;
1238e139cf23SCaroline Tice 
1239e139cf23SCaroline Tice     // Create the first variant for the first (and only) argument for this command.
1240e139cf23SCaroline Tice     bp_id_arg.arg_type = eArgTypeBreakpointID;
1241405fe67fSCaroline Tice     bp_id_arg.arg_repetition = eArgRepeatOptional;
1242e139cf23SCaroline Tice 
1243e139cf23SCaroline Tice     // Create the second variant for the first (and only) argument for this command.
1244e139cf23SCaroline Tice     bp_id_range_arg.arg_type = eArgTypeBreakpointIDRange;
1245405fe67fSCaroline Tice     bp_id_range_arg.arg_repetition = eArgRepeatOptional;
1246e139cf23SCaroline Tice 
1247e139cf23SCaroline Tice     // The first (and only) argument for this command could be either a bp_id or a bp_id_range.
1248e139cf23SCaroline Tice     // Push both variants into the entry for the first argument for this command.
1249e139cf23SCaroline Tice     arg.push_back (bp_id_arg);
1250e139cf23SCaroline Tice     arg.push_back (bp_id_range_arg);
1251e139cf23SCaroline Tice 
1252e139cf23SCaroline Tice     // Add the entry for the first argument for this command to the object's arguments vector.
1253e139cf23SCaroline Tice     m_arguments.push_back (arg);
125430fdc8d8SChris Lattner }
125530fdc8d8SChris Lattner 
125630fdc8d8SChris Lattner 
125730fdc8d8SChris Lattner CommandObjectBreakpointDelete::~CommandObjectBreakpointDelete ()
125830fdc8d8SChris Lattner {
125930fdc8d8SChris Lattner }
126030fdc8d8SChris Lattner 
126130fdc8d8SChris Lattner bool
12626611103cSGreg Clayton CommandObjectBreakpointDelete::Execute
12636611103cSGreg Clayton (
12646611103cSGreg Clayton     Args& args,
12656611103cSGreg Clayton     CommandReturnObject &result
12666611103cSGreg Clayton )
126730fdc8d8SChris Lattner {
1268a7015092SGreg Clayton     Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get();
126930fdc8d8SChris Lattner     if (target == NULL)
127030fdc8d8SChris Lattner     {
12719068d794SCaroline Tice         result.AppendError ("Invalid target. No existing target or breakpoints.");
127230fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
127330fdc8d8SChris Lattner         return false;
127430fdc8d8SChris Lattner     }
127530fdc8d8SChris Lattner 
12761b54c88cSJim Ingham     Mutex::Locker locker;
12771b54c88cSJim Ingham     target->GetBreakpointList().GetListMutex(locker);
12781b54c88cSJim Ingham 
127930fdc8d8SChris Lattner     const BreakpointList &breakpoints = target->GetBreakpointList();
12801b54c88cSJim Ingham 
128130fdc8d8SChris Lattner     size_t num_breakpoints = breakpoints.GetSize();
128230fdc8d8SChris Lattner 
128330fdc8d8SChris Lattner     if (num_breakpoints == 0)
128430fdc8d8SChris Lattner     {
128530fdc8d8SChris Lattner         result.AppendError ("No breakpoints exist to be deleted.");
128630fdc8d8SChris Lattner         result.SetStatus (eReturnStatusFailed);
128730fdc8d8SChris Lattner         return false;
128830fdc8d8SChris Lattner     }
128930fdc8d8SChris Lattner 
129030fdc8d8SChris Lattner     if (args.GetArgumentCount() == 0)
129130fdc8d8SChris Lattner     {
129236f3b369SJim Ingham         if (!m_interpreter.Confirm ("About to delete all breakpoints, do you want to do that?", true))
129330fdc8d8SChris Lattner         {
129436f3b369SJim Ingham             result.AppendMessage("Operation cancelled...");
129530fdc8d8SChris Lattner         }
129636f3b369SJim Ingham         else
129736f3b369SJim Ingham         {
129830fdc8d8SChris Lattner             target->RemoveAllBreakpoints ();
129930fdc8d8SChris Lattner             result.AppendMessageWithFormat ("All breakpoints removed. (%d breakpoints)\n", num_breakpoints);
130036f3b369SJim Ingham         }
130130fdc8d8SChris Lattner         result.SetStatus (eReturnStatusSuccessFinishNoResult);
130230fdc8d8SChris Lattner     }
130330fdc8d8SChris Lattner     else
130430fdc8d8SChris Lattner     {
130530fdc8d8SChris Lattner         // Particular breakpoint selected; disable that breakpoint.
130630fdc8d8SChris Lattner         BreakpointIDList valid_bp_ids;
130730fdc8d8SChris Lattner         CommandObjectMultiwordBreakpoint::VerifyBreakpointIDs (args, target, result, &valid_bp_ids);
130830fdc8d8SChris Lattner 
130930fdc8d8SChris Lattner         if (result.Succeeded())
131030fdc8d8SChris Lattner         {
131130fdc8d8SChris Lattner             int delete_count = 0;
131230fdc8d8SChris Lattner             int disable_count = 0;
1313c982c768SGreg Clayton             const size_t count = valid_bp_ids.GetSize();
1314c982c768SGreg Clayton             for (size_t i = 0; i < count; ++i)
131530fdc8d8SChris Lattner             {
131630fdc8d8SChris Lattner                 BreakpointID cur_bp_id = valid_bp_ids.GetBreakpointIDAtIndex (i);
131730fdc8d8SChris Lattner 
131830fdc8d8SChris Lattner                 if (cur_bp_id.GetBreakpointID() != LLDB_INVALID_BREAK_ID)
131930fdc8d8SChris Lattner                 {
132030fdc8d8SChris Lattner                     if (cur_bp_id.GetLocationID() != LLDB_INVALID_BREAK_ID)
132130fdc8d8SChris Lattner                     {
132230fdc8d8SChris Lattner                         Breakpoint *breakpoint = target->GetBreakpointByID (cur_bp_id.GetBreakpointID()).get();
132330fdc8d8SChris Lattner                         BreakpointLocation *location = breakpoint->FindLocationByID (cur_bp_id.GetLocationID()).get();
132430fdc8d8SChris Lattner                         // It makes no sense to try to delete individual locations, so we disable them instead.
132530fdc8d8SChris Lattner                         if (location)
132630fdc8d8SChris Lattner                         {
132730fdc8d8SChris Lattner                             location->SetEnabled (false);
132830fdc8d8SChris Lattner                             ++disable_count;
132930fdc8d8SChris Lattner                         }
133030fdc8d8SChris Lattner                     }
133130fdc8d8SChris Lattner                     else
133230fdc8d8SChris Lattner                     {
133330fdc8d8SChris Lattner                         target->RemoveBreakpointByID (cur_bp_id.GetBreakpointID());
133430fdc8d8SChris Lattner                         ++delete_count;
133530fdc8d8SChris Lattner                     }
133630fdc8d8SChris Lattner                 }
133730fdc8d8SChris Lattner             }
133830fdc8d8SChris Lattner             result.AppendMessageWithFormat ("%d breakpoints deleted; %d breakpoint locations disabled.\n",
133930fdc8d8SChris Lattner                                            delete_count, disable_count);
134030fdc8d8SChris Lattner             result.SetStatus (eReturnStatusSuccessFinishNoResult);
134130fdc8d8SChris Lattner         }
134230fdc8d8SChris Lattner     }
134330fdc8d8SChris Lattner     return result.Succeeded();
134430fdc8d8SChris Lattner }
13451b54c88cSJim Ingham 
13461b54c88cSJim Ingham //-------------------------------------------------------------------------
1347ae1c4cf5SJim Ingham // CommandObjectBreakpointModify::CommandOptions
13481b54c88cSJim Ingham //-------------------------------------------------------------------------
1349ae1c4cf5SJim Ingham #pragma mark Modify::CommandOptions
13501b54c88cSJim Ingham 
1351eb0103f2SGreg Clayton CommandObjectBreakpointModify::CommandOptions::CommandOptions(CommandInterpreter &interpreter) :
1352eb0103f2SGreg Clayton     Options (interpreter),
1353c982c768SGreg Clayton     m_ignore_count (0),
13541b54c88cSJim Ingham     m_thread_id(LLDB_INVALID_THREAD_ID),
1355e0a97848SJim Ingham     m_thread_id_passed(false),
1356c982c768SGreg Clayton     m_thread_index (UINT32_MAX),
1357e0a97848SJim Ingham     m_thread_index_passed(false),
13581b54c88cSJim Ingham     m_thread_name(),
13591b54c88cSJim Ingham     m_queue_name(),
136036f3b369SJim Ingham     m_condition (),
1361c982c768SGreg Clayton     m_enable_passed (false),
1362c982c768SGreg Clayton     m_enable_value (false),
1363c982c768SGreg Clayton     m_name_passed (false),
136436f3b369SJim Ingham     m_queue_passed (false),
136536f3b369SJim Ingham     m_condition_passed (false)
13661b54c88cSJim Ingham {
13671b54c88cSJim Ingham }
13681b54c88cSJim Ingham 
1369ae1c4cf5SJim Ingham CommandObjectBreakpointModify::CommandOptions::~CommandOptions ()
13701b54c88cSJim Ingham {
13711b54c88cSJim Ingham }
13721b54c88cSJim Ingham 
1373e0d378b3SGreg Clayton OptionDefinition
1374ae1c4cf5SJim Ingham CommandObjectBreakpointModify::CommandOptions::g_option_table[] =
13751b54c88cSJim Ingham {
1376deaab222SCaroline Tice { LLDB_OPT_SET_ALL, false, "ignore-count", 'i', required_argument, NULL, NULL, eArgTypeCount, "Set the number of times this breakpoint is skipped before stopping." },
1377deaab222SCaroline Tice { LLDB_OPT_SET_ALL, false, "thread-index", 'x', required_argument, NULL, NULL, eArgTypeThreadIndex, "The breakpoint stops only for the thread whose indeX matches this argument."},
1378deaab222SCaroline Tice { LLDB_OPT_SET_ALL, false, "thread-id",    't', required_argument, NULL, NULL, eArgTypeThreadID, "The breakpoint stops only for the thread whose TID matches this argument."},
1379deaab222SCaroline Tice { LLDB_OPT_SET_ALL, false, "thread-name",  'T', required_argument, NULL, NULL, eArgTypeThreadName, "The breakpoint stops only for the thread whose thread name matches this argument."},
1380deaab222SCaroline Tice { LLDB_OPT_SET_ALL, false, "queue-name",   'q', required_argument, NULL, NULL, eArgTypeQueueName, "The breakpoint stops only for threads in the queue whose name is given by this argument."},
138136f3b369SJim Ingham { LLDB_OPT_SET_ALL, false, "condition",    'c', required_argument, NULL, NULL, eArgTypeExpression, "The breakpoint stops only if this condition expression evaluates to true."},
1382deaab222SCaroline Tice { LLDB_OPT_SET_1,   false, "enable",       'e', no_argument,       NULL, NULL, eArgTypeNone, "Enable the breakpoint."},
1383deaab222SCaroline Tice { LLDB_OPT_SET_2,   false, "disable",      'd', no_argument,       NULL, NULL, eArgTypeNone, "Disable the breakpoint."},
1384deaab222SCaroline Tice { 0,                false, NULL,            0 , 0,                 NULL, 0,    eArgTypeNone, NULL }
13851b54c88cSJim Ingham };
13861b54c88cSJim Ingham 
1387e0d378b3SGreg Clayton const OptionDefinition*
1388ae1c4cf5SJim Ingham CommandObjectBreakpointModify::CommandOptions::GetDefinitions ()
13891b54c88cSJim Ingham {
13901b54c88cSJim Ingham     return g_option_table;
13911b54c88cSJim Ingham }
13921b54c88cSJim Ingham 
13931b54c88cSJim Ingham Error
1394f6b8b581SGreg Clayton CommandObjectBreakpointModify::CommandOptions::SetOptionValue (uint32_t option_idx, const char *option_arg)
13951b54c88cSJim Ingham {
13961b54c88cSJim Ingham     Error error;
13971b54c88cSJim Ingham     char short_option = (char) m_getopt_table[option_idx].val;
13981b54c88cSJim Ingham 
13991b54c88cSJim Ingham     switch (short_option)
14001b54c88cSJim Ingham     {
140136f3b369SJim Ingham         case 'c':
140236f3b369SJim Ingham             if (option_arg != NULL)
1403357132ebSGreg Clayton                 m_condition.assign (option_arg);
140436f3b369SJim Ingham             else
140536f3b369SJim Ingham                 m_condition.clear();
140636f3b369SJim Ingham             m_condition_passed = true;
140736f3b369SJim Ingham             break;
1408ae1c4cf5SJim Ingham         case 'd':
1409ae1c4cf5SJim Ingham             m_enable_passed = true;
1410ae1c4cf5SJim Ingham             m_enable_value = false;
1411ae1c4cf5SJim Ingham             break;
1412ae1c4cf5SJim Ingham         case 'e':
1413ae1c4cf5SJim Ingham             m_enable_passed = true;
1414ae1c4cf5SJim Ingham             m_enable_value = true;
1415ae1c4cf5SJim Ingham             break;
1416ed8a705cSGreg Clayton         case 'i':
14171b54c88cSJim Ingham         {
14180292f4a5SJim Ingham             m_ignore_count = Args::StringToUInt32(option_arg, UINT32_MAX, 0);
1419c982c768SGreg Clayton             if (m_ignore_count == UINT32_MAX)
14200292f4a5SJim Ingham                error.SetErrorStringWithFormat ("Invalid ignore count '%s'.\n", option_arg);
14211b54c88cSJim Ingham         }
1422ae1c4cf5SJim Ingham         break;
14231b54c88cSJim Ingham         case 't' :
14241b54c88cSJim Ingham         {
14250292f4a5SJim Ingham             if (option_arg[0] == '\0')
1426e0a97848SJim Ingham             {
1427e0a97848SJim Ingham                 m_thread_id = LLDB_INVALID_THREAD_ID;
1428e0a97848SJim Ingham                 m_thread_id_passed = true;
1429e0a97848SJim Ingham             }
1430e0a97848SJim Ingham             else
1431e0a97848SJim Ingham             {
14320292f4a5SJim Ingham                 m_thread_id = Args::StringToUInt64(option_arg, LLDB_INVALID_THREAD_ID, 0);
14331b54c88cSJim Ingham                 if (m_thread_id == LLDB_INVALID_THREAD_ID)
14340292f4a5SJim Ingham                    error.SetErrorStringWithFormat ("Invalid thread id string '%s'.\n", option_arg);
1435e0a97848SJim Ingham                 else
1436e0a97848SJim Ingham                     m_thread_id_passed = true;
1437e0a97848SJim Ingham             }
14381b54c88cSJim Ingham         }
14391b54c88cSJim Ingham         break;
14401b54c88cSJim Ingham         case 'T':
1441b2a38a72SJim Ingham             if (option_arg != NULL)
1442357132ebSGreg Clayton                 m_thread_name.assign (option_arg);
1443b2a38a72SJim Ingham             else
1444b2a38a72SJim Ingham                 m_thread_name.clear();
1445b2a38a72SJim Ingham             m_name_passed = true;
14461b54c88cSJim Ingham             break;
14471b54c88cSJim Ingham         case 'q':
1448b2a38a72SJim Ingham             if (option_arg != NULL)
1449357132ebSGreg Clayton                 m_queue_name.assign (option_arg);
1450b2a38a72SJim Ingham             else
1451b2a38a72SJim Ingham                 m_queue_name.clear();
1452b2a38a72SJim Ingham             m_queue_passed = true;
14531b54c88cSJim Ingham             break;
14541b54c88cSJim Ingham         case 'x':
14551b54c88cSJim Ingham         {
14560292f4a5SJim Ingham             if (option_arg[0] == '\n')
1457e0a97848SJim Ingham             {
1458e0a97848SJim Ingham                 m_thread_index = UINT32_MAX;
1459e0a97848SJim Ingham                 m_thread_index_passed = true;
1460e0a97848SJim Ingham             }
1461e0a97848SJim Ingham             else
1462e0a97848SJim Ingham             {
14630292f4a5SJim Ingham                 m_thread_index = Args::StringToUInt32 (option_arg, UINT32_MAX, 0);
1464c982c768SGreg Clayton                 if (m_thread_id == UINT32_MAX)
14650292f4a5SJim Ingham                    error.SetErrorStringWithFormat ("Invalid thread index string '%s'.\n", option_arg);
1466e0a97848SJim Ingham                 else
1467e0a97848SJim Ingham                     m_thread_index_passed = true;
1468e0a97848SJim Ingham             }
14691b54c88cSJim Ingham         }
14701b54c88cSJim Ingham         break;
14711b54c88cSJim Ingham         default:
14721b54c88cSJim Ingham             error.SetErrorStringWithFormat ("Unrecognized option '%c'.\n", short_option);
14731b54c88cSJim Ingham             break;
14741b54c88cSJim Ingham     }
14751b54c88cSJim Ingham 
14761b54c88cSJim Ingham     return error;
14771b54c88cSJim Ingham }
14781b54c88cSJim Ingham 
14791b54c88cSJim Ingham void
1480f6b8b581SGreg Clayton CommandObjectBreakpointModify::CommandOptions::OptionParsingStarting ()
14811b54c88cSJim Ingham {
1482c982c768SGreg Clayton     m_ignore_count = 0;
14831b54c88cSJim Ingham     m_thread_id = LLDB_INVALID_THREAD_ID;
1484e0a97848SJim Ingham     m_thread_id_passed = false;
1485c982c768SGreg Clayton     m_thread_index = UINT32_MAX;
1486e0a97848SJim Ingham     m_thread_index_passed = false;
14871b54c88cSJim Ingham     m_thread_name.clear();
14881b54c88cSJim Ingham     m_queue_name.clear();
148936f3b369SJim Ingham     m_condition.clear();
1490ae1c4cf5SJim Ingham     m_enable_passed = false;
1491b2a38a72SJim Ingham     m_queue_passed = false;
1492b2a38a72SJim Ingham     m_name_passed = false;
149336f3b369SJim Ingham     m_condition_passed = false;
14941b54c88cSJim Ingham }
14951b54c88cSJim Ingham 
14961b54c88cSJim Ingham //-------------------------------------------------------------------------
1497ae1c4cf5SJim Ingham // CommandObjectBreakpointModify
14981b54c88cSJim Ingham //-------------------------------------------------------------------------
1499ae1c4cf5SJim Ingham #pragma mark Modify
15001b54c88cSJim Ingham 
1501a7015092SGreg Clayton CommandObjectBreakpointModify::CommandObjectBreakpointModify (CommandInterpreter &interpreter) :
1502a7015092SGreg Clayton     CommandObject (interpreter,
1503a7015092SGreg Clayton                    "breakpoint modify",
1504a571c010SJim Ingham                    "Modify the options on a breakpoint or set of breakpoints in the executable.  "
1505e0a97848SJim Ingham                    "If no breakpoint is specified, acts on the last created breakpoint.  "
1506e0a97848SJim Ingham                    "With the exception of -e, -d and -i, passing an empty argument clears the modification.",
1507eb0103f2SGreg Clayton                    NULL),
1508eb0103f2SGreg Clayton     m_options (interpreter)
15091b54c88cSJim Ingham {
1510e139cf23SCaroline Tice     CommandArgumentEntry arg;
1511e139cf23SCaroline Tice     CommandArgumentData bp_id_arg;
1512e139cf23SCaroline Tice     CommandArgumentData bp_id_range_arg;
1513e139cf23SCaroline Tice 
1514e139cf23SCaroline Tice     // Create the first variant for the first (and only) argument for this command.
1515e139cf23SCaroline Tice     bp_id_arg.arg_type = eArgTypeBreakpointID;
1516405fe67fSCaroline Tice     bp_id_arg.arg_repetition = eArgRepeatPlain;
1517e139cf23SCaroline Tice 
1518e139cf23SCaroline Tice     // Create the second variant for the first (and only) argument for this command.
1519e139cf23SCaroline Tice     bp_id_range_arg.arg_type = eArgTypeBreakpointIDRange;
1520405fe67fSCaroline Tice     bp_id_range_arg.arg_repetition = eArgRepeatPlain;
1521e139cf23SCaroline Tice 
1522e139cf23SCaroline Tice     // The first (and only) argument for this command could be either a bp_id or a bp_id_range.
1523e139cf23SCaroline Tice     // Push both variants into the entry for the first argument for this command.
1524e139cf23SCaroline Tice     arg.push_back (bp_id_arg);
1525e139cf23SCaroline Tice     arg.push_back (bp_id_range_arg);
1526e139cf23SCaroline Tice 
1527e139cf23SCaroline Tice     // Add the entry for the first argument for this command to the object's arguments vector.
1528e139cf23SCaroline Tice     m_arguments.push_back (arg);
15291b54c88cSJim Ingham }
15301b54c88cSJim Ingham 
1531ae1c4cf5SJim Ingham CommandObjectBreakpointModify::~CommandObjectBreakpointModify ()
15321b54c88cSJim Ingham {
15331b54c88cSJim Ingham }
15341b54c88cSJim Ingham 
15351b54c88cSJim Ingham Options *
1536ae1c4cf5SJim Ingham CommandObjectBreakpointModify::GetOptions ()
15371b54c88cSJim Ingham {
15381b54c88cSJim Ingham     return &m_options;
15391b54c88cSJim Ingham }
15401b54c88cSJim Ingham 
15411b54c88cSJim Ingham bool
1542ae1c4cf5SJim Ingham CommandObjectBreakpointModify::Execute
15431b54c88cSJim Ingham (
15441b54c88cSJim Ingham     Args& command,
15451b54c88cSJim Ingham     CommandReturnObject &result
15461b54c88cSJim Ingham )
15471b54c88cSJim Ingham {
1548a7015092SGreg Clayton     Target *target = m_interpreter.GetDebugger().GetSelectedTarget().get();
15491b54c88cSJim Ingham     if (target == NULL)
15501b54c88cSJim Ingham     {
15519068d794SCaroline Tice         result.AppendError ("Invalid target.  No existing target or breakpoints.");
15521b54c88cSJim Ingham         result.SetStatus (eReturnStatusFailed);
15531b54c88cSJim Ingham         return false;
15541b54c88cSJim Ingham     }
15551b54c88cSJim Ingham 
15561b54c88cSJim Ingham     Mutex::Locker locker;
15571b54c88cSJim Ingham     target->GetBreakpointList().GetListMutex(locker);
15581b54c88cSJim Ingham 
15591b54c88cSJim Ingham     BreakpointIDList valid_bp_ids;
15601b54c88cSJim Ingham 
15611b54c88cSJim Ingham     CommandObjectMultiwordBreakpoint::VerifyBreakpointIDs (command, target, result, &valid_bp_ids);
15621b54c88cSJim Ingham 
15631b54c88cSJim Ingham     if (result.Succeeded())
15641b54c88cSJim Ingham     {
1565c982c768SGreg Clayton         const size_t count = valid_bp_ids.GetSize();
1566c982c768SGreg Clayton         for (size_t i = 0; i < count; ++i)
15671b54c88cSJim Ingham         {
15681b54c88cSJim Ingham             BreakpointID cur_bp_id = valid_bp_ids.GetBreakpointIDAtIndex (i);
15691b54c88cSJim Ingham 
15701b54c88cSJim Ingham             if (cur_bp_id.GetBreakpointID() != LLDB_INVALID_BREAK_ID)
15711b54c88cSJim Ingham             {
15721b54c88cSJim Ingham                 Breakpoint *bp = target->GetBreakpointByID (cur_bp_id.GetBreakpointID()).get();
15731b54c88cSJim Ingham                 if (cur_bp_id.GetLocationID() != LLDB_INVALID_BREAK_ID)
15741b54c88cSJim Ingham                 {
15751b54c88cSJim Ingham                     BreakpointLocation *location = bp->FindLocationByID (cur_bp_id.GetLocationID()).get();
15761b54c88cSJim Ingham                     if (location)
15771b54c88cSJim Ingham                     {
1578e0a97848SJim Ingham                         if (m_options.m_thread_id_passed)
15791b54c88cSJim Ingham                             location->SetThreadID (m_options.m_thread_id);
15801b54c88cSJim Ingham 
1581e0a97848SJim Ingham                         if (m_options.m_thread_index_passed)
15821b54c88cSJim Ingham                             location->GetLocationOptions()->GetThreadSpec()->SetIndex(m_options.m_thread_index);
15831b54c88cSJim Ingham 
1584b2a38a72SJim Ingham                         if (m_options.m_name_passed)
15851b54c88cSJim Ingham                             location->GetLocationOptions()->GetThreadSpec()->SetName(m_options.m_thread_name.c_str());
15861b54c88cSJim Ingham 
1587b2a38a72SJim Ingham                         if (m_options.m_queue_passed)
15881b54c88cSJim Ingham                             location->GetLocationOptions()->GetThreadSpec()->SetQueueName(m_options.m_queue_name.c_str());
15891b54c88cSJim Ingham 
1590c982c768SGreg Clayton                         if (m_options.m_ignore_count != 0)
15911b54c88cSJim Ingham                             location->GetLocationOptions()->SetIgnoreCount(m_options.m_ignore_count);
1592ae1c4cf5SJim Ingham 
1593ae1c4cf5SJim Ingham                         if (m_options.m_enable_passed)
1594ae1c4cf5SJim Ingham                             location->SetEnabled (m_options.m_enable_value);
159536f3b369SJim Ingham 
159636f3b369SJim Ingham                         if (m_options.m_condition_passed)
159736f3b369SJim Ingham                             location->SetCondition (m_options.m_condition.c_str());
15981b54c88cSJim Ingham                     }
15991b54c88cSJim Ingham                 }
16001b54c88cSJim Ingham                 else
16011b54c88cSJim Ingham                 {
1602e0a97848SJim Ingham                     if (m_options.m_thread_id_passed)
16031b54c88cSJim Ingham                         bp->SetThreadID (m_options.m_thread_id);
16041b54c88cSJim Ingham 
1605e0a97848SJim Ingham                     if (m_options.m_thread_index_passed)
16061b54c88cSJim Ingham                         bp->GetOptions()->GetThreadSpec()->SetIndex(m_options.m_thread_index);
16071b54c88cSJim Ingham 
1608b2a38a72SJim Ingham                     if (m_options.m_name_passed)
16091b54c88cSJim Ingham                         bp->GetOptions()->GetThreadSpec()->SetName(m_options.m_thread_name.c_str());
16101b54c88cSJim Ingham 
1611b2a38a72SJim Ingham                     if (m_options.m_queue_passed)
16121b54c88cSJim Ingham                         bp->GetOptions()->GetThreadSpec()->SetQueueName(m_options.m_queue_name.c_str());
16131b54c88cSJim Ingham 
1614c982c768SGreg Clayton                     if (m_options.m_ignore_count != 0)
16151b54c88cSJim Ingham                         bp->GetOptions()->SetIgnoreCount(m_options.m_ignore_count);
1616ae1c4cf5SJim Ingham 
1617ae1c4cf5SJim Ingham                     if (m_options.m_enable_passed)
1618ae1c4cf5SJim Ingham                         bp->SetEnabled (m_options.m_enable_value);
1619ae1c4cf5SJim Ingham 
162036f3b369SJim Ingham                     if (m_options.m_condition_passed)
162136f3b369SJim Ingham                         bp->SetCondition (m_options.m_condition.c_str());
16221b54c88cSJim Ingham                 }
16231b54c88cSJim Ingham             }
16241b54c88cSJim Ingham         }
16251b54c88cSJim Ingham     }
16261b54c88cSJim Ingham 
16271b54c88cSJim Ingham     return result.Succeeded();
16281b54c88cSJim Ingham }
16291b54c88cSJim Ingham 
16301b54c88cSJim Ingham 
1631