1696bd635SAlexander Shaposhnikov //===-- SBThreadPlan.cpp ----------------------------------------*- C++ -*-===//
22bdbfd50SJim Ingham //
3*2946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*2946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
5*2946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
62bdbfd50SJim Ingham //
72bdbfd50SJim Ingham //===----------------------------------------------------------------------===//
82bdbfd50SJim Ingham 
92bdbfd50SJim Ingham #include "lldb/API/SBThread.h"
102bdbfd50SJim Ingham 
112bdbfd50SJim Ingham #include "lldb/API/SBFileSpec.h"
122bdbfd50SJim Ingham #include "lldb/API/SBStream.h"
13b9c1b51eSKate Stone #include "lldb/API/SBSymbolContext.h"
142bdbfd50SJim Ingham #include "lldb/Breakpoint/BreakpointLocation.h"
152bdbfd50SJim Ingham #include "lldb/Core/Debugger.h"
162bdbfd50SJim Ingham #include "lldb/Core/StreamFile.h"
172bdbfd50SJim Ingham #include "lldb/Interpreter/CommandInterpreter.h"
18b9c1b51eSKate Stone #include "lldb/Symbol/CompileUnit.h"
19b9c1b51eSKate Stone #include "lldb/Symbol/SymbolContext.h"
202bdbfd50SJim Ingham #include "lldb/Target/Process.h"
212bdbfd50SJim Ingham #include "lldb/Target/Queue.h"
222bdbfd50SJim Ingham #include "lldb/Target/StopInfo.h"
23b9c1b51eSKate Stone #include "lldb/Target/SystemRuntime.h"
242bdbfd50SJim Ingham #include "lldb/Target/Target.h"
25b9c1b51eSKate Stone #include "lldb/Target/Thread.h"
26b9c1b51eSKate Stone #include "lldb/Target/ThreadPlan.h"
272bdbfd50SJim Ingham #include "lldb/Target/ThreadPlanPython.h"
28b9c1b51eSKate Stone #include "lldb/Target/ThreadPlanStepInRange.h"
292bdbfd50SJim Ingham #include "lldb/Target/ThreadPlanStepInstruction.h"
302bdbfd50SJim Ingham #include "lldb/Target/ThreadPlanStepOut.h"
312bdbfd50SJim Ingham #include "lldb/Target/ThreadPlanStepRange.h"
32d821c997SPavel Labath #include "lldb/Utility/State.h"
33bf9a7730SZachary Turner #include "lldb/Utility/Stream.h"
34f2a8bccfSPavel Labath #include "lldb/Utility/StructuredData.h"
352bdbfd50SJim Ingham 
362bdbfd50SJim Ingham #include "lldb/API/SBAddress.h"
372bdbfd50SJim Ingham #include "lldb/API/SBDebugger.h"
382bdbfd50SJim Ingham #include "lldb/API/SBEvent.h"
392bdbfd50SJim Ingham #include "lldb/API/SBFrame.h"
402bdbfd50SJim Ingham #include "lldb/API/SBProcess.h"
412bdbfd50SJim Ingham #include "lldb/API/SBThreadPlan.h"
422bdbfd50SJim Ingham #include "lldb/API/SBValue.h"
432bdbfd50SJim Ingham 
442bdbfd50SJim Ingham using namespace lldb;
452bdbfd50SJim Ingham using namespace lldb_private;
462bdbfd50SJim Ingham 
472bdbfd50SJim Ingham //----------------------------------------------------------------------
482bdbfd50SJim Ingham // Constructors
492bdbfd50SJim Ingham //----------------------------------------------------------------------
50b9c1b51eSKate Stone SBThreadPlan::SBThreadPlan() {}
512bdbfd50SJim Ingham 
52b9c1b51eSKate Stone SBThreadPlan::SBThreadPlan(const ThreadPlanSP &lldb_object_sp)
53b9c1b51eSKate Stone     : m_opaque_sp(lldb_object_sp) {}
542bdbfd50SJim Ingham 
55b9c1b51eSKate Stone SBThreadPlan::SBThreadPlan(const SBThreadPlan &rhs)
56b9c1b51eSKate Stone     : m_opaque_sp(rhs.m_opaque_sp) {}
572bdbfd50SJim Ingham 
58b9c1b51eSKate Stone SBThreadPlan::SBThreadPlan(lldb::SBThread &sb_thread, const char *class_name) {
592bdbfd50SJim Ingham   Thread *thread = sb_thread.get();
602bdbfd50SJim Ingham   if (thread)
612bdbfd50SJim Ingham     m_opaque_sp.reset(new ThreadPlanPython(*thread, class_name));
622bdbfd50SJim Ingham }
632bdbfd50SJim Ingham 
642bdbfd50SJim Ingham //----------------------------------------------------------------------
652bdbfd50SJim Ingham // Assignment operator
662bdbfd50SJim Ingham //----------------------------------------------------------------------
672bdbfd50SJim Ingham 
68b9c1b51eSKate Stone const lldb::SBThreadPlan &SBThreadPlan::operator=(const SBThreadPlan &rhs) {
692bdbfd50SJim Ingham   if (this != &rhs)
702bdbfd50SJim Ingham     m_opaque_sp = rhs.m_opaque_sp;
712bdbfd50SJim Ingham   return *this;
722bdbfd50SJim Ingham }
732bdbfd50SJim Ingham //----------------------------------------------------------------------
742bdbfd50SJim Ingham // Destructor
752bdbfd50SJim Ingham //----------------------------------------------------------------------
76b9c1b51eSKate Stone SBThreadPlan::~SBThreadPlan() {}
772bdbfd50SJim Ingham 
78b9c1b51eSKate Stone lldb_private::ThreadPlan *SBThreadPlan::get() { return m_opaque_sp.get(); }
792bdbfd50SJim Ingham 
80b9c1b51eSKate Stone bool SBThreadPlan::IsValid() const { return m_opaque_sp.get() != NULL; }
812bdbfd50SJim Ingham 
82b9c1b51eSKate Stone void SBThreadPlan::Clear() { m_opaque_sp.reset(); }
832bdbfd50SJim Ingham 
84b9c1b51eSKate Stone lldb::StopReason SBThreadPlan::GetStopReason() { return eStopReasonNone; }
852bdbfd50SJim Ingham 
86b9c1b51eSKate Stone size_t SBThreadPlan::GetStopReasonDataCount() { return 0; }
872bdbfd50SJim Ingham 
88b9c1b51eSKate Stone uint64_t SBThreadPlan::GetStopReasonDataAtIndex(uint32_t idx) { return 0; }
892bdbfd50SJim Ingham 
90b9c1b51eSKate Stone SBThread SBThreadPlan::GetThread() const {
91b9c1b51eSKate Stone   if (m_opaque_sp) {
922bdbfd50SJim Ingham     return SBThread(m_opaque_sp->GetThread().shared_from_this());
93b9c1b51eSKate Stone   } else
942bdbfd50SJim Ingham     return SBThread();
952bdbfd50SJim Ingham }
962bdbfd50SJim Ingham 
97b9c1b51eSKate Stone bool SBThreadPlan::GetDescription(lldb::SBStream &description) const {
98b9c1b51eSKate Stone   if (m_opaque_sp) {
992bdbfd50SJim Ingham     m_opaque_sp->GetDescription(description.get(), eDescriptionLevelFull);
100b9c1b51eSKate Stone   } else {
1012bdbfd50SJim Ingham     description.Printf("Empty SBThreadPlan");
1022bdbfd50SJim Ingham   }
1032bdbfd50SJim Ingham   return true;
1042bdbfd50SJim Ingham }
1052bdbfd50SJim Ingham 
106b9c1b51eSKate Stone void SBThreadPlan::SetThreadPlan(const ThreadPlanSP &lldb_object_sp) {
1072bdbfd50SJim Ingham   m_opaque_sp = lldb_object_sp;
1082bdbfd50SJim Ingham }
1092bdbfd50SJim Ingham 
110b9c1b51eSKate Stone void SBThreadPlan::SetPlanComplete(bool success) {
1112bdbfd50SJim Ingham   if (m_opaque_sp)
1122bdbfd50SJim Ingham     m_opaque_sp->SetPlanComplete(success);
1132bdbfd50SJim Ingham }
1142bdbfd50SJim Ingham 
115b9c1b51eSKate Stone bool SBThreadPlan::IsPlanComplete() {
1162bdbfd50SJim Ingham   if (m_opaque_sp)
1172bdbfd50SJim Ingham     return m_opaque_sp->IsPlanComplete();
1182bdbfd50SJim Ingham   else
1192bdbfd50SJim Ingham     return true;
1202bdbfd50SJim Ingham }
1212bdbfd50SJim Ingham 
122b9c1b51eSKate Stone bool SBThreadPlan::IsPlanStale() {
123c915a7d2SJim Ingham   if (m_opaque_sp)
124c915a7d2SJim Ingham     return m_opaque_sp->IsPlanStale();
125c915a7d2SJim Ingham   else
126c915a7d2SJim Ingham     return true;
127c915a7d2SJim Ingham }
128c915a7d2SJim Ingham 
129b9c1b51eSKate Stone bool SBThreadPlan::IsValid() {
1302bdbfd50SJim Ingham   if (m_opaque_sp)
1312bdbfd50SJim Ingham     return m_opaque_sp->ValidatePlan(nullptr);
1322bdbfd50SJim Ingham   else
1332bdbfd50SJim Ingham     return false;
1342bdbfd50SJim Ingham }
1352bdbfd50SJim Ingham 
136b9c1b51eSKate Stone // This section allows an SBThreadPlan to push another of the common types of
137b9c1b51eSKate Stone // plans...
1382bdbfd50SJim Ingham //
139b9c1b51eSKate Stone // FIXME, you should only be able to queue thread plans from inside the methods
14005097246SAdrian Prantl // of a Scripted Thread Plan.  Need a way to enforce that.
1412bdbfd50SJim Ingham 
1422bdbfd50SJim Ingham SBThreadPlan
1432bdbfd50SJim Ingham SBThreadPlan::QueueThreadPlanForStepOverRange(SBAddress &sb_start_address,
144b9c1b51eSKate Stone                                               lldb::addr_t size) {
145e103ae92SJonas Devlieghere   SBError error;
146e103ae92SJonas Devlieghere   return QueueThreadPlanForStepOverRange(sb_start_address, size, error);
147e103ae92SJonas Devlieghere }
148e103ae92SJonas Devlieghere 
149e103ae92SJonas Devlieghere SBThreadPlan SBThreadPlan::QueueThreadPlanForStepOverRange(
150e103ae92SJonas Devlieghere     SBAddress &sb_start_address, lldb::addr_t size, SBError &error) {
151b9c1b51eSKate Stone   if (m_opaque_sp) {
1522bdbfd50SJim Ingham     Address *start_address = sb_start_address.get();
153b9c1b51eSKate Stone     if (!start_address) {
1542bdbfd50SJim Ingham       return SBThreadPlan();
1552bdbfd50SJim Ingham     }
1562bdbfd50SJim Ingham 
1572bdbfd50SJim Ingham     AddressRange range(*start_address, size);
1582bdbfd50SJim Ingham     SymbolContext sc;
1592bdbfd50SJim Ingham     start_address->CalculateSymbolContext(&sc);
160e103ae92SJonas Devlieghere     Status plan_status;
161e103ae92SJonas Devlieghere 
162e103ae92SJonas Devlieghere     SBThreadPlan plan =
163e103ae92SJonas Devlieghere         SBThreadPlan(m_opaque_sp->GetThread().QueueThreadPlanForStepOverRange(
164e103ae92SJonas Devlieghere             false, range, sc, eAllThreads, plan_status));
165e103ae92SJonas Devlieghere 
166e103ae92SJonas Devlieghere     if (plan_status.Fail())
167e103ae92SJonas Devlieghere       error.SetErrorString(plan_status.AsCString());
168e103ae92SJonas Devlieghere 
169e103ae92SJonas Devlieghere     return plan;
170b9c1b51eSKate Stone   } else {
1712bdbfd50SJim Ingham     return SBThreadPlan();
1722bdbfd50SJim Ingham   }
1732bdbfd50SJim Ingham }
1742bdbfd50SJim Ingham 
1752bdbfd50SJim Ingham SBThreadPlan
1762bdbfd50SJim Ingham SBThreadPlan::QueueThreadPlanForStepInRange(SBAddress &sb_start_address,
177b9c1b51eSKate Stone                                             lldb::addr_t size) {
178e103ae92SJonas Devlieghere   SBError error;
179e103ae92SJonas Devlieghere   return QueueThreadPlanForStepInRange(sb_start_address, size, error);
180e103ae92SJonas Devlieghere }
181e103ae92SJonas Devlieghere 
182e103ae92SJonas Devlieghere SBThreadPlan
183e103ae92SJonas Devlieghere SBThreadPlan::QueueThreadPlanForStepInRange(SBAddress &sb_start_address,
184e103ae92SJonas Devlieghere                                             lldb::addr_t size, SBError &error) {
185b9c1b51eSKate Stone   if (m_opaque_sp) {
1862bdbfd50SJim Ingham     Address *start_address = sb_start_address.get();
187b9c1b51eSKate Stone     if (!start_address) {
1882bdbfd50SJim Ingham       return SBThreadPlan();
1892bdbfd50SJim Ingham     }
1902bdbfd50SJim Ingham 
1912bdbfd50SJim Ingham     AddressRange range(*start_address, size);
1922bdbfd50SJim Ingham     SymbolContext sc;
1932bdbfd50SJim Ingham     start_address->CalculateSymbolContext(&sc);
194e103ae92SJonas Devlieghere 
195e103ae92SJonas Devlieghere     Status plan_status;
196e103ae92SJonas Devlieghere     SBThreadPlan plan =
197e103ae92SJonas Devlieghere         SBThreadPlan(m_opaque_sp->GetThread().QueueThreadPlanForStepInRange(
198e103ae92SJonas Devlieghere             false, range, sc, NULL, eAllThreads, plan_status));
199e103ae92SJonas Devlieghere 
200e103ae92SJonas Devlieghere     if (plan_status.Fail())
201e103ae92SJonas Devlieghere       error.SetErrorString(plan_status.AsCString());
202e103ae92SJonas Devlieghere 
203e103ae92SJonas Devlieghere     return plan;
204b9c1b51eSKate Stone   } else {
2052bdbfd50SJim Ingham     return SBThreadPlan();
2062bdbfd50SJim Ingham   }
2072bdbfd50SJim Ingham }
2082bdbfd50SJim Ingham 
2092bdbfd50SJim Ingham SBThreadPlan
210b9c1b51eSKate Stone SBThreadPlan::QueueThreadPlanForStepOut(uint32_t frame_idx_to_step_to,
211b9c1b51eSKate Stone                                         bool first_insn) {
212e103ae92SJonas Devlieghere   SBError error;
213e103ae92SJonas Devlieghere   return QueueThreadPlanForStepOut(frame_idx_to_step_to, first_insn, error);
214e103ae92SJonas Devlieghere }
215e103ae92SJonas Devlieghere 
216e103ae92SJonas Devlieghere SBThreadPlan
217e103ae92SJonas Devlieghere SBThreadPlan::QueueThreadPlanForStepOut(uint32_t frame_idx_to_step_to,
218e103ae92SJonas Devlieghere                                         bool first_insn, SBError &error) {
219b9c1b51eSKate Stone   if (m_opaque_sp) {
2202bdbfd50SJim Ingham     SymbolContext sc;
221b9c1b51eSKate Stone     sc = m_opaque_sp->GetThread().GetStackFrameAtIndex(0)->GetSymbolContext(
222b9c1b51eSKate Stone         lldb::eSymbolContextEverything);
223e103ae92SJonas Devlieghere 
224e103ae92SJonas Devlieghere     Status plan_status;
225e103ae92SJonas Devlieghere     SBThreadPlan plan =
226e103ae92SJonas Devlieghere         SBThreadPlan(m_opaque_sp->GetThread().QueueThreadPlanForStepOut(
227b9c1b51eSKate Stone             false, &sc, first_insn, false, eVoteYes, eVoteNoOpinion,
228e103ae92SJonas Devlieghere             frame_idx_to_step_to, plan_status));
229e103ae92SJonas Devlieghere 
230e103ae92SJonas Devlieghere     if (plan_status.Fail())
231e103ae92SJonas Devlieghere       error.SetErrorString(plan_status.AsCString());
232e103ae92SJonas Devlieghere 
233e103ae92SJonas Devlieghere     return plan;
234b9c1b51eSKate Stone   } else {
2352bdbfd50SJim Ingham     return SBThreadPlan();
2362bdbfd50SJim Ingham   }
2372bdbfd50SJim Ingham }
2382bdbfd50SJim Ingham 
2392bdbfd50SJim Ingham SBThreadPlan
240b9c1b51eSKate Stone SBThreadPlan::QueueThreadPlanForRunToAddress(SBAddress sb_address) {
241e103ae92SJonas Devlieghere   SBError error;
242e103ae92SJonas Devlieghere   return QueueThreadPlanForRunToAddress(sb_address, error);
243e103ae92SJonas Devlieghere }
244e103ae92SJonas Devlieghere 
245e103ae92SJonas Devlieghere SBThreadPlan SBThreadPlan::QueueThreadPlanForRunToAddress(SBAddress sb_address,
246e103ae92SJonas Devlieghere                                                           SBError &error) {
247b9c1b51eSKate Stone   if (m_opaque_sp) {
2482bdbfd50SJim Ingham     Address *address = sb_address.get();
2492bdbfd50SJim Ingham     if (!address)
2502bdbfd50SJim Ingham       return SBThreadPlan();
2512bdbfd50SJim Ingham 
252e103ae92SJonas Devlieghere     Status plan_status;
253e103ae92SJonas Devlieghere     SBThreadPlan plan =
254e103ae92SJonas Devlieghere         SBThreadPlan(m_opaque_sp->GetThread().QueueThreadPlanForRunToAddress(
255e103ae92SJonas Devlieghere             false, *address, false, plan_status));
256e103ae92SJonas Devlieghere 
257e103ae92SJonas Devlieghere     if (plan_status.Fail())
258e103ae92SJonas Devlieghere       error.SetErrorString(plan_status.AsCString());
259e103ae92SJonas Devlieghere 
260e103ae92SJonas Devlieghere     return plan;
261b9c1b51eSKate Stone   } else {
2622bdbfd50SJim Ingham     return SBThreadPlan();
2632bdbfd50SJim Ingham   }
2642bdbfd50SJim Ingham }
265c1c0fac7SAleksandr Urakov 
266c1c0fac7SAleksandr Urakov SBThreadPlan
267c1c0fac7SAleksandr Urakov SBThreadPlan::QueueThreadPlanForStepScripted(const char *script_class_name) {
268e103ae92SJonas Devlieghere   SBError error;
269e103ae92SJonas Devlieghere   return QueueThreadPlanForStepScripted(script_class_name, error);
270e103ae92SJonas Devlieghere }
271e103ae92SJonas Devlieghere 
272e103ae92SJonas Devlieghere SBThreadPlan
273e103ae92SJonas Devlieghere SBThreadPlan::QueueThreadPlanForStepScripted(const char *script_class_name,
274e103ae92SJonas Devlieghere                                              SBError &error) {
275c1c0fac7SAleksandr Urakov   if (m_opaque_sp) {
276e103ae92SJonas Devlieghere     Status plan_status;
277e103ae92SJonas Devlieghere     SBThreadPlan plan =
278e103ae92SJonas Devlieghere         SBThreadPlan(m_opaque_sp->GetThread().QueueThreadPlanForStepScripted(
279e103ae92SJonas Devlieghere             false, script_class_name, false, plan_status));
280e103ae92SJonas Devlieghere 
281e103ae92SJonas Devlieghere     if (plan_status.Fail())
282e103ae92SJonas Devlieghere       error.SetErrorString(plan_status.AsCString());
283e103ae92SJonas Devlieghere 
284e103ae92SJonas Devlieghere     return plan;
285c1c0fac7SAleksandr Urakov   } else {
286c1c0fac7SAleksandr Urakov     return SBThreadPlan();
287c1c0fac7SAleksandr Urakov   }
288c1c0fac7SAleksandr Urakov }
289