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 
1030fdc8d8SChris Lattner // C Includes
1130fdc8d8SChris Lattner // C++ Includes
1230fdc8d8SChris Lattner // Other libraries and framework includes
1330fdc8d8SChris Lattner // Project includes
14*e65b2cf2SEugene Zelenko #include "lldb/Target/ThreadPlanStepUntil.h"
1530fdc8d8SChris Lattner #include "lldb/Breakpoint/Breakpoint.h"
1630fdc8d8SChris Lattner #include "lldb/Core/Log.h"
1730fdc8d8SChris Lattner #include "lldb/Target/Process.h"
1830fdc8d8SChris Lattner #include "lldb/Target/RegisterContext.h"
19f4b47e15SGreg Clayton #include "lldb/Target/StopInfo.h"
2030fdc8d8SChris Lattner #include "lldb/Target/Target.h"
2130fdc8d8SChris Lattner 
2230fdc8d8SChris Lattner using namespace lldb;
2330fdc8d8SChris Lattner using namespace lldb_private;
2430fdc8d8SChris Lattner 
2530fdc8d8SChris Lattner //----------------------------------------------------------------------
2630fdc8d8SChris Lattner // ThreadPlanStepUntil: Run until we reach a given line number or step out of the current frame
2730fdc8d8SChris Lattner //----------------------------------------------------------------------
2830fdc8d8SChris Lattner 
2930fdc8d8SChris Lattner ThreadPlanStepUntil::ThreadPlanStepUntil
3030fdc8d8SChris Lattner (
3130fdc8d8SChris Lattner     Thread &thread,
3230fdc8d8SChris Lattner     lldb::addr_t *address_list,
3330fdc8d8SChris Lattner     size_t num_addresses,
34481cef25SGreg Clayton     bool stop_others,
35481cef25SGreg Clayton     uint32_t frame_idx
3630fdc8d8SChris Lattner ) :
37b01e742aSJim Ingham     ThreadPlan (ThreadPlan::eKindStepUntil, "Step until", thread, eVoteNoOpinion, eVoteNoOpinion),
3830fdc8d8SChris Lattner     m_step_from_insn (LLDB_INVALID_ADDRESS),
3930fdc8d8SChris Lattner     m_return_bp_id (LLDB_INVALID_BREAK_ID),
40c982c768SGreg Clayton     m_return_addr (LLDB_INVALID_ADDRESS),
4130fdc8d8SChris Lattner     m_stepped_out (false),
4230fdc8d8SChris Lattner     m_should_stop (false),
4330fdc8d8SChris Lattner     m_ran_analyze (false),
44c982c768SGreg Clayton     m_explains_stop (false),
45c982c768SGreg Clayton     m_until_points (),
4630fdc8d8SChris Lattner     m_stop_others (stop_others)
4730fdc8d8SChris Lattner {
4830fdc8d8SChris Lattner     // Stash away our "until" addresses:
491ac04c30SGreg Clayton     TargetSP target_sp (m_thread.CalculateTarget());
5030fdc8d8SChris Lattner 
51b57e4a1bSJason Molenda     StackFrameSP frame_sp (m_thread.GetStackFrameAtIndex (frame_idx));
52481cef25SGreg Clayton     if (frame_sp)
53481cef25SGreg Clayton     {
54481cef25SGreg Clayton         m_step_from_insn = frame_sp->GetStackID().GetPC();
5530fdc8d8SChris Lattner         lldb::user_id_t thread_id = m_thread.GetID();
5630fdc8d8SChris Lattner 
5730fdc8d8SChris Lattner         // Find the return address and set a breakpoint there:
5830fdc8d8SChris Lattner         // FIXME - can we do this more securely if we know first_insn?
5930fdc8d8SChris Lattner 
60b57e4a1bSJason Molenda         StackFrameSP return_frame_sp (m_thread.GetStackFrameAtIndex(frame_idx + 1));
61481cef25SGreg Clayton         if (return_frame_sp)
62481cef25SGreg Clayton         {
639da7bd07SGreg Clayton             // TODO: add inline functionality
64481cef25SGreg Clayton             m_return_addr = return_frame_sp->GetStackID().GetPC();
65eb023e75SGreg Clayton             Breakpoint *return_bp = target_sp->CreateBreakpoint (m_return_addr, true, false).get();
66*e65b2cf2SEugene Zelenko             if (return_bp != nullptr)
6730fdc8d8SChris Lattner             {
6830fdc8d8SChris Lattner                 return_bp->SetThreadID(thread_id);
6930fdc8d8SChris Lattner                 m_return_bp_id = return_bp->GetID();
702995077dSJim Ingham                 return_bp->SetBreakpointKind ("until-return-backstop");
7130fdc8d8SChris Lattner             }
7230fdc8d8SChris Lattner         }
7330fdc8d8SChris Lattner 
7476447851SJim Ingham         m_stack_id = frame_sp->GetStackID();
7530fdc8d8SChris Lattner 
7630fdc8d8SChris Lattner         // Now set breakpoints on all our return addresses:
77a297a97eSAndy Gibbs         for (size_t i = 0; i < num_addresses; i++)
7830fdc8d8SChris Lattner         {
79eb023e75SGreg Clayton             Breakpoint *until_bp = target_sp->CreateBreakpoint (address_list[i], true, false).get();
80*e65b2cf2SEugene Zelenko             if (until_bp != nullptr)
8130fdc8d8SChris Lattner             {
8230fdc8d8SChris Lattner                 until_bp->SetThreadID(thread_id);
8330fdc8d8SChris Lattner                 m_until_points[address_list[i]] = until_bp->GetID();
842995077dSJim Ingham                 until_bp->SetBreakpointKind("until-target");
8530fdc8d8SChris Lattner             }
8630fdc8d8SChris Lattner             else
8730fdc8d8SChris Lattner             {
8830fdc8d8SChris Lattner                 m_until_points[address_list[i]] = LLDB_INVALID_BREAK_ID;
8930fdc8d8SChris Lattner             }
9030fdc8d8SChris Lattner         }
9130fdc8d8SChris Lattner     }
92481cef25SGreg Clayton }
9330fdc8d8SChris Lattner 
9430fdc8d8SChris Lattner ThreadPlanStepUntil::~ThreadPlanStepUntil ()
9530fdc8d8SChris Lattner {
9630fdc8d8SChris Lattner     Clear();
9730fdc8d8SChris Lattner }
9830fdc8d8SChris Lattner 
9930fdc8d8SChris Lattner void
10030fdc8d8SChris Lattner ThreadPlanStepUntil::Clear()
10130fdc8d8SChris Lattner {
1021ac04c30SGreg Clayton     TargetSP target_sp (m_thread.CalculateTarget());
1031ac04c30SGreg Clayton     if (target_sp)
1041ac04c30SGreg Clayton     {
10530fdc8d8SChris Lattner         if (m_return_bp_id != LLDB_INVALID_BREAK_ID)
10630fdc8d8SChris Lattner         {
1071ac04c30SGreg Clayton             target_sp->RemoveBreakpointByID(m_return_bp_id);
10830fdc8d8SChris Lattner             m_return_bp_id = LLDB_INVALID_BREAK_ID;
10930fdc8d8SChris Lattner         }
11030fdc8d8SChris Lattner 
11130fdc8d8SChris Lattner         until_collection::iterator pos, end = m_until_points.end();
11230fdc8d8SChris Lattner         for (pos = m_until_points.begin(); pos != end; pos++)
11330fdc8d8SChris Lattner         {
1141ac04c30SGreg Clayton             target_sp->RemoveBreakpointByID((*pos).second);
1151ac04c30SGreg Clayton         }
11630fdc8d8SChris Lattner     }
11730fdc8d8SChris Lattner     m_until_points.clear();
11830fdc8d8SChris Lattner }
11930fdc8d8SChris Lattner 
12030fdc8d8SChris Lattner void
12130fdc8d8SChris Lattner ThreadPlanStepUntil::GetDescription (Stream *s, lldb::DescriptionLevel level)
12230fdc8d8SChris Lattner {
12330fdc8d8SChris Lattner     if (level == lldb::eDescriptionLevelBrief)
12430fdc8d8SChris Lattner     {
12530fdc8d8SChris Lattner         s->Printf ("step until");
12630fdc8d8SChris Lattner         if (m_stepped_out)
12730fdc8d8SChris Lattner             s->Printf (" - stepped out");
12830fdc8d8SChris Lattner     }
12930fdc8d8SChris Lattner     else
13030fdc8d8SChris Lattner     {
13130fdc8d8SChris Lattner         if (m_until_points.size() == 1)
132d01b2953SDaniel Malea             s->Printf ("Stepping from address 0x%" PRIx64 " until we reach 0x%" PRIx64 " using breakpoint %d",
13330fdc8d8SChris Lattner                        (uint64_t)m_step_from_insn,
13430fdc8d8SChris Lattner                        (uint64_t) (*m_until_points.begin()).first,
13530fdc8d8SChris Lattner                        (*m_until_points.begin()).second);
13630fdc8d8SChris Lattner         else
13730fdc8d8SChris Lattner         {
13830fdc8d8SChris Lattner             until_collection::iterator pos, end = m_until_points.end();
139d01b2953SDaniel Malea             s->Printf ("Stepping from address 0x%" PRIx64 " until we reach one of:",
14030fdc8d8SChris Lattner                        (uint64_t)m_step_from_insn);
14130fdc8d8SChris Lattner             for (pos = m_until_points.begin(); pos != end; pos++)
14230fdc8d8SChris Lattner             {
143d01b2953SDaniel Malea                 s->Printf ("\n\t0x%" PRIx64 " (bp: %d)", (uint64_t) (*pos).first, (*pos).second);
14430fdc8d8SChris Lattner             }
14530fdc8d8SChris Lattner         }
146d01b2953SDaniel Malea         s->Printf(" stepped out address is 0x%" PRIx64 ".", (uint64_t) m_return_addr);
14730fdc8d8SChris Lattner     }
14830fdc8d8SChris Lattner }
14930fdc8d8SChris Lattner 
15030fdc8d8SChris Lattner bool
15130fdc8d8SChris Lattner ThreadPlanStepUntil::ValidatePlan (Stream *error)
15230fdc8d8SChris Lattner {
15330fdc8d8SChris Lattner     if (m_return_bp_id == LLDB_INVALID_BREAK_ID)
15430fdc8d8SChris Lattner         return false;
15530fdc8d8SChris Lattner     else
15630fdc8d8SChris Lattner     {
15730fdc8d8SChris Lattner         until_collection::iterator pos, end = m_until_points.end();
15830fdc8d8SChris Lattner         for (pos = m_until_points.begin(); pos != end; pos++)
15930fdc8d8SChris Lattner         {
16030fdc8d8SChris Lattner             if (!LLDB_BREAK_ID_IS_VALID ((*pos).second))
16130fdc8d8SChris Lattner                 return false;
16230fdc8d8SChris Lattner         }
16330fdc8d8SChris Lattner         return true;
16430fdc8d8SChris Lattner     }
16530fdc8d8SChris Lattner }
16630fdc8d8SChris Lattner 
16730fdc8d8SChris Lattner void
16830fdc8d8SChris Lattner ThreadPlanStepUntil::AnalyzeStop()
16930fdc8d8SChris Lattner {
17030fdc8d8SChris Lattner     if (m_ran_analyze)
17130fdc8d8SChris Lattner         return;
17230fdc8d8SChris Lattner 
17360c4118cSJim Ingham     StopInfoSP stop_info_sp = GetPrivateStopInfo ();
17430fdc8d8SChris Lattner     m_should_stop = true;
17530fdc8d8SChris Lattner     m_explains_stop = false;
17630fdc8d8SChris Lattner 
177b15bfc75SJim Ingham     if (stop_info_sp)
17830fdc8d8SChris Lattner     {
179b15bfc75SJim Ingham         StopReason reason = stop_info_sp->GetStopReason();
18030fdc8d8SChris Lattner 
1819b03fa0cSJim Ingham         if (reason == eStopReasonBreakpoint)
18230fdc8d8SChris Lattner         {
18330fdc8d8SChris Lattner             // If this is OUR breakpoint, we're fine, otherwise we don't know why this happened...
1841ac04c30SGreg Clayton             BreakpointSiteSP this_site = m_thread.GetProcess()->GetBreakpointSiteList().FindByID (stop_info_sp->GetValue());
18530fdc8d8SChris Lattner             if (!this_site)
18630fdc8d8SChris Lattner             {
18730fdc8d8SChris Lattner                 m_explains_stop = false;
18830fdc8d8SChris Lattner                 return;
18930fdc8d8SChris Lattner             }
19030fdc8d8SChris Lattner 
19130fdc8d8SChris Lattner             if (this_site->IsBreakpointAtThisSite (m_return_bp_id))
19230fdc8d8SChris Lattner             {
19330fdc8d8SChris Lattner                 // If we are at our "step out" breakpoint, and the stack depth has shrunk, then
19430fdc8d8SChris Lattner                 // this is indeed our stop.
19530fdc8d8SChris Lattner                 // If the stack depth has grown, then we've hit our step out breakpoint recursively.
19630fdc8d8SChris Lattner                 // If we are the only breakpoint at that location, then we do explain the stop, and
19730fdc8d8SChris Lattner                 // we'll just continue.
19830fdc8d8SChris Lattner                 // If there was another breakpoint here, then we don't explain the stop, but we won't
19930fdc8d8SChris Lattner                 // mark ourselves Completed, because maybe that breakpoint will continue, and then
20030fdc8d8SChris Lattner                 // we'll finish the "until".
201b5c0d1ccSJim Ingham                 bool done;
202b5c0d1ccSJim Ingham                 StackID cur_frame_zero_id;
203b5c0d1ccSJim Ingham 
204*e65b2cf2SEugene Zelenko                 done = (m_stack_id < cur_frame_zero_id);
205b5c0d1ccSJim Ingham 
206b5c0d1ccSJim Ingham                 if (done)
20730fdc8d8SChris Lattner                 {
20830fdc8d8SChris Lattner                     m_stepped_out = true;
20930fdc8d8SChris Lattner                     SetPlanComplete();
21030fdc8d8SChris Lattner                 }
21130fdc8d8SChris Lattner                 else
21230fdc8d8SChris Lattner                     m_should_stop = false;
21330fdc8d8SChris Lattner 
21430fdc8d8SChris Lattner                 if (this_site->GetNumberOfOwners() == 1)
21530fdc8d8SChris Lattner                     m_explains_stop = true;
21630fdc8d8SChris Lattner                 else
21730fdc8d8SChris Lattner                     m_explains_stop = false;
21830fdc8d8SChris Lattner                 return;
21930fdc8d8SChris Lattner             }
22030fdc8d8SChris Lattner             else
22130fdc8d8SChris Lattner             {
22230fdc8d8SChris Lattner                 // Check if we've hit one of our "until" breakpoints.
22330fdc8d8SChris Lattner                 until_collection::iterator pos, end = m_until_points.end();
22430fdc8d8SChris Lattner                 for (pos = m_until_points.begin(); pos != end; pos++)
22530fdc8d8SChris Lattner                 {
22630fdc8d8SChris Lattner                     if (this_site->IsBreakpointAtThisSite ((*pos).second))
22730fdc8d8SChris Lattner                     {
22830fdc8d8SChris Lattner                         // If we're at the right stack depth, then we're done.
229b5c0d1ccSJim Ingham 
230b5c0d1ccSJim Ingham                         bool done;
231b5c0d1ccSJim Ingham                         StackID frame_zero_id = m_thread.GetStackFrameAtIndex(0)->GetStackID();
232b5c0d1ccSJim Ingham 
233b5c0d1ccSJim Ingham                         if (frame_zero_id == m_stack_id)
234b5c0d1ccSJim Ingham                             done = true;
235b5c0d1ccSJim Ingham                         else if (frame_zero_id < m_stack_id)
236b5c0d1ccSJim Ingham                             done = false;
237b5c0d1ccSJim Ingham                         else
238b5c0d1ccSJim Ingham                         {
239b57e4a1bSJason Molenda                             StackFrameSP older_frame_sp = m_thread.GetStackFrameAtIndex(1);
240b5c0d1ccSJim Ingham 
241b5c0d1ccSJim Ingham                             // But if we can't even unwind one frame we should just get out of here & stop...
242b5c0d1ccSJim Ingham                             if (older_frame_sp)
243b5c0d1ccSJim Ingham                             {
244b5c0d1ccSJim Ingham                                 const SymbolContext &older_context
245b5c0d1ccSJim Ingham                                     = older_frame_sp->GetSymbolContext(eSymbolContextEverything);
246b5c0d1ccSJim Ingham                                 SymbolContext stack_context;
247b5c0d1ccSJim Ingham                                 m_stack_id.GetSymbolContextScope()->CalculateSymbolContext(&stack_context);
248b5c0d1ccSJim Ingham 
249*e65b2cf2SEugene Zelenko                                 done = (older_context == stack_context);
250b5c0d1ccSJim Ingham                             }
251b5c0d1ccSJim Ingham                             else
252b5c0d1ccSJim Ingham                                 done = false;
253b5c0d1ccSJim Ingham                         }
254b5c0d1ccSJim Ingham 
255b5c0d1ccSJim Ingham                         if (done)
25630fdc8d8SChris Lattner                             SetPlanComplete();
25730fdc8d8SChris Lattner                         else
25830fdc8d8SChris Lattner                             m_should_stop = false;
25930fdc8d8SChris Lattner 
26030fdc8d8SChris Lattner                         // Otherwise we've hit this breakpoint recursively.  If we're the
26130fdc8d8SChris Lattner                         // only breakpoint here, then we do explain the stop, and we'll continue.
26230fdc8d8SChris Lattner                         // If not then we should let higher plans handle this stop.
26330fdc8d8SChris Lattner                         if (this_site->GetNumberOfOwners() == 1)
26430fdc8d8SChris Lattner                             m_explains_stop = true;
26530fdc8d8SChris Lattner                         else
26630fdc8d8SChris Lattner                         {
26730fdc8d8SChris Lattner                             m_should_stop = true;
26830fdc8d8SChris Lattner                             m_explains_stop = false;
26930fdc8d8SChris Lattner                         }
27030fdc8d8SChris Lattner                         return;
27130fdc8d8SChris Lattner                     }
27230fdc8d8SChris Lattner                 }
27330fdc8d8SChris Lattner             }
27430fdc8d8SChris Lattner             // If we get here we haven't hit any of our breakpoints, so let the higher
27530fdc8d8SChris Lattner             // plans take care of the stop.
27630fdc8d8SChris Lattner             m_explains_stop = false;
27730fdc8d8SChris Lattner             return;
27830fdc8d8SChris Lattner         }
2799b03fa0cSJim Ingham         else if (IsUsuallyUnexplainedStopReason(reason))
2809b03fa0cSJim Ingham         {
28130fdc8d8SChris Lattner             m_explains_stop = false;
2829b03fa0cSJim Ingham         }
2839b03fa0cSJim Ingham         else
2849b03fa0cSJim Ingham         {
28530fdc8d8SChris Lattner             m_explains_stop = true;
28630fdc8d8SChris Lattner         }
28730fdc8d8SChris Lattner     }
28830fdc8d8SChris Lattner }
28930fdc8d8SChris Lattner 
29030fdc8d8SChris Lattner bool
291221d51cfSJim Ingham ThreadPlanStepUntil::DoPlanExplainsStop (Event *event_ptr)
29230fdc8d8SChris Lattner {
29330fdc8d8SChris Lattner     // We don't explain signals or breakpoints (breakpoints that handle stepping in or
29430fdc8d8SChris Lattner     // out will be handled by a child plan.
29530fdc8d8SChris Lattner     AnalyzeStop();
29630fdc8d8SChris Lattner     return m_explains_stop;
29730fdc8d8SChris Lattner }
29830fdc8d8SChris Lattner 
29930fdc8d8SChris Lattner bool
30030fdc8d8SChris Lattner ThreadPlanStepUntil::ShouldStop (Event *event_ptr)
30130fdc8d8SChris Lattner {
30230fdc8d8SChris Lattner     // If we've told our self in ExplainsStop that we plan to continue, then
30330fdc8d8SChris Lattner     // do so here.  Otherwise, as long as this thread has stopped for a reason,
30430fdc8d8SChris Lattner     // we will stop.
30530fdc8d8SChris Lattner 
30660c4118cSJim Ingham     StopInfoSP stop_info_sp = GetPrivateStopInfo ();
3079a028519SSean Callanan     if (!stop_info_sp || stop_info_sp->GetStopReason() == eStopReasonNone)
30830fdc8d8SChris Lattner         return false;
30930fdc8d8SChris Lattner 
31030fdc8d8SChris Lattner     AnalyzeStop();
31130fdc8d8SChris Lattner     return m_should_stop;
31230fdc8d8SChris Lattner }
31330fdc8d8SChris Lattner 
31430fdc8d8SChris Lattner bool
31530fdc8d8SChris Lattner ThreadPlanStepUntil::StopOthers ()
31630fdc8d8SChris Lattner {
31730fdc8d8SChris Lattner     return m_stop_others;
31830fdc8d8SChris Lattner }
31930fdc8d8SChris Lattner 
32030fdc8d8SChris Lattner StateType
32106e827ccSJim Ingham ThreadPlanStepUntil::GetPlanRunState ()
32230fdc8d8SChris Lattner {
32330fdc8d8SChris Lattner     return eStateRunning;
32430fdc8d8SChris Lattner }
32530fdc8d8SChris Lattner 
32630fdc8d8SChris Lattner bool
327221d51cfSJim Ingham ThreadPlanStepUntil::DoWillResume (StateType resume_state, bool current_plan)
32830fdc8d8SChris Lattner {
32930fdc8d8SChris Lattner     if (current_plan)
33030fdc8d8SChris Lattner     {
3311ac04c30SGreg Clayton         TargetSP target_sp (m_thread.CalculateTarget());
3321ac04c30SGreg Clayton         if (target_sp)
3331ac04c30SGreg Clayton         {
3341ac04c30SGreg Clayton             Breakpoint *return_bp = target_sp->GetBreakpointByID(m_return_bp_id).get();
335*e65b2cf2SEugene Zelenko             if (return_bp != nullptr)
33630fdc8d8SChris Lattner                 return_bp->SetEnabled (true);
33730fdc8d8SChris Lattner 
33830fdc8d8SChris Lattner             until_collection::iterator pos, end = m_until_points.end();
33930fdc8d8SChris Lattner             for (pos = m_until_points.begin(); pos != end; pos++)
34030fdc8d8SChris Lattner             {
3411ac04c30SGreg Clayton                 Breakpoint *until_bp = target_sp->GetBreakpointByID((*pos).second).get();
342*e65b2cf2SEugene Zelenko                 if (until_bp != nullptr)
34330fdc8d8SChris Lattner                     until_bp->SetEnabled (true);
34430fdc8d8SChris Lattner             }
34530fdc8d8SChris Lattner         }
3461ac04c30SGreg Clayton     }
34730fdc8d8SChris Lattner 
34830fdc8d8SChris Lattner     m_should_stop = true;
34930fdc8d8SChris Lattner     m_ran_analyze = false;
35030fdc8d8SChris Lattner     m_explains_stop = false;
35130fdc8d8SChris Lattner     return true;
35230fdc8d8SChris Lattner }
35330fdc8d8SChris Lattner 
35430fdc8d8SChris Lattner bool
35530fdc8d8SChris Lattner ThreadPlanStepUntil::WillStop ()
35630fdc8d8SChris Lattner {
3571ac04c30SGreg Clayton     TargetSP target_sp (m_thread.CalculateTarget());
3581ac04c30SGreg Clayton     if (target_sp)
3591ac04c30SGreg Clayton     {
3601ac04c30SGreg Clayton         Breakpoint *return_bp = target_sp->GetBreakpointByID(m_return_bp_id).get();
361*e65b2cf2SEugene Zelenko         if (return_bp != nullptr)
36230fdc8d8SChris Lattner             return_bp->SetEnabled (false);
36330fdc8d8SChris Lattner 
36430fdc8d8SChris Lattner         until_collection::iterator pos, end = m_until_points.end();
36530fdc8d8SChris Lattner         for (pos = m_until_points.begin(); pos != end; pos++)
36630fdc8d8SChris Lattner         {
3671ac04c30SGreg Clayton             Breakpoint *until_bp = target_sp->GetBreakpointByID((*pos).second).get();
368*e65b2cf2SEugene Zelenko             if (until_bp != nullptr)
36930fdc8d8SChris Lattner                 until_bp->SetEnabled (false);
37030fdc8d8SChris Lattner         }
3711ac04c30SGreg Clayton     }
37230fdc8d8SChris Lattner     return true;
37330fdc8d8SChris Lattner }
37430fdc8d8SChris Lattner 
37530fdc8d8SChris Lattner bool
37630fdc8d8SChris Lattner ThreadPlanStepUntil::MischiefManaged ()
37730fdc8d8SChris Lattner {
37830fdc8d8SChris Lattner     // I'm letting "PlanExplainsStop" do all the work, and just reporting that here.
37930fdc8d8SChris Lattner     bool done = false;
38030fdc8d8SChris Lattner     if (IsPlanComplete())
38130fdc8d8SChris Lattner     {
3825160ce5cSGreg Clayton         Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP));
38330fdc8d8SChris Lattner         if (log)
38430fdc8d8SChris Lattner             log->Printf("Completed step until plan.");
38530fdc8d8SChris Lattner 
38630fdc8d8SChris Lattner         Clear();
38730fdc8d8SChris Lattner         done = true;
38830fdc8d8SChris Lattner     }
38930fdc8d8SChris Lattner     if (done)
39030fdc8d8SChris Lattner         ThreadPlan::MischiefManaged ();
39130fdc8d8SChris Lattner 
39230fdc8d8SChris Lattner     return done;
39330fdc8d8SChris Lattner }
394