130fdc8d8SChris Lattner //===-- ThreadPlanStepOverBreakpoint.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/ThreadPlanStepOverBreakpoint.h" 1130fdc8d8SChris Lattner 1230fdc8d8SChris Lattner // C Includes 1330fdc8d8SChris Lattner // C++ Includes 1430fdc8d8SChris Lattner // Other libraries and framework includes 1530fdc8d8SChris Lattner // Project includes 1630fdc8d8SChris Lattner #include "lldb/lldb-private-log.h" 1730fdc8d8SChris Lattner #include "lldb/Core/Log.h" 1830fdc8d8SChris Lattner #include "lldb/Core/Stream.h" 1930fdc8d8SChris Lattner #include "lldb/Target/Process.h" 2030fdc8d8SChris Lattner #include "lldb/Target/RegisterContext.h" 2130fdc8d8SChris Lattner 2230fdc8d8SChris Lattner using namespace lldb; 2330fdc8d8SChris Lattner using namespace lldb_private; 2430fdc8d8SChris Lattner 2530fdc8d8SChris Lattner //---------------------------------------------------------------------- 2630fdc8d8SChris Lattner // ThreadPlanStepOverBreakpoint: Single steps over a breakpoint bp_site_sp at the pc. 2730fdc8d8SChris Lattner //---------------------------------------------------------------------- 2830fdc8d8SChris Lattner 2930fdc8d8SChris Lattner ThreadPlanStepOverBreakpoint::ThreadPlanStepOverBreakpoint (Thread &thread) : 30b01e742aSJim Ingham ThreadPlan (ThreadPlan::eKindStepOverBreakpoint, "Step over breakpoint trap", 3130fdc8d8SChris Lattner thread, 3230fdc8d8SChris Lattner eVoteNo, 3330fdc8d8SChris Lattner eVoteNoOpinion), // We need to report the run since this happens 3430fdc8d8SChris Lattner // first in the thread plan stack when stepping 3530fdc8d8SChris Lattner // over a breakpoint 36b01e742aSJim Ingham m_breakpoint_addr (LLDB_INVALID_ADDRESS), 376e0ff1a3SGreg Clayton m_auto_continue(false), 386e0ff1a3SGreg Clayton m_reenabled_breakpoint_site (false) 39b01e742aSJim Ingham 4030fdc8d8SChris Lattner { 4130fdc8d8SChris Lattner m_breakpoint_addr = m_thread.GetRegisterContext()->GetPC(); 421ac04c30SGreg Clayton m_breakpoint_site_id = m_thread.GetProcess()->GetBreakpointSiteList().FindIDByAddress (m_breakpoint_addr); 4330fdc8d8SChris Lattner } 4430fdc8d8SChris Lattner 4530fdc8d8SChris Lattner ThreadPlanStepOverBreakpoint::~ThreadPlanStepOverBreakpoint () 4630fdc8d8SChris Lattner { 4730fdc8d8SChris Lattner } 4830fdc8d8SChris Lattner 4930fdc8d8SChris Lattner void 5030fdc8d8SChris Lattner ThreadPlanStepOverBreakpoint::GetDescription (Stream *s, lldb::DescriptionLevel level) 5130fdc8d8SChris Lattner { 52d01b2953SDaniel Malea s->Printf("Single stepping past breakpoint site %" PRIu64 " at 0x%" PRIx64, m_breakpoint_site_id, (uint64_t)m_breakpoint_addr); 5330fdc8d8SChris Lattner } 5430fdc8d8SChris Lattner 5530fdc8d8SChris Lattner bool 5630fdc8d8SChris Lattner ThreadPlanStepOverBreakpoint::ValidatePlan (Stream *error) 5730fdc8d8SChris Lattner { 5830fdc8d8SChris Lattner return true; 5930fdc8d8SChris Lattner } 6030fdc8d8SChris Lattner 6130fdc8d8SChris Lattner bool 62221d51cfSJim Ingham ThreadPlanStepOverBreakpoint::DoPlanExplainsStop (Event *event_ptr) 6330fdc8d8SChris Lattner { 6460c4118cSJim Ingham StopInfoSP stop_info_sp = GetPrivateStopInfo (); 65bc1d7f7aSJim Ingham if (stop_info_sp) 66bc1d7f7aSJim Ingham { 67*8b91d0cdSJim Ingham // It's a little surprising that we stop here for a breakpoint hit. However, when you single step ONTO a breakpoint 68*8b91d0cdSJim Ingham // we still want to call that a breakpoint hit, and trigger the actions, etc. Otherwise you would see the 69*8b91d0cdSJim Ingham // PC at the breakpoint without having triggered the actions, then you'd continue, the PC wouldn't change, 70*8b91d0cdSJim Ingham // and you'd see the breakpoint hit, which would be odd. 71*8b91d0cdSJim Ingham // So the lower levels fake "step onto breakpoint address" and return that as a breakpoint. So our trace 72*8b91d0cdSJim Ingham // step COULD appear as a breakpoint hit if the next instruction also contained a breakpoint. 73bc1d7f7aSJim Ingham StopReason reason = stop_info_sp->GetStopReason(); 74*8b91d0cdSJim Ingham 75*8b91d0cdSJim Ingham switch (reason) 76*8b91d0cdSJim Ingham { 77*8b91d0cdSJim Ingham case eStopReasonTrace: 78*8b91d0cdSJim Ingham case eStopReasonNone: 7930fdc8d8SChris Lattner return true; 80*8b91d0cdSJim Ingham case eStopReasonBreakpoint: 81*8b91d0cdSJim Ingham // It's a little surprising that we stop here for a breakpoint hit. However, when you single step ONTO a 82*8b91d0cdSJim Ingham // breakpoint we still want to call that a breakpoint hit, and trigger the actions, etc. Otherwise you 83*8b91d0cdSJim Ingham // would see the PC at the breakpoint without having triggered the actions, then you'd continue, the PC 84*8b91d0cdSJim Ingham // wouldn't change, and you'd see the breakpoint hit, which would be odd. 85*8b91d0cdSJim Ingham // So the lower levels fake "step onto breakpoint address" and return that as a breakpoint hit. So our trace 86*8b91d0cdSJim Ingham // step COULD appear as a breakpoint hit if the next instruction also contained a breakpoint. We don't want 87*8b91d0cdSJim Ingham // to handle that, since we really don't know what to do with breakpoint hits. But make sure we don't set 88*8b91d0cdSJim Ingham // ourselves to auto-continue or we'll wrench control away from the plans that can deal with this. 89*8b91d0cdSJim Ingham SetAutoContinue(false); 90bc1d7f7aSJim Ingham return false; 91*8b91d0cdSJim Ingham default: 92*8b91d0cdSJim Ingham return false; 93*8b91d0cdSJim Ingham } 94bc1d7f7aSJim Ingham } 95bc1d7f7aSJim Ingham return false; 9630fdc8d8SChris Lattner } 9730fdc8d8SChris Lattner 9830fdc8d8SChris Lattner bool 9930fdc8d8SChris Lattner ThreadPlanStepOverBreakpoint::ShouldStop (Event *event_ptr) 10030fdc8d8SChris Lattner { 101*8b91d0cdSJim Ingham return !ShouldAutoContinue(event_ptr); 10230fdc8d8SChris Lattner } 10330fdc8d8SChris Lattner 10430fdc8d8SChris Lattner bool 10530fdc8d8SChris Lattner ThreadPlanStepOverBreakpoint::StopOthers () 10630fdc8d8SChris Lattner { 10730fdc8d8SChris Lattner return true; 10830fdc8d8SChris Lattner } 10930fdc8d8SChris Lattner 11030fdc8d8SChris Lattner StateType 11106e827ccSJim Ingham ThreadPlanStepOverBreakpoint::GetPlanRunState () 11230fdc8d8SChris Lattner { 11330fdc8d8SChris Lattner return eStateStepping; 11430fdc8d8SChris Lattner } 11530fdc8d8SChris Lattner 11630fdc8d8SChris Lattner bool 117221d51cfSJim Ingham ThreadPlanStepOverBreakpoint::DoWillResume (StateType resume_state, bool current_plan) 11830fdc8d8SChris Lattner { 11930fdc8d8SChris Lattner if (current_plan) 12030fdc8d8SChris Lattner { 1211ac04c30SGreg Clayton BreakpointSiteSP bp_site_sp (m_thread.GetProcess()->GetBreakpointSiteList().FindByAddress (m_breakpoint_addr)); 12230fdc8d8SChris Lattner if (bp_site_sp && bp_site_sp->IsEnabled()) 123299c0c1cSJim Ingham m_thread.GetProcess()->DisableBreakpointSite (bp_site_sp.get()); 12430fdc8d8SChris Lattner } 12530fdc8d8SChris Lattner return true; 12630fdc8d8SChris Lattner } 12730fdc8d8SChris Lattner 12830fdc8d8SChris Lattner bool 12930fdc8d8SChris Lattner ThreadPlanStepOverBreakpoint::WillStop () 13030fdc8d8SChris Lattner { 1316e0ff1a3SGreg Clayton ReenableBreakpointSite (); 13230fdc8d8SChris Lattner return true; 13330fdc8d8SChris Lattner } 13430fdc8d8SChris Lattner 13530fdc8d8SChris Lattner bool 13630fdc8d8SChris Lattner ThreadPlanStepOverBreakpoint::MischiefManaged () 13730fdc8d8SChris Lattner { 13830fdc8d8SChris Lattner lldb::addr_t pc_addr = m_thread.GetRegisterContext()->GetPC(); 13930fdc8d8SChris Lattner 14030fdc8d8SChris Lattner if (pc_addr == m_breakpoint_addr) 14130fdc8d8SChris Lattner { 14230fdc8d8SChris Lattner // If we are still at the PC of our breakpoint, then for some reason we didn't 14330fdc8d8SChris Lattner // get a chance to run. 14430fdc8d8SChris Lattner return false; 14530fdc8d8SChris Lattner } 14630fdc8d8SChris Lattner else 14730fdc8d8SChris Lattner { 1485160ce5cSGreg Clayton Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP)); 14930fdc8d8SChris Lattner if (log) 15030fdc8d8SChris Lattner log->Printf("Completed step over breakpoint plan."); 15130fdc8d8SChris Lattner // Otherwise, re-enable the breakpoint we were stepping over, and we're done. 1526e0ff1a3SGreg Clayton ReenableBreakpointSite (); 15330fdc8d8SChris Lattner ThreadPlan::MischiefManaged (); 15430fdc8d8SChris Lattner return true; 15530fdc8d8SChris Lattner } 15630fdc8d8SChris Lattner } 15730fdc8d8SChris Lattner 158b01e742aSJim Ingham void 1596e0ff1a3SGreg Clayton ThreadPlanStepOverBreakpoint::ReenableBreakpointSite () 1606e0ff1a3SGreg Clayton { 1616e0ff1a3SGreg Clayton if (!m_reenabled_breakpoint_site) 1626e0ff1a3SGreg Clayton { 1636e0ff1a3SGreg Clayton m_reenabled_breakpoint_site = true; 1646e0ff1a3SGreg Clayton BreakpointSiteSP bp_site_sp (m_thread.GetProcess()->GetBreakpointSiteList().FindByAddress (m_breakpoint_addr)); 1656e0ff1a3SGreg Clayton if (bp_site_sp) 1666e0ff1a3SGreg Clayton { 1676e0ff1a3SGreg Clayton m_thread.GetProcess()->EnableBreakpointSite (bp_site_sp.get()); 1686e0ff1a3SGreg Clayton } 1696e0ff1a3SGreg Clayton } 1706e0ff1a3SGreg Clayton } 1716e0ff1a3SGreg Clayton void 1726e0ff1a3SGreg Clayton ThreadPlanStepOverBreakpoint::ThreadDestroyed () 1736e0ff1a3SGreg Clayton { 1746e0ff1a3SGreg Clayton ReenableBreakpointSite (); 1756e0ff1a3SGreg Clayton } 1766e0ff1a3SGreg Clayton 1776e0ff1a3SGreg Clayton void 178b01e742aSJim Ingham ThreadPlanStepOverBreakpoint::SetAutoContinue (bool do_it) 179b01e742aSJim Ingham { 180b01e742aSJim Ingham m_auto_continue = do_it; 181b01e742aSJim Ingham } 182b01e742aSJim Ingham 183b01e742aSJim Ingham bool 184b01e742aSJim Ingham ThreadPlanStepOverBreakpoint::ShouldAutoContinue (Event *event_ptr) 185b01e742aSJim Ingham { 186b01e742aSJim Ingham return m_auto_continue; 187b01e742aSJim Ingham } 188*8b91d0cdSJim Ingham 189*8b91d0cdSJim Ingham bool 190*8b91d0cdSJim Ingham ThreadPlanStepOverBreakpoint::IsPlanStale() 191*8b91d0cdSJim Ingham { 192*8b91d0cdSJim Ingham return m_thread.GetRegisterContext()->GetPC() != m_breakpoint_addr; 193*8b91d0cdSJim Ingham } 194*8b91d0cdSJim Ingham 195