130fdc8d8SChris Lattner //===-- ThreadPlanStepOverRange.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/ThreadPlanStepOverRange.h" 1130fdc8d8SChris Lattner 1230fdc8d8SChris Lattner // C Includes 1330fdc8d8SChris Lattner // C++ Includes 1430fdc8d8SChris Lattner // Other libraries and framework includes 1530fdc8d8SChris Lattner // Project includes 1630fdc8d8SChris Lattner 1730fdc8d8SChris Lattner #include "lldb/lldb-private-log.h" 1830fdc8d8SChris Lattner #include "lldb/Core/Log.h" 1930fdc8d8SChris Lattner #include "lldb/Core/Stream.h" 2030fdc8d8SChris Lattner #include "lldb/Target/Process.h" 2130fdc8d8SChris Lattner #include "lldb/Target/RegisterContext.h" 22514487e8SGreg Clayton #include "lldb/Target/Target.h" 2330fdc8d8SChris Lattner #include "lldb/Target/Thread.h" 2430fdc8d8SChris Lattner #include "lldb/Target/ThreadPlanStepOut.h" 2530fdc8d8SChris Lattner #include "lldb/Target/ThreadPlanStepThrough.h" 2630fdc8d8SChris Lattner 2730fdc8d8SChris Lattner using namespace lldb_private; 282d4edfbcSGreg Clayton using namespace lldb; 2930fdc8d8SChris Lattner 3030fdc8d8SChris Lattner 3130fdc8d8SChris Lattner //---------------------------------------------------------------------- 3230fdc8d8SChris Lattner // ThreadPlanStepOverRange: Step through a stack range, either stepping over or into 3330fdc8d8SChris Lattner // based on the value of \a type. 3430fdc8d8SChris Lattner //---------------------------------------------------------------------- 3530fdc8d8SChris Lattner 3630fdc8d8SChris Lattner ThreadPlanStepOverRange::ThreadPlanStepOverRange 3730fdc8d8SChris Lattner ( 3830fdc8d8SChris Lattner Thread &thread, 3930fdc8d8SChris Lattner const AddressRange &range, 4030fdc8d8SChris Lattner const SymbolContext &addr_context, 4130fdc8d8SChris Lattner lldb::RunMode stop_others, 4230fdc8d8SChris Lattner bool okay_to_discard 4330fdc8d8SChris Lattner ) : 44b01e742aSJim Ingham ThreadPlanStepRange (ThreadPlan::eKindStepOverRange, "Step range stepping over", thread, range, addr_context, stop_others) 4530fdc8d8SChris Lattner { 4630fdc8d8SChris Lattner SetOkayToDiscard (okay_to_discard); 4730fdc8d8SChris Lattner } 4830fdc8d8SChris Lattner 4930fdc8d8SChris Lattner ThreadPlanStepOverRange::~ThreadPlanStepOverRange () 5030fdc8d8SChris Lattner { 5130fdc8d8SChris Lattner } 5230fdc8d8SChris Lattner 5330fdc8d8SChris Lattner void 5430fdc8d8SChris Lattner ThreadPlanStepOverRange::GetDescription (Stream *s, lldb::DescriptionLevel level) 5530fdc8d8SChris Lattner { 5630fdc8d8SChris Lattner if (level == lldb::eDescriptionLevelBrief) 5730fdc8d8SChris Lattner s->Printf("step over"); 5830fdc8d8SChris Lattner else 5930fdc8d8SChris Lattner { 6030fdc8d8SChris Lattner s->Printf ("stepping through range (stepping over functions): "); 61c4c9fedcSJim Ingham DumpRanges(s); 6230fdc8d8SChris Lattner } 6330fdc8d8SChris Lattner } 6430fdc8d8SChris Lattner 6530fdc8d8SChris Lattner bool 6625f66700SJim Ingham ThreadPlanStepOverRange::PlanExplainsStop () 6725f66700SJim Ingham { 6825f66700SJim Ingham // We don't explain signals or breakpoints (breakpoints that handle stepping in or 6925f66700SJim Ingham // out will be handled by a child plan. 7025f66700SJim Ingham StopInfoSP stop_info_sp = GetPrivateStopReason(); 7125f66700SJim Ingham if (stop_info_sp) 7225f66700SJim Ingham { 7325f66700SJim Ingham StopReason reason = stop_info_sp->GetStopReason(); 7425f66700SJim Ingham 7525f66700SJim Ingham switch (reason) 7625f66700SJim Ingham { 7725f66700SJim Ingham case eStopReasonBreakpoint: 7825f66700SJim Ingham case eStopReasonWatchpoint: 7925f66700SJim Ingham case eStopReasonSignal: 8025f66700SJim Ingham case eStopReasonException: 8125f66700SJim Ingham return false; 8225f66700SJim Ingham default: 8325f66700SJim Ingham return true; 8425f66700SJim Ingham } 8525f66700SJim Ingham } 8625f66700SJim Ingham return true; 8725f66700SJim Ingham } 8825f66700SJim Ingham 8925f66700SJim Ingham 9025f66700SJim Ingham bool 9130fdc8d8SChris Lattner ThreadPlanStepOverRange::ShouldStop (Event *event_ptr) 9230fdc8d8SChris Lattner { 932d4edfbcSGreg Clayton LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP)); 9430fdc8d8SChris Lattner 9530fdc8d8SChris Lattner if (log) 9630fdc8d8SChris Lattner { 9730fdc8d8SChris Lattner StreamString s; 98514487e8SGreg Clayton s.Address (m_thread.GetRegisterContext()->GetPC(), 991ac04c30SGreg Clayton m_thread.CalculateTarget()->GetArchitecture().GetAddressByteSize()); 10030fdc8d8SChris Lattner log->Printf("ThreadPlanStepOverRange reached %s.", s.GetData()); 10130fdc8d8SChris Lattner } 10230fdc8d8SChris Lattner 10330fdc8d8SChris Lattner // If we're still in the range, keep going. 10430fdc8d8SChris Lattner if (InRange()) 10530fdc8d8SChris Lattner return false; 10630fdc8d8SChris Lattner 10730fdc8d8SChris Lattner // If we're out of the range but in the same frame or in our caller's frame 10830fdc8d8SChris Lattner // then we should stop. 10930fdc8d8SChris Lattner // When stepping out we only step if we are forcing running one thread. 11030fdc8d8SChris Lattner bool stop_others; 11130fdc8d8SChris Lattner if (m_stop_others == lldb::eOnlyThisThread) 11230fdc8d8SChris Lattner stop_others = true; 11330fdc8d8SChris Lattner else 11430fdc8d8SChris Lattner stop_others = false; 11530fdc8d8SChris Lattner 11630fdc8d8SChris Lattner ThreadPlan* new_plan = NULL; 11730fdc8d8SChris Lattner 118*b5c0d1ccSJim Ingham FrameComparison frame_order = CompareCurrentFrameToStartFrame(); 119*b5c0d1ccSJim Ingham 120*b5c0d1ccSJim Ingham if (frame_order == eFrameCompareOlder) 1215822173bSJim Ingham { 1225822173bSJim Ingham // If we're in an older frame then we should stop. 1235822173bSJim Ingham // 1245822173bSJim Ingham // A caveat to this is if we think the frame is older but we're actually in a trampoline. 1255822173bSJim Ingham // I'm going to make the assumption that you wouldn't RETURN to a trampoline. So if we are 1265822173bSJim Ingham // in a trampoline we think the frame is older because the trampoline confused the backtracer. 1275822173bSJim Ingham // As below, we step through first, and then try to figure out how to get back out again. 1285822173bSJim Ingham 1295822173bSJim Ingham new_plan = m_thread.QueueThreadPlanForStepThrough (false, stop_others); 1305822173bSJim Ingham 1315822173bSJim Ingham if (new_plan != NULL && log) 1325822173bSJim Ingham log->Printf("Thought I stepped out, but in fact arrived at a trampoline."); 1335822173bSJim Ingham } 134*b5c0d1ccSJim Ingham else if (frame_order == eFrameCompareYounger) 135*b5c0d1ccSJim Ingham { 136*b5c0d1ccSJim Ingham // Make sure we really are in a new frame. Do that by unwinding and seeing if the 137*b5c0d1ccSJim Ingham // start function really is our start function... 138*b5c0d1ccSJim Ingham StackFrameSP older_frame_sp = m_thread.GetStackFrameAtIndex(1); 139*b5c0d1ccSJim Ingham 140*b5c0d1ccSJim Ingham // But if we can't even unwind one frame we should just get out of here & stop... 141*b5c0d1ccSJim Ingham if (older_frame_sp) 142*b5c0d1ccSJim Ingham { 143*b5c0d1ccSJim Ingham const SymbolContext &older_context = older_frame_sp->GetSymbolContext(eSymbolContextEverything); 144*b5c0d1ccSJim Ingham if (older_context == m_addr_context) 14530fdc8d8SChris Lattner { 146481cef25SGreg Clayton new_plan = m_thread.QueueThreadPlanForStepOut (false, 147481cef25SGreg Clayton NULL, 148481cef25SGreg Clayton true, 149481cef25SGreg Clayton stop_others, 150e0d378b3SGreg Clayton eVoteNo, 151e0d378b3SGreg Clayton eVoteNoOpinion, 152481cef25SGreg Clayton 0); 15330fdc8d8SChris Lattner } 154*b5c0d1ccSJim Ingham else 155*b5c0d1ccSJim Ingham { 156*b5c0d1ccSJim Ingham new_plan = m_thread.QueueThreadPlanForStepThrough (false, stop_others); 157*b5c0d1ccSJim Ingham 158*b5c0d1ccSJim Ingham } 159*b5c0d1ccSJim Ingham } 160*b5c0d1ccSJim Ingham } 16130fdc8d8SChris Lattner else if (!InSymbol()) 16230fdc8d8SChris Lattner { 16330fdc8d8SChris Lattner // This one is a little tricky. Sometimes we may be in a stub or something similar, 16430fdc8d8SChris Lattner // in which case we need to get out of there. But if we are in a stub then it's 16530fdc8d8SChris Lattner // likely going to be hard to get out from here. It is probably easiest to step into the 16630fdc8d8SChris Lattner // stub, and then it will be straight-forward to step out. 16730fdc8d8SChris Lattner new_plan = m_thread.QueueThreadPlanForStepThrough (false, stop_others); 16830fdc8d8SChris Lattner } 16930fdc8d8SChris Lattner 17030fdc8d8SChris Lattner if (new_plan == NULL) 17130fdc8d8SChris Lattner m_no_more_plans = true; 17230fdc8d8SChris Lattner else 17330fdc8d8SChris Lattner m_no_more_plans = false; 17430fdc8d8SChris Lattner 17530fdc8d8SChris Lattner if (new_plan == NULL) 17630fdc8d8SChris Lattner { 17730fdc8d8SChris Lattner // For efficiencies sake, we know we're done here so we don't have to do this 17830fdc8d8SChris Lattner // calculation again in MischiefManaged. 17930fdc8d8SChris Lattner SetPlanComplete(); 18030fdc8d8SChris Lattner return true; 18130fdc8d8SChris Lattner } 18230fdc8d8SChris Lattner else 18330fdc8d8SChris Lattner return false; 18430fdc8d8SChris Lattner } 185