130fdc8d8SChris Lattner //===-- ThreadPlanStepUntil.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 //m_should_stop
1030fdc8d8SChris Lattner 
1130fdc8d8SChris Lattner //
1230fdc8d8SChris Lattner //===----------------------------------------------------------------------===//
1330fdc8d8SChris Lattner 
1430fdc8d8SChris Lattner #include "lldb/Target/ThreadPlanStepUntil.h"
1530fdc8d8SChris Lattner 
1630fdc8d8SChris Lattner // C Includes
1730fdc8d8SChris Lattner // C++ Includes
1830fdc8d8SChris Lattner // Other libraries and framework includes
1930fdc8d8SChris Lattner // Project includes
2030fdc8d8SChris Lattner #include "lldb/Breakpoint/Breakpoint.h"
2130fdc8d8SChris Lattner #include "lldb/lldb-private-log.h"
2230fdc8d8SChris Lattner #include "lldb/Core/Log.h"
2330fdc8d8SChris Lattner #include "lldb/Target/Process.h"
2430fdc8d8SChris Lattner #include "lldb/Target/RegisterContext.h"
25f4b47e15SGreg Clayton #include "lldb/Target/StopInfo.h"
2630fdc8d8SChris Lattner #include "lldb/Target/Target.h"
2730fdc8d8SChris Lattner 
2830fdc8d8SChris Lattner using namespace lldb;
2930fdc8d8SChris Lattner using namespace lldb_private;
3030fdc8d8SChris Lattner 
3130fdc8d8SChris Lattner //----------------------------------------------------------------------
3230fdc8d8SChris Lattner // ThreadPlanStepUntil: Run until we reach a given line number or step out of the current frame
3330fdc8d8SChris Lattner //----------------------------------------------------------------------
3430fdc8d8SChris Lattner 
3530fdc8d8SChris Lattner ThreadPlanStepUntil::ThreadPlanStepUntil
3630fdc8d8SChris Lattner (
3730fdc8d8SChris Lattner     Thread &thread,
3830fdc8d8SChris Lattner     lldb::addr_t *address_list,
3930fdc8d8SChris Lattner     size_t num_addresses,
40481cef25SGreg Clayton     bool stop_others,
41481cef25SGreg Clayton     uint32_t frame_idx
4230fdc8d8SChris Lattner ) :
43b01e742aSJim Ingham     ThreadPlan (ThreadPlan::eKindStepUntil, "Step until", thread, eVoteNoOpinion, eVoteNoOpinion),
4430fdc8d8SChris Lattner     m_step_from_insn (LLDB_INVALID_ADDRESS),
4530fdc8d8SChris Lattner     m_return_bp_id (LLDB_INVALID_BREAK_ID),
46c982c768SGreg Clayton     m_return_addr (LLDB_INVALID_ADDRESS),
4730fdc8d8SChris Lattner     m_stepped_out (false),
4830fdc8d8SChris Lattner     m_should_stop (false),
4930fdc8d8SChris Lattner     m_ran_analyze (false),
50c982c768SGreg Clayton     m_explains_stop (false),
51c982c768SGreg Clayton     m_until_points (),
5230fdc8d8SChris Lattner     m_stop_others (stop_others)
5330fdc8d8SChris Lattner {
5430fdc8d8SChris Lattner 
5530fdc8d8SChris Lattner     SetOkayToDiscard(true);
5630fdc8d8SChris Lattner     // Stash away our "until" addresses:
571ac04c30SGreg Clayton     TargetSP target_sp (m_thread.CalculateTarget());
5830fdc8d8SChris Lattner 
59481cef25SGreg Clayton     StackFrameSP frame_sp (m_thread.GetStackFrameAtIndex (frame_idx));
60481cef25SGreg Clayton     if (frame_sp)
61481cef25SGreg Clayton     {
62481cef25SGreg Clayton         m_step_from_insn = frame_sp->GetStackID().GetPC();
6330fdc8d8SChris Lattner         lldb::user_id_t thread_id = m_thread.GetID();
6430fdc8d8SChris Lattner 
6530fdc8d8SChris Lattner         // Find the return address and set a breakpoint there:
6630fdc8d8SChris Lattner         // FIXME - can we do this more securely if we know first_insn?
6730fdc8d8SChris Lattner 
68481cef25SGreg Clayton         StackFrameSP return_frame_sp (m_thread.GetStackFrameAtIndex(frame_idx + 1));
69481cef25SGreg Clayton         if (return_frame_sp)
70481cef25SGreg Clayton         {
719da7bd07SGreg Clayton             // TODO: add inline functionality
72481cef25SGreg Clayton             m_return_addr = return_frame_sp->GetStackID().GetPC();
731ac04c30SGreg Clayton             Breakpoint *return_bp = target_sp->CreateBreakpoint (m_return_addr, true).get();
7430fdc8d8SChris Lattner             if (return_bp != NULL)
7530fdc8d8SChris Lattner             {
7630fdc8d8SChris Lattner                 return_bp->SetThreadID(thread_id);
7730fdc8d8SChris Lattner                 m_return_bp_id = return_bp->GetID();
7830fdc8d8SChris Lattner             }
7930fdc8d8SChris Lattner         }
8030fdc8d8SChris Lattner 
81*b5c0d1ccSJim Ingham         m_stack_id = m_thread.GetStackFrameAtIndex(frame_idx)->GetStackID();
8230fdc8d8SChris Lattner 
8330fdc8d8SChris Lattner         // Now set breakpoints on all our return addresses:
8430fdc8d8SChris Lattner         for (int i = 0; i < num_addresses; i++)
8530fdc8d8SChris Lattner         {
861ac04c30SGreg Clayton             Breakpoint *until_bp = target_sp->CreateBreakpoint (address_list[i], true).get();
8730fdc8d8SChris Lattner             if (until_bp != NULL)
8830fdc8d8SChris Lattner             {
8930fdc8d8SChris Lattner                 until_bp->SetThreadID(thread_id);
9030fdc8d8SChris Lattner                 m_until_points[address_list[i]] = until_bp->GetID();
9130fdc8d8SChris Lattner             }
9230fdc8d8SChris Lattner             else
9330fdc8d8SChris Lattner             {
9430fdc8d8SChris Lattner                 m_until_points[address_list[i]] = LLDB_INVALID_BREAK_ID;
9530fdc8d8SChris Lattner             }
9630fdc8d8SChris Lattner         }
9730fdc8d8SChris Lattner     }
98481cef25SGreg Clayton }
9930fdc8d8SChris Lattner 
10030fdc8d8SChris Lattner ThreadPlanStepUntil::~ThreadPlanStepUntil ()
10130fdc8d8SChris Lattner {
10230fdc8d8SChris Lattner     Clear();
10330fdc8d8SChris Lattner }
10430fdc8d8SChris Lattner 
10530fdc8d8SChris Lattner void
10630fdc8d8SChris Lattner ThreadPlanStepUntil::Clear()
10730fdc8d8SChris Lattner {
1081ac04c30SGreg Clayton     TargetSP target_sp (m_thread.CalculateTarget());
1091ac04c30SGreg Clayton     if (target_sp)
1101ac04c30SGreg Clayton     {
11130fdc8d8SChris Lattner         if (m_return_bp_id != LLDB_INVALID_BREAK_ID)
11230fdc8d8SChris Lattner         {
1131ac04c30SGreg Clayton             target_sp->RemoveBreakpointByID(m_return_bp_id);
11430fdc8d8SChris Lattner             m_return_bp_id = LLDB_INVALID_BREAK_ID;
11530fdc8d8SChris Lattner         }
11630fdc8d8SChris Lattner 
11730fdc8d8SChris Lattner         until_collection::iterator pos, end = m_until_points.end();
11830fdc8d8SChris Lattner         for (pos = m_until_points.begin(); pos != end; pos++)
11930fdc8d8SChris Lattner         {
1201ac04c30SGreg Clayton             target_sp->RemoveBreakpointByID((*pos).second);
1211ac04c30SGreg Clayton         }
12230fdc8d8SChris Lattner     }
12330fdc8d8SChris Lattner     m_until_points.clear();
12430fdc8d8SChris Lattner }
12530fdc8d8SChris Lattner 
12630fdc8d8SChris Lattner void
12730fdc8d8SChris Lattner ThreadPlanStepUntil::GetDescription (Stream *s, lldb::DescriptionLevel level)
12830fdc8d8SChris Lattner {
12930fdc8d8SChris Lattner     if (level == lldb::eDescriptionLevelBrief)
13030fdc8d8SChris Lattner     {
13130fdc8d8SChris Lattner         s->Printf ("step until");
13230fdc8d8SChris Lattner         if (m_stepped_out)
13330fdc8d8SChris Lattner             s->Printf (" - stepped out");
13430fdc8d8SChris Lattner     }
13530fdc8d8SChris Lattner     else
13630fdc8d8SChris Lattner     {
13730fdc8d8SChris Lattner         if (m_until_points.size() == 1)
13830fdc8d8SChris Lattner             s->Printf ("Stepping from address 0x%llx until we reach 0x%llx using breakpoint %d",
13930fdc8d8SChris Lattner                        (uint64_t)m_step_from_insn,
14030fdc8d8SChris Lattner                        (uint64_t) (*m_until_points.begin()).first,
14130fdc8d8SChris Lattner                        (*m_until_points.begin()).second);
14230fdc8d8SChris Lattner         else
14330fdc8d8SChris Lattner         {
14430fdc8d8SChris Lattner             until_collection::iterator pos, end = m_until_points.end();
14530fdc8d8SChris Lattner             s->Printf ("Stepping from address 0x%llx until we reach one of:",
14630fdc8d8SChris Lattner                        (uint64_t)m_step_from_insn);
14730fdc8d8SChris Lattner             for (pos = m_until_points.begin(); pos != end; pos++)
14830fdc8d8SChris Lattner             {
14930fdc8d8SChris Lattner                 s->Printf ("\n\t0x%llx (bp: %d)", (uint64_t) (*pos).first, (*pos).second);
15030fdc8d8SChris Lattner             }
15130fdc8d8SChris Lattner         }
152fd54b368SJason Molenda         s->Printf(" stepped out address is 0x%llx.", (uint64_t) m_return_addr);
15330fdc8d8SChris Lattner     }
15430fdc8d8SChris Lattner }
15530fdc8d8SChris Lattner 
15630fdc8d8SChris Lattner bool
15730fdc8d8SChris Lattner ThreadPlanStepUntil::ValidatePlan (Stream *error)
15830fdc8d8SChris Lattner {
15930fdc8d8SChris Lattner     if (m_return_bp_id == LLDB_INVALID_BREAK_ID)
16030fdc8d8SChris Lattner         return false;
16130fdc8d8SChris Lattner     else
16230fdc8d8SChris Lattner     {
16330fdc8d8SChris Lattner         until_collection::iterator pos, end = m_until_points.end();
16430fdc8d8SChris Lattner         for (pos = m_until_points.begin(); pos != end; pos++)
16530fdc8d8SChris Lattner         {
16630fdc8d8SChris Lattner             if (!LLDB_BREAK_ID_IS_VALID ((*pos).second))
16730fdc8d8SChris Lattner                 return false;
16830fdc8d8SChris Lattner         }
16930fdc8d8SChris Lattner         return true;
17030fdc8d8SChris Lattner     }
17130fdc8d8SChris Lattner }
17230fdc8d8SChris Lattner 
17330fdc8d8SChris Lattner void
17430fdc8d8SChris Lattner ThreadPlanStepUntil::AnalyzeStop()
17530fdc8d8SChris Lattner {
17630fdc8d8SChris Lattner     if (m_ran_analyze)
17730fdc8d8SChris Lattner         return;
17830fdc8d8SChris Lattner 
179b15bfc75SJim Ingham     StopInfoSP stop_info_sp = GetPrivateStopReason();
18030fdc8d8SChris Lattner     m_should_stop = true;
18130fdc8d8SChris Lattner     m_explains_stop = false;
18230fdc8d8SChris Lattner 
183b15bfc75SJim Ingham     if (stop_info_sp)
18430fdc8d8SChris Lattner     {
185b15bfc75SJim Ingham         StopReason reason = stop_info_sp->GetStopReason();
18630fdc8d8SChris Lattner 
18730fdc8d8SChris Lattner         switch (reason)
18830fdc8d8SChris Lattner         {
18930fdc8d8SChris Lattner             case eStopReasonBreakpoint:
19030fdc8d8SChris Lattner             {
19130fdc8d8SChris Lattner                 // If this is OUR breakpoint, we're fine, otherwise we don't know why this happened...
1921ac04c30SGreg Clayton                 BreakpointSiteSP this_site = m_thread.GetProcess()->GetBreakpointSiteList().FindByID (stop_info_sp->GetValue());
19330fdc8d8SChris Lattner                 if (!this_site)
19430fdc8d8SChris Lattner                 {
19530fdc8d8SChris Lattner                     m_explains_stop = false;
19630fdc8d8SChris Lattner                     return;
19730fdc8d8SChris Lattner                 }
19830fdc8d8SChris Lattner 
19930fdc8d8SChris Lattner                 if (this_site->IsBreakpointAtThisSite (m_return_bp_id))
20030fdc8d8SChris Lattner                 {
20130fdc8d8SChris Lattner                     // If we are at our "step out" breakpoint, and the stack depth has shrunk, then
20230fdc8d8SChris Lattner                     // this is indeed our stop.
20330fdc8d8SChris Lattner                     // If the stack depth has grown, then we've hit our step out breakpoint recursively.
20430fdc8d8SChris Lattner                     // If we are the only breakpoint at that location, then we do explain the stop, and
20530fdc8d8SChris Lattner                     // we'll just continue.
20630fdc8d8SChris Lattner                     // If there was another breakpoint here, then we don't explain the stop, but we won't
20730fdc8d8SChris Lattner                     // mark ourselves Completed, because maybe that breakpoint will continue, and then
20830fdc8d8SChris Lattner                     // we'll finish the "until".
209*b5c0d1ccSJim Ingham                     bool done;
210*b5c0d1ccSJim Ingham                     StackID cur_frame_zero_id;
211*b5c0d1ccSJim Ingham 
212*b5c0d1ccSJim Ingham                     if (m_stack_id < cur_frame_zero_id)
213*b5c0d1ccSJim Ingham                         done = true;
214*b5c0d1ccSJim Ingham                     else
215*b5c0d1ccSJim Ingham                         done = false;
216*b5c0d1ccSJim Ingham 
217*b5c0d1ccSJim Ingham                     if (done)
21830fdc8d8SChris Lattner                     {
21930fdc8d8SChris Lattner                         m_stepped_out = true;
22030fdc8d8SChris Lattner                         SetPlanComplete();
22130fdc8d8SChris Lattner                     }
22230fdc8d8SChris Lattner                     else
22330fdc8d8SChris Lattner                         m_should_stop = false;
22430fdc8d8SChris Lattner 
22530fdc8d8SChris Lattner                     if (this_site->GetNumberOfOwners() == 1)
22630fdc8d8SChris Lattner                         m_explains_stop = true;
22730fdc8d8SChris Lattner                     else
22830fdc8d8SChris Lattner                         m_explains_stop = false;
22930fdc8d8SChris Lattner                     return;
23030fdc8d8SChris Lattner                 }
23130fdc8d8SChris Lattner                 else
23230fdc8d8SChris Lattner                 {
23330fdc8d8SChris Lattner                     // Check if we've hit one of our "until" breakpoints.
23430fdc8d8SChris Lattner                     until_collection::iterator pos, end = m_until_points.end();
23530fdc8d8SChris Lattner                     for (pos = m_until_points.begin(); pos != end; pos++)
23630fdc8d8SChris Lattner                     {
23730fdc8d8SChris Lattner                         if (this_site->IsBreakpointAtThisSite ((*pos).second))
23830fdc8d8SChris Lattner                         {
23930fdc8d8SChris Lattner                             // If we're at the right stack depth, then we're done.
240*b5c0d1ccSJim Ingham 
241*b5c0d1ccSJim Ingham                             bool done;
242*b5c0d1ccSJim Ingham                             StackID frame_zero_id = m_thread.GetStackFrameAtIndex(0)->GetStackID();
243*b5c0d1ccSJim Ingham 
244*b5c0d1ccSJim Ingham                             if (frame_zero_id == m_stack_id)
245*b5c0d1ccSJim Ingham                                 done = true;
246*b5c0d1ccSJim Ingham                             else if (frame_zero_id < m_stack_id)
247*b5c0d1ccSJim Ingham                                 done = false;
248*b5c0d1ccSJim Ingham                             else
249*b5c0d1ccSJim Ingham                             {
250*b5c0d1ccSJim Ingham                                 StackFrameSP older_frame_sp = m_thread.GetStackFrameAtIndex(1);
251*b5c0d1ccSJim Ingham 
252*b5c0d1ccSJim Ingham                                 // But if we can't even unwind one frame we should just get out of here & stop...
253*b5c0d1ccSJim Ingham                                 if (older_frame_sp)
254*b5c0d1ccSJim Ingham                                 {
255*b5c0d1ccSJim Ingham                                     const SymbolContext &older_context
256*b5c0d1ccSJim Ingham                                         = older_frame_sp->GetSymbolContext(eSymbolContextEverything);
257*b5c0d1ccSJim Ingham                                     SymbolContext stack_context;
258*b5c0d1ccSJim Ingham                                     m_stack_id.GetSymbolContextScope()->CalculateSymbolContext(&stack_context);
259*b5c0d1ccSJim Ingham 
260*b5c0d1ccSJim Ingham                                     if (older_context == stack_context)
261*b5c0d1ccSJim Ingham                                         done = true;
262*b5c0d1ccSJim Ingham                                     else
263*b5c0d1ccSJim Ingham                                         done = false;
264*b5c0d1ccSJim Ingham                                 }
265*b5c0d1ccSJim Ingham                                 else
266*b5c0d1ccSJim Ingham                                     done = false;
267*b5c0d1ccSJim Ingham                             }
268*b5c0d1ccSJim Ingham 
269*b5c0d1ccSJim Ingham                             if (done)
27030fdc8d8SChris Lattner                                 SetPlanComplete();
27130fdc8d8SChris Lattner                             else
27230fdc8d8SChris Lattner                                 m_should_stop = false;
27330fdc8d8SChris Lattner 
27430fdc8d8SChris Lattner                             // Otherwise we've hit this breakpoint recursively.  If we're the
27530fdc8d8SChris Lattner                             // only breakpoint here, then we do explain the stop, and we'll continue.
27630fdc8d8SChris Lattner                             // If not then we should let higher plans handle this stop.
27730fdc8d8SChris Lattner                             if (this_site->GetNumberOfOwners() == 1)
27830fdc8d8SChris Lattner                                 m_explains_stop = true;
27930fdc8d8SChris Lattner                             else
28030fdc8d8SChris Lattner                             {
28130fdc8d8SChris Lattner                                 m_should_stop = true;
28230fdc8d8SChris Lattner                                 m_explains_stop = false;
28330fdc8d8SChris Lattner                             }
28430fdc8d8SChris Lattner                             return;
28530fdc8d8SChris Lattner                         }
28630fdc8d8SChris Lattner                     }
28730fdc8d8SChris Lattner                 }
28830fdc8d8SChris Lattner                 // If we get here we haven't hit any of our breakpoints, so let the higher
28930fdc8d8SChris Lattner                 // plans take care of the stop.
29030fdc8d8SChris Lattner                 m_explains_stop = false;
29130fdc8d8SChris Lattner                 return;
29230fdc8d8SChris Lattner             }
29330fdc8d8SChris Lattner             case eStopReasonWatchpoint:
29430fdc8d8SChris Lattner             case eStopReasonSignal:
29530fdc8d8SChris Lattner             case eStopReasonException:
29630fdc8d8SChris Lattner                 m_explains_stop = false;
29730fdc8d8SChris Lattner                 break;
29830fdc8d8SChris Lattner             default:
29930fdc8d8SChris Lattner                 m_explains_stop = true;
30030fdc8d8SChris Lattner                 break;
30130fdc8d8SChris Lattner         }
30230fdc8d8SChris Lattner     }
30330fdc8d8SChris Lattner }
30430fdc8d8SChris Lattner 
30530fdc8d8SChris Lattner bool
30630fdc8d8SChris Lattner ThreadPlanStepUntil::PlanExplainsStop ()
30730fdc8d8SChris Lattner {
30830fdc8d8SChris Lattner     // We don't explain signals or breakpoints (breakpoints that handle stepping in or
30930fdc8d8SChris Lattner     // out will be handled by a child plan.
31030fdc8d8SChris Lattner     AnalyzeStop();
31130fdc8d8SChris Lattner     return m_explains_stop;
31230fdc8d8SChris Lattner }
31330fdc8d8SChris Lattner 
31430fdc8d8SChris Lattner bool
31530fdc8d8SChris Lattner ThreadPlanStepUntil::ShouldStop (Event *event_ptr)
31630fdc8d8SChris Lattner {
31730fdc8d8SChris Lattner     // If we've told our self in ExplainsStop that we plan to continue, then
31830fdc8d8SChris Lattner     // do so here.  Otherwise, as long as this thread has stopped for a reason,
31930fdc8d8SChris Lattner     // we will stop.
32030fdc8d8SChris Lattner 
321b15bfc75SJim Ingham     StopInfoSP stop_info_sp = GetPrivateStopReason();
322b15bfc75SJim Ingham     if (stop_info_sp == NULL || stop_info_sp->GetStopReason() == eStopReasonNone)
32330fdc8d8SChris Lattner         return false;
32430fdc8d8SChris Lattner 
32530fdc8d8SChris Lattner     AnalyzeStop();
32630fdc8d8SChris Lattner     return m_should_stop;
32730fdc8d8SChris Lattner }
32830fdc8d8SChris Lattner 
32930fdc8d8SChris Lattner bool
33030fdc8d8SChris Lattner ThreadPlanStepUntil::StopOthers ()
33130fdc8d8SChris Lattner {
33230fdc8d8SChris Lattner     return m_stop_others;
33330fdc8d8SChris Lattner }
33430fdc8d8SChris Lattner 
33530fdc8d8SChris Lattner StateType
33606e827ccSJim Ingham ThreadPlanStepUntil::GetPlanRunState ()
33730fdc8d8SChris Lattner {
33830fdc8d8SChris Lattner     return eStateRunning;
33930fdc8d8SChris Lattner }
34030fdc8d8SChris Lattner 
34130fdc8d8SChris Lattner bool
34230fdc8d8SChris Lattner ThreadPlanStepUntil::WillResume (StateType resume_state, bool current_plan)
34330fdc8d8SChris Lattner {
34430fdc8d8SChris Lattner     ThreadPlan::WillResume (resume_state, current_plan);
34530fdc8d8SChris Lattner     if (current_plan)
34630fdc8d8SChris Lattner     {
3471ac04c30SGreg Clayton         TargetSP target_sp (m_thread.CalculateTarget());
3481ac04c30SGreg Clayton         if (target_sp)
3491ac04c30SGreg Clayton         {
3501ac04c30SGreg Clayton             Breakpoint *return_bp = target_sp->GetBreakpointByID(m_return_bp_id).get();
35130fdc8d8SChris Lattner             if (return_bp != NULL)
35230fdc8d8SChris Lattner                 return_bp->SetEnabled (true);
35330fdc8d8SChris Lattner 
35430fdc8d8SChris Lattner             until_collection::iterator pos, end = m_until_points.end();
35530fdc8d8SChris Lattner             for (pos = m_until_points.begin(); pos != end; pos++)
35630fdc8d8SChris Lattner             {
3571ac04c30SGreg Clayton                 Breakpoint *until_bp = target_sp->GetBreakpointByID((*pos).second).get();
35830fdc8d8SChris Lattner                 if (until_bp != NULL)
35930fdc8d8SChris Lattner                     until_bp->SetEnabled (true);
36030fdc8d8SChris Lattner             }
36130fdc8d8SChris Lattner         }
3621ac04c30SGreg Clayton     }
36330fdc8d8SChris Lattner 
36430fdc8d8SChris Lattner     m_should_stop = true;
36530fdc8d8SChris Lattner     m_ran_analyze = false;
36630fdc8d8SChris Lattner     m_explains_stop = false;
36730fdc8d8SChris Lattner     return true;
36830fdc8d8SChris Lattner }
36930fdc8d8SChris Lattner 
37030fdc8d8SChris Lattner bool
37130fdc8d8SChris Lattner ThreadPlanStepUntil::WillStop ()
37230fdc8d8SChris Lattner {
3731ac04c30SGreg Clayton     TargetSP target_sp (m_thread.CalculateTarget());
3741ac04c30SGreg Clayton     if (target_sp)
3751ac04c30SGreg Clayton     {
3761ac04c30SGreg Clayton         Breakpoint *return_bp = target_sp->GetBreakpointByID(m_return_bp_id).get();
37730fdc8d8SChris Lattner         if (return_bp != NULL)
37830fdc8d8SChris Lattner             return_bp->SetEnabled (false);
37930fdc8d8SChris Lattner 
38030fdc8d8SChris Lattner         until_collection::iterator pos, end = m_until_points.end();
38130fdc8d8SChris Lattner         for (pos = m_until_points.begin(); pos != end; pos++)
38230fdc8d8SChris Lattner         {
3831ac04c30SGreg Clayton             Breakpoint *until_bp = target_sp->GetBreakpointByID((*pos).second).get();
38430fdc8d8SChris Lattner             if (until_bp != NULL)
38530fdc8d8SChris Lattner                 until_bp->SetEnabled (false);
38630fdc8d8SChris Lattner         }
3871ac04c30SGreg Clayton     }
38830fdc8d8SChris Lattner     return true;
38930fdc8d8SChris Lattner }
39030fdc8d8SChris Lattner 
39130fdc8d8SChris Lattner bool
39230fdc8d8SChris Lattner ThreadPlanStepUntil::MischiefManaged ()
39330fdc8d8SChris Lattner {
39430fdc8d8SChris Lattner 
39530fdc8d8SChris Lattner     // I'm letting "PlanExplainsStop" do all the work, and just reporting that here.
39630fdc8d8SChris Lattner     bool done = false;
39730fdc8d8SChris Lattner     if (IsPlanComplete())
39830fdc8d8SChris Lattner     {
3992d4edfbcSGreg Clayton         LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP));
40030fdc8d8SChris Lattner         if (log)
40130fdc8d8SChris Lattner             log->Printf("Completed step until plan.");
40230fdc8d8SChris Lattner 
40330fdc8d8SChris Lattner         Clear();
40430fdc8d8SChris Lattner         done = true;
40530fdc8d8SChris Lattner     }
40630fdc8d8SChris Lattner     if (done)
40730fdc8d8SChris Lattner         ThreadPlan::MischiefManaged ();
40830fdc8d8SChris Lattner 
40930fdc8d8SChris Lattner     return done;
41030fdc8d8SChris Lattner 
41130fdc8d8SChris Lattner }
41230fdc8d8SChris Lattner 
413