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), 3930fdc8d8SChris Lattner m_okay_to_discard (false) 4030fdc8d8SChris Lattner { 4130fdc8d8SChris Lattner SetID (GetNextID()); 4230fdc8d8SChris Lattner } 4330fdc8d8SChris Lattner 4430fdc8d8SChris Lattner //---------------------------------------------------------------------- 4530fdc8d8SChris Lattner // Destructor 4630fdc8d8SChris Lattner //---------------------------------------------------------------------- 4730fdc8d8SChris Lattner ThreadPlan::~ThreadPlan() 4830fdc8d8SChris Lattner { 4930fdc8d8SChris Lattner } 5030fdc8d8SChris Lattner 5130fdc8d8SChris Lattner const char * 5230fdc8d8SChris Lattner ThreadPlan::GetName () const 5330fdc8d8SChris Lattner { 5430fdc8d8SChris Lattner return m_name.c_str(); 5530fdc8d8SChris Lattner } 5630fdc8d8SChris Lattner 5730fdc8d8SChris Lattner Thread & 5830fdc8d8SChris Lattner ThreadPlan::GetThread() 5930fdc8d8SChris Lattner { 6030fdc8d8SChris Lattner return m_thread; 6130fdc8d8SChris Lattner } 6230fdc8d8SChris Lattner 6330fdc8d8SChris Lattner 6430fdc8d8SChris Lattner const Thread & 6530fdc8d8SChris Lattner ThreadPlan::GetThread() const 6630fdc8d8SChris Lattner { 6730fdc8d8SChris Lattner return m_thread; 6830fdc8d8SChris Lattner } 6930fdc8d8SChris Lattner 7030fdc8d8SChris Lattner bool 7130fdc8d8SChris Lattner ThreadPlan::IsPlanComplete () 7230fdc8d8SChris Lattner { 73b132097bSGreg Clayton Mutex::Locker locker(m_plan_complete_mutex); 7430fdc8d8SChris Lattner return m_plan_complete; 7530fdc8d8SChris Lattner } 7630fdc8d8SChris Lattner 7730fdc8d8SChris Lattner void 7830fdc8d8SChris Lattner ThreadPlan::SetPlanComplete () 7930fdc8d8SChris Lattner { 80b132097bSGreg Clayton Mutex::Locker locker(m_plan_complete_mutex); 8130fdc8d8SChris Lattner m_plan_complete = true; 8230fdc8d8SChris Lattner } 8330fdc8d8SChris Lattner 8430fdc8d8SChris Lattner bool 8530fdc8d8SChris Lattner ThreadPlan::MischiefManaged () 8630fdc8d8SChris Lattner { 87b132097bSGreg Clayton Mutex::Locker locker(m_plan_complete_mutex); 8830fdc8d8SChris Lattner m_plan_complete = true; 8930fdc8d8SChris Lattner return true; 9030fdc8d8SChris Lattner } 9130fdc8d8SChris Lattner 9230fdc8d8SChris Lattner Vote 9330fdc8d8SChris Lattner ThreadPlan::ShouldReportStop (Event *event_ptr) 9430fdc8d8SChris Lattner { 952d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP)); 962cad65a5SGreg Clayton 9730fdc8d8SChris Lattner if (m_stop_vote == eVoteNoOpinion) 9830fdc8d8SChris Lattner { 9930fdc8d8SChris Lattner ThreadPlan *prev_plan = GetPreviousPlan (); 10030fdc8d8SChris Lattner if (prev_plan) 1012cad65a5SGreg Clayton { 1022cad65a5SGreg Clayton Vote prev_vote = prev_plan->ShouldReportStop (event_ptr); 1032cad65a5SGreg Clayton if (log) 1041346f7e0SGreg Clayton log->Printf ("ThreadPlan::ShouldReportStop() returning previous thread plan vote: %s", GetVoteAsCString (prev_vote)); 1052cad65a5SGreg Clayton return prev_vote; 10630fdc8d8SChris Lattner } 1072cad65a5SGreg Clayton } 1082cad65a5SGreg Clayton if (log) 1091346f7e0SGreg Clayton log->Printf ("ThreadPlan::ShouldReportStop() returning vote: %s", GetVoteAsCString (m_stop_vote)); 11030fdc8d8SChris Lattner return m_stop_vote; 11130fdc8d8SChris Lattner } 11230fdc8d8SChris Lattner 11330fdc8d8SChris Lattner Vote 11430fdc8d8SChris Lattner ThreadPlan::ShouldReportRun (Event *event_ptr) 11530fdc8d8SChris Lattner { 11630fdc8d8SChris Lattner if (m_run_vote == eVoteNoOpinion) 11730fdc8d8SChris Lattner { 11830fdc8d8SChris Lattner ThreadPlan *prev_plan = GetPreviousPlan (); 11930fdc8d8SChris Lattner if (prev_plan) 12030fdc8d8SChris Lattner return prev_plan->ShouldReportRun (event_ptr); 12130fdc8d8SChris Lattner } 12230fdc8d8SChris Lattner return m_run_vote; 12330fdc8d8SChris Lattner } 12430fdc8d8SChris Lattner 12530fdc8d8SChris Lattner bool 12630fdc8d8SChris Lattner ThreadPlan::StopOthers () 12730fdc8d8SChris Lattner { 12830fdc8d8SChris Lattner ThreadPlan *prev_plan; 12930fdc8d8SChris Lattner prev_plan = GetPreviousPlan (); 13030fdc8d8SChris Lattner if (prev_plan == NULL) 13130fdc8d8SChris Lattner return false; 13230fdc8d8SChris Lattner else 13330fdc8d8SChris Lattner return prev_plan->StopOthers(); 13430fdc8d8SChris Lattner } 13530fdc8d8SChris Lattner 136*f48169bbSJim Ingham void 137*f48169bbSJim Ingham ThreadPlan::SetStopOthers (bool new_value) 138*f48169bbSJim Ingham { 139*f48169bbSJim Ingham // SetStopOthers doesn't work up the hierarchy. You have to set the 140*f48169bbSJim Ingham // explicit ThreadPlan you want to affect. 141*f48169bbSJim Ingham } 142*f48169bbSJim Ingham 14330fdc8d8SChris Lattner bool 14430fdc8d8SChris Lattner ThreadPlan::WillResume (StateType resume_state, bool current_plan) 14530fdc8d8SChris Lattner { 14630fdc8d8SChris Lattner if (current_plan) 14730fdc8d8SChris Lattner { 1482d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP)); 14930fdc8d8SChris Lattner 15030fdc8d8SChris Lattner if (log) 1512cad65a5SGreg Clayton { 1522cad65a5SGreg Clayton RegisterContext *reg_ctx = m_thread.GetRegisterContext(); 1532cad65a5SGreg Clayton addr_t pc = reg_ctx->GetPC(); 1542cad65a5SGreg Clayton addr_t sp = reg_ctx->GetSP(); 1552cad65a5SGreg Clayton addr_t fp = reg_ctx->GetFP(); 1561346f7e0SGreg Clayton log->Printf("%s Thread #%u: tid = 0x%4.4x, pc = 0x%8.8llx, sp = 0x%8.8llx, fp = 0x%8.8llx, plan = '%s', state = %s, stop others = %d", 1571346f7e0SGreg Clayton __FUNCTION__, 1582cad65a5SGreg Clayton m_thread.GetIndexID(), 1592cad65a5SGreg Clayton m_thread.GetID(), 1602cad65a5SGreg Clayton (uint64_t)pc, 1612cad65a5SGreg Clayton (uint64_t)sp, 1622cad65a5SGreg Clayton (uint64_t)fp, 1632cad65a5SGreg Clayton m_name.c_str(), 1642cad65a5SGreg Clayton StateAsCString(resume_state), 1652cad65a5SGreg Clayton StopOthers()); 1662cad65a5SGreg Clayton } 16730fdc8d8SChris Lattner } 16830fdc8d8SChris Lattner return true; 16930fdc8d8SChris Lattner } 17030fdc8d8SChris Lattner 17130fdc8d8SChris Lattner lldb::user_id_t 17230fdc8d8SChris Lattner ThreadPlan::GetNextID() 17330fdc8d8SChris Lattner { 17430fdc8d8SChris Lattner static uint32_t g_nextPlanID = 0; 17530fdc8d8SChris Lattner return ++g_nextPlanID; 17630fdc8d8SChris Lattner } 17730fdc8d8SChris Lattner 17830fdc8d8SChris Lattner void 17930fdc8d8SChris Lattner ThreadPlan::DidPush() 18030fdc8d8SChris Lattner { 18130fdc8d8SChris Lattner } 18230fdc8d8SChris Lattner 18330fdc8d8SChris Lattner void 18430fdc8d8SChris Lattner ThreadPlan::WillPop() 18530fdc8d8SChris Lattner { 18630fdc8d8SChris Lattner } 18730fdc8d8SChris Lattner 18830fdc8d8SChris Lattner void 18930fdc8d8SChris Lattner ThreadPlan::PushPlan (ThreadPlanSP &thread_plan_sp) 19030fdc8d8SChris Lattner { 19130fdc8d8SChris Lattner m_thread.PushPlan (thread_plan_sp); 19230fdc8d8SChris Lattner } 19330fdc8d8SChris Lattner 19430fdc8d8SChris Lattner ThreadPlan * 19530fdc8d8SChris Lattner ThreadPlan::GetPreviousPlan () 19630fdc8d8SChris Lattner { 19730fdc8d8SChris Lattner return m_thread.GetPreviousPlan (this); 19830fdc8d8SChris Lattner } 19930fdc8d8SChris Lattner 20030fdc8d8SChris Lattner void 20130fdc8d8SChris Lattner ThreadPlan::SetPrivate (bool input) 20230fdc8d8SChris Lattner { 20330fdc8d8SChris Lattner m_plan_private = input; 20430fdc8d8SChris Lattner } 20530fdc8d8SChris Lattner 20630fdc8d8SChris Lattner bool 20730fdc8d8SChris Lattner ThreadPlan::GetPrivate (void) 20830fdc8d8SChris Lattner { 20930fdc8d8SChris Lattner return m_plan_private; 21030fdc8d8SChris Lattner } 21130fdc8d8SChris Lattner 21230fdc8d8SChris Lattner bool 21330fdc8d8SChris Lattner ThreadPlan::OkayToDiscard() 21430fdc8d8SChris Lattner { 21530fdc8d8SChris Lattner if (!IsMasterPlan()) 21630fdc8d8SChris Lattner return true; 21730fdc8d8SChris Lattner else 21830fdc8d8SChris Lattner return m_okay_to_discard; 21930fdc8d8SChris Lattner } 22030fdc8d8SChris Lattner 22106e827ccSJim Ingham lldb::StateType 22206e827ccSJim Ingham ThreadPlan::RunState () 22306e827ccSJim Ingham { 22406e827ccSJim Ingham if (m_tracer_sp && m_tracer_sp->TracingEnabled() && m_tracer_sp->SingleStepEnabled()) 22506e827ccSJim Ingham return eStateStepping; 22606e827ccSJim Ingham else 22706e827ccSJim Ingham return GetPlanRunState(); 22806e827ccSJim Ingham } 229