130fdc8d8SChris Lattner //===-- ThreadPlan.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 "lldb/Target/ThreadPlan.h"
1130fdc8d8SChris Lattner 
1230fdc8d8SChris Lattner // C Includes
1330fdc8d8SChris Lattner // C++ Includes
1430fdc8d8SChris Lattner // Other libraries and framework includes
1530fdc8d8SChris Lattner // Project includes
1606e827ccSJim Ingham #include "lldb/Core/Debugger.h"
1730fdc8d8SChris Lattner #include "lldb/Core/Log.h"
1830fdc8d8SChris Lattner #include "lldb/Core/State.h"
192cad65a5SGreg Clayton #include "lldb/Target/RegisterContext.h"
202cad65a5SGreg Clayton #include "lldb/Target/Thread.h"
2106e827ccSJim Ingham #include "lldb/Target/Process.h"
2206e827ccSJim Ingham #include "lldb/Target/Target.h"
2330fdc8d8SChris Lattner 
2430fdc8d8SChris Lattner using namespace lldb;
2530fdc8d8SChris Lattner using namespace lldb_private;
2630fdc8d8SChris Lattner 
2730fdc8d8SChris Lattner //----------------------------------------------------------------------
2830fdc8d8SChris Lattner // ThreadPlan constructor
2930fdc8d8SChris Lattner //----------------------------------------------------------------------
30b01e742aSJim Ingham ThreadPlan::ThreadPlan(ThreadPlanKind kind, const char *name, Thread &thread, Vote stop_vote, Vote run_vote) :
3130fdc8d8SChris Lattner     m_thread (thread),
3230fdc8d8SChris Lattner     m_stop_vote (stop_vote),
3330fdc8d8SChris Lattner     m_run_vote (run_vote),
341ee0d4f7SBenjamin Kramer     m_kind (kind),
351ee0d4f7SBenjamin Kramer     m_name (name),
361ee0d4f7SBenjamin Kramer     m_plan_complete_mutex (Mutex::eMutexTypeRecursive),
371ee0d4f7SBenjamin Kramer     m_plan_complete (false),
381ee0d4f7SBenjamin Kramer     m_plan_private (false),
3964e7ead1SJim Ingham     m_okay_to_discard (true),
40fbbfe6ecSJim Ingham     m_is_master_plan (false),
41fbbfe6ecSJim Ingham     m_plan_succeeded(true)
4230fdc8d8SChris Lattner {
4330fdc8d8SChris Lattner     SetID (GetNextID());
4430fdc8d8SChris Lattner }
4530fdc8d8SChris Lattner 
4630fdc8d8SChris Lattner //----------------------------------------------------------------------
4730fdc8d8SChris Lattner // Destructor
4830fdc8d8SChris Lattner //----------------------------------------------------------------------
4930fdc8d8SChris Lattner ThreadPlan::~ThreadPlan()
5030fdc8d8SChris Lattner {
5130fdc8d8SChris Lattner }
5230fdc8d8SChris Lattner 
5330fdc8d8SChris Lattner bool
5430fdc8d8SChris Lattner ThreadPlan::IsPlanComplete ()
5530fdc8d8SChris Lattner {
56b132097bSGreg Clayton     Mutex::Locker locker(m_plan_complete_mutex);
5730fdc8d8SChris Lattner     return m_plan_complete;
5830fdc8d8SChris Lattner }
5930fdc8d8SChris Lattner 
6030fdc8d8SChris Lattner void
61fbbfe6ecSJim Ingham ThreadPlan::SetPlanComplete (bool success)
6230fdc8d8SChris Lattner {
63b132097bSGreg Clayton     Mutex::Locker locker(m_plan_complete_mutex);
6430fdc8d8SChris Lattner     m_plan_complete = true;
65fbbfe6ecSJim Ingham     m_plan_succeeded = success;
6630fdc8d8SChris Lattner }
6730fdc8d8SChris Lattner 
6830fdc8d8SChris Lattner bool
6930fdc8d8SChris Lattner ThreadPlan::MischiefManaged ()
7030fdc8d8SChris Lattner {
71b132097bSGreg Clayton     Mutex::Locker locker(m_plan_complete_mutex);
72*18de2fdcSJim Ingham     // Mark the plan is complete, but don't override the success flag.
7330fdc8d8SChris Lattner     m_plan_complete = true;
7430fdc8d8SChris Lattner     return true;
7530fdc8d8SChris Lattner }
7630fdc8d8SChris Lattner 
7730fdc8d8SChris Lattner Vote
7830fdc8d8SChris Lattner ThreadPlan::ShouldReportStop (Event *event_ptr)
7930fdc8d8SChris Lattner {
802d4edfbcSGreg Clayton     LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP));
812cad65a5SGreg Clayton 
8230fdc8d8SChris Lattner     if (m_stop_vote == eVoteNoOpinion)
8330fdc8d8SChris Lattner     {
8430fdc8d8SChris Lattner         ThreadPlan *prev_plan = GetPreviousPlan ();
8530fdc8d8SChris Lattner         if (prev_plan)
862cad65a5SGreg Clayton         {
872cad65a5SGreg Clayton             Vote prev_vote = prev_plan->ShouldReportStop (event_ptr);
882cad65a5SGreg Clayton             if (log)
89b5c0d1ccSJim Ingham                 log->Printf ("ThreadPlan::ShouldReportStop() returning previous thread plan vote: %s",
90b5c0d1ccSJim Ingham                              GetVoteAsCString (prev_vote));
912cad65a5SGreg Clayton             return prev_vote;
9230fdc8d8SChris Lattner         }
932cad65a5SGreg Clayton     }
942cad65a5SGreg Clayton     if (log)
951346f7e0SGreg Clayton         log->Printf ("ThreadPlan::ShouldReportStop() returning vote: %s", GetVoteAsCString (m_stop_vote));
9630fdc8d8SChris Lattner     return m_stop_vote;
9730fdc8d8SChris Lattner }
9830fdc8d8SChris Lattner 
9930fdc8d8SChris Lattner Vote
10030fdc8d8SChris Lattner ThreadPlan::ShouldReportRun (Event *event_ptr)
10130fdc8d8SChris Lattner {
10230fdc8d8SChris Lattner     if (m_run_vote == eVoteNoOpinion)
10330fdc8d8SChris Lattner     {
10430fdc8d8SChris Lattner         ThreadPlan *prev_plan = GetPreviousPlan ();
10530fdc8d8SChris Lattner         if (prev_plan)
10630fdc8d8SChris Lattner             return prev_plan->ShouldReportRun (event_ptr);
10730fdc8d8SChris Lattner     }
10830fdc8d8SChris Lattner     return m_run_vote;
10930fdc8d8SChris Lattner }
11030fdc8d8SChris Lattner 
11130fdc8d8SChris Lattner bool
11230fdc8d8SChris Lattner ThreadPlan::StopOthers ()
11330fdc8d8SChris Lattner {
11430fdc8d8SChris Lattner     ThreadPlan *prev_plan;
11530fdc8d8SChris Lattner     prev_plan = GetPreviousPlan ();
11630fdc8d8SChris Lattner     if (prev_plan == NULL)
11730fdc8d8SChris Lattner         return false;
11830fdc8d8SChris Lattner     else
11930fdc8d8SChris Lattner         return prev_plan->StopOthers();
12030fdc8d8SChris Lattner }
12130fdc8d8SChris Lattner 
122f48169bbSJim Ingham void
123f48169bbSJim Ingham ThreadPlan::SetStopOthers (bool new_value)
124f48169bbSJim Ingham {
125f48169bbSJim Ingham 	// SetStopOthers doesn't work up the hierarchy.  You have to set the
126f48169bbSJim Ingham     // explicit ThreadPlan you want to affect.
127f48169bbSJim Ingham }
128f48169bbSJim Ingham 
12930fdc8d8SChris Lattner bool
13030fdc8d8SChris Lattner ThreadPlan::WillResume (StateType resume_state, bool current_plan)
13130fdc8d8SChris Lattner {
13230fdc8d8SChris Lattner     if (current_plan)
13330fdc8d8SChris Lattner     {
1342d4edfbcSGreg Clayton         LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP));
13530fdc8d8SChris Lattner 
13630fdc8d8SChris Lattner         if (log)
1372cad65a5SGreg Clayton         {
1385ccbd294SGreg Clayton             RegisterContext *reg_ctx = m_thread.GetRegisterContext().get();
1392cad65a5SGreg Clayton             addr_t pc = reg_ctx->GetPC();
1402cad65a5SGreg Clayton             addr_t sp = reg_ctx->GetSP();
1412cad65a5SGreg Clayton             addr_t fp = reg_ctx->GetFP();
142b5c0d1ccSJim Ingham             log->Printf("%s Thread #%u: tid = 0x%4.4llx, pc = 0x%8.8llx, sp = 0x%8.8llx, fp = 0x%8.8llx, "
143b5c0d1ccSJim Ingham                         "plan = '%s', state = %s, stop others = %d",
1441346f7e0SGreg Clayton                         __FUNCTION__,
1452cad65a5SGreg Clayton                         m_thread.GetIndexID(),
1462cad65a5SGreg Clayton                         m_thread.GetID(),
1472cad65a5SGreg Clayton                         (uint64_t)pc,
1482cad65a5SGreg Clayton                         (uint64_t)sp,
1492cad65a5SGreg Clayton                         (uint64_t)fp,
1502cad65a5SGreg Clayton                         m_name.c_str(),
1512cad65a5SGreg Clayton                         StateAsCString(resume_state),
1522cad65a5SGreg Clayton                         StopOthers());
1532cad65a5SGreg Clayton         }
15430fdc8d8SChris Lattner     }
15530fdc8d8SChris Lattner     return true;
15630fdc8d8SChris Lattner }
15730fdc8d8SChris Lattner 
15830fdc8d8SChris Lattner lldb::user_id_t
15930fdc8d8SChris Lattner ThreadPlan::GetNextID()
16030fdc8d8SChris Lattner {
16130fdc8d8SChris Lattner     static uint32_t g_nextPlanID = 0;
16230fdc8d8SChris Lattner     return ++g_nextPlanID;
16330fdc8d8SChris Lattner }
16430fdc8d8SChris Lattner 
16530fdc8d8SChris Lattner void
16630fdc8d8SChris Lattner ThreadPlan::DidPush()
16730fdc8d8SChris Lattner {
16830fdc8d8SChris Lattner }
16930fdc8d8SChris Lattner 
17030fdc8d8SChris Lattner void
17130fdc8d8SChris Lattner ThreadPlan::WillPop()
17230fdc8d8SChris Lattner {
17330fdc8d8SChris Lattner }
17430fdc8d8SChris Lattner 
17530fdc8d8SChris Lattner bool
17630fdc8d8SChris Lattner ThreadPlan::OkayToDiscard()
17730fdc8d8SChris Lattner {
17830fdc8d8SChris Lattner     if (!IsMasterPlan())
17930fdc8d8SChris Lattner         return true;
18030fdc8d8SChris Lattner     else
18130fdc8d8SChris Lattner         return m_okay_to_discard;
18230fdc8d8SChris Lattner }
18330fdc8d8SChris Lattner 
18406e827ccSJim Ingham lldb::StateType
18506e827ccSJim Ingham ThreadPlan::RunState ()
18606e827ccSJim Ingham {
18706e827ccSJim Ingham     if (m_tracer_sp && m_tracer_sp->TracingEnabled() && m_tracer_sp->SingleStepEnabled())
18806e827ccSJim Ingham         return eStateStepping;
18906e827ccSJim Ingham     else
19006e827ccSJim Ingham         return GetPlanRunState();
19106e827ccSJim Ingham }
192