blob: 5314b5d9b5f56ac8339ae8bebdb34e1b6fecc8c3 [file] [log] [blame]
Chris Lattner30fdc8d2010-06-08 16:52:24 +00001//===-- ThreadPlanShouldStopHere.cpp ----------------------------*- C++ -*-===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
Jim Ingham25f66702011-12-03 01:52:59 +000010#include "lldb/Target/RegisterContext.h"
Chris Lattner30fdc8d2010-06-08 16:52:24 +000011#include "lldb/Target/Thread.h"
12#include "lldb/Target/ThreadPlanShouldStopHere.h"
Jim Ingham25f66702011-12-03 01:52:59 +000013#include "lldb/Core/Log.h"
Chris Lattner30fdc8d2010-06-08 16:52:24 +000014
15using namespace lldb;
16using namespace lldb_private;
17
18// C Includes
19// C++ Includes
20// Other libraries and framework includes
21// Project includes
22
23//----------------------------------------------------------------------
24// ThreadPlanShouldStopHere constructor
25//----------------------------------------------------------------------
Jim Ingham4b4b2472014-03-13 02:47:14 +000026ThreadPlanShouldStopHere::ThreadPlanShouldStopHere(ThreadPlan *owner) :
27 m_callbacks (),
28 m_baton (NULL),
Chris Lattner30fdc8d2010-06-08 16:52:24 +000029 m_owner (owner),
30 m_flags (ThreadPlanShouldStopHere::eNone)
31{
Jim Ingham4b4b2472014-03-13 02:47:14 +000032 m_callbacks.should_stop_here_callback = ThreadPlanShouldStopHere::DefaultShouldStopHereCallback;
33 m_callbacks.step_from_here_callback = ThreadPlanShouldStopHere::DefaultStepFromHereCallback;
34}
35
36ThreadPlanShouldStopHere::ThreadPlanShouldStopHere(ThreadPlan *owner, const ThreadPlanShouldStopHereCallbacks *callbacks, void *baton) :
37 m_callbacks (),
38 m_baton (),
39 m_owner (owner),
40 m_flags (ThreadPlanShouldStopHere::eNone)
41{
42 SetShouldStopHereCallbacks(callbacks, baton);
Chris Lattner30fdc8d2010-06-08 16:52:24 +000043}
44
45//----------------------------------------------------------------------
46// Destructor
47//----------------------------------------------------------------------
48ThreadPlanShouldStopHere::~ThreadPlanShouldStopHere()
49{
50}
51
Jim Ingham4b4b2472014-03-13 02:47:14 +000052bool
53ThreadPlanShouldStopHere::InvokeShouldStopHereCallback (FrameComparison operation)
Chris Lattner30fdc8d2010-06-08 16:52:24 +000054{
Jim Ingham4b4b2472014-03-13 02:47:14 +000055 bool should_stop_here = true;
56 if (m_callbacks.should_stop_here_callback)
Jim Ingham25f66702011-12-03 01:52:59 +000057 {
Jim Ingham4b4b2472014-03-13 02:47:14 +000058 should_stop_here = m_callbacks.should_stop_here_callback (m_owner, m_flags, operation, m_baton);
Greg Clayton5160ce52013-03-27 23:08:40 +000059 Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP));
Jim Ingham25f66702011-12-03 01:52:59 +000060 if (log)
61 {
62 lldb::addr_t current_addr = m_owner->GetThread().GetRegisterContext()->GetPC(0);
63
Jim Ingham4b4b2472014-03-13 02:47:14 +000064 log->Printf ("ShouldStopHere callback returned %u from 0x%" PRIx64 ".", should_stop_here, current_addr);
Jim Ingham25f66702011-12-03 01:52:59 +000065 }
Jim Ingham25f66702011-12-03 01:52:59 +000066 }
Jim Ingham4b4b2472014-03-13 02:47:14 +000067
68 return should_stop_here;
69}
70
71bool
72ThreadPlanShouldStopHere::DefaultShouldStopHereCallback (ThreadPlan *current_plan,
73 Flags &flags,
74 FrameComparison operation,
75 void *baton)
76{
77 bool should_stop_here = true;
78 StackFrame *frame = current_plan->GetThread().GetStackFrameAtIndex(0).get();
Jim Ingham0eed7382014-03-17 23:03:34 +000079 if (!frame)
80 return true;
81
Jim Ingham4b4b2472014-03-13 02:47:14 +000082 Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP));
83
84 if ((operation == eFrameCompareOlder && flags.Test(eStepOutAvoidNoDebug))
85 || (operation == eFrameCompareYounger && flags.Test(eStepInAvoidNoDebug)))
86 {
87 if (!frame->HasDebugInformation())
88 {
89 if (log)
90 log->Printf ("Stepping out of frame with no debug info");
91
92 should_stop_here = false;
93 }
94 }
95
Jim Ingham0eed7382014-03-17 23:03:34 +000096 // Always avoid code with line number 0.
97 // FIXME: At present the ShouldStop and the StepFromHere calculate this independently. If this ever
98 // becomes expensive (this one isn't) we can try to have this set a state that the StepFromHere can use.
99 if (frame)
100 {
101 SymbolContext sc;
102 sc = frame->GetSymbolContext (eSymbolContextLineEntry);
103 if (sc.line_entry.line == 0)
104 should_stop_here = false;
105 }
106
Jim Ingham4b4b2472014-03-13 02:47:14 +0000107 return should_stop_here;
108}
109
110ThreadPlanSP
111ThreadPlanShouldStopHere::DefaultStepFromHereCallback (ThreadPlan *current_plan,
112 Flags &flags,
113 FrameComparison operation,
114 void *baton)
115{
Jim Ingham0eed7382014-03-17 23:03:34 +0000116 const bool stop_others = false;
117 const size_t frame_index = 0;
118 ThreadPlanSP return_plan_sp;
119 // If we are stepping through code at line number 0, then we need to step over this range. Otherwise
120 // we will step out.
121 StackFrame *frame = current_plan->GetThread().GetStackFrameAtIndex(0).get();
122 if (!frame)
Jim Ingham4b4b2472014-03-13 02:47:14 +0000123 return return_plan_sp;
Jim Ingham0eed7382014-03-17 23:03:34 +0000124 SymbolContext sc;
125 sc = frame->GetSymbolContext (eSymbolContextLineEntry);
126 if (sc.line_entry.line == 0)
127 {
128 AddressRange range = sc.line_entry.range;
129 return_plan_sp = current_plan->GetThread().QueueThreadPlanForStepOverRange(false,
130 range,
131 sc,
132 eOnlyDuringStepping,
133 eLazyBoolNo);
134 }
135
136 if (!return_plan_sp)
137 return_plan_sp = current_plan->GetThread().QueueThreadPlanForStepOutNoShouldStop (false,
138 NULL,
139 true,
140 stop_others,
141 eVoteNo,
142 eVoteNoOpinion,
143 frame_index);
144 return return_plan_sp;
Jim Ingham4b4b2472014-03-13 02:47:14 +0000145}
146
147ThreadPlanSP
148ThreadPlanShouldStopHere::QueueStepOutFromHerePlan(lldb_private::Flags &flags, lldb::FrameComparison operation)
149{
150 ThreadPlanSP return_plan_sp;
151 if (m_callbacks.step_from_here_callback)
152 {
153 return_plan_sp = m_callbacks.step_from_here_callback (m_owner, flags, operation, m_baton);
154 }
155 return return_plan_sp;
156
157}
158
159lldb::ThreadPlanSP
160ThreadPlanShouldStopHere::CheckShouldStopHereAndQueueStepOut (lldb::FrameComparison operation)
161{
162 if (!InvokeShouldStopHereCallback(operation))
163 return QueueStepOutFromHerePlan(m_flags, operation);
Chris Lattner30fdc8d2010-06-08 16:52:24 +0000164 else
Jim Ingham4d56e9c2013-07-18 21:48:26 +0000165 return ThreadPlanSP();
Greg Clayton1a65ae12011-01-25 23:55:37 +0000166}
Jim Ingham4b4b2472014-03-13 02:47:14 +0000167