2010-06-09 00:52:24 +08:00
|
|
|
//===-- ThreadPlanShouldStopHere.cpp ----------------------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2015-12-15 09:33:19 +08:00
|
|
|
// C Includes
|
|
|
|
// C++ Includes
|
|
|
|
// Other libraries and framework includes
|
|
|
|
// Project includes
|
2016-05-07 07:44:10 +08:00
|
|
|
#include "lldb/Core/Log.h"
|
|
|
|
#include "lldb/Symbol/Symbol.h"
|
2011-12-03 09:52:59 +08:00
|
|
|
#include "lldb/Target/RegisterContext.h"
|
2010-06-09 00:52:24 +08:00
|
|
|
#include "lldb/Target/Thread.h"
|
|
|
|
#include "lldb/Target/ThreadPlanShouldStopHere.h"
|
|
|
|
|
|
|
|
using namespace lldb;
|
|
|
|
using namespace lldb_private;
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
// ThreadPlanShouldStopHere constructor
|
|
|
|
//----------------------------------------------------------------------
|
2014-03-13 10:47:14 +08:00
|
|
|
ThreadPlanShouldStopHere::ThreadPlanShouldStopHere(ThreadPlan *owner) :
|
2015-12-15 09:33:19 +08:00
|
|
|
m_callbacks(),
|
|
|
|
m_baton(nullptr),
|
|
|
|
m_owner(owner),
|
|
|
|
m_flags(ThreadPlanShouldStopHere::eNone)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2014-03-13 10:47:14 +08:00
|
|
|
m_callbacks.should_stop_here_callback = ThreadPlanShouldStopHere::DefaultShouldStopHereCallback;
|
|
|
|
m_callbacks.step_from_here_callback = ThreadPlanShouldStopHere::DefaultStepFromHereCallback;
|
|
|
|
}
|
|
|
|
|
|
|
|
ThreadPlanShouldStopHere::ThreadPlanShouldStopHere(ThreadPlan *owner, const ThreadPlanShouldStopHereCallbacks *callbacks, void *baton) :
|
|
|
|
m_callbacks (),
|
|
|
|
m_baton (),
|
|
|
|
m_owner (owner),
|
|
|
|
m_flags (ThreadPlanShouldStopHere::eNone)
|
|
|
|
{
|
|
|
|
SetShouldStopHereCallbacks(callbacks, baton);
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2015-12-15 09:33:19 +08:00
|
|
|
ThreadPlanShouldStopHere::~ThreadPlanShouldStopHere() = default;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2014-03-13 10:47:14 +08:00
|
|
|
bool
|
|
|
|
ThreadPlanShouldStopHere::InvokeShouldStopHereCallback (FrameComparison operation)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2014-03-13 10:47:14 +08:00
|
|
|
bool should_stop_here = true;
|
|
|
|
if (m_callbacks.should_stop_here_callback)
|
2011-12-03 09:52:59 +08:00
|
|
|
{
|
2014-03-13 10:47:14 +08:00
|
|
|
should_stop_here = m_callbacks.should_stop_here_callback (m_owner, m_flags, operation, m_baton);
|
2013-03-28 07:08:40 +08:00
|
|
|
Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP));
|
2011-12-03 09:52:59 +08:00
|
|
|
if (log)
|
|
|
|
{
|
|
|
|
lldb::addr_t current_addr = m_owner->GetThread().GetRegisterContext()->GetPC(0);
|
|
|
|
|
2014-03-13 10:47:14 +08:00
|
|
|
log->Printf ("ShouldStopHere callback returned %u from 0x%" PRIx64 ".", should_stop_here, current_addr);
|
2011-12-03 09:52:59 +08:00
|
|
|
}
|
2014-03-13 10:47:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return should_stop_here;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
ThreadPlanShouldStopHere::DefaultShouldStopHereCallback (ThreadPlan *current_plan,
|
|
|
|
Flags &flags,
|
|
|
|
FrameComparison operation,
|
|
|
|
void *baton)
|
|
|
|
{
|
|
|
|
bool should_stop_here = true;
|
|
|
|
StackFrame *frame = current_plan->GetThread().GetStackFrameAtIndex(0).get();
|
2014-03-18 07:03:34 +08:00
|
|
|
if (!frame)
|
|
|
|
return true;
|
|
|
|
|
2014-03-13 10:47:14 +08:00
|
|
|
Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP));
|
|
|
|
|
|
|
|
if ((operation == eFrameCompareOlder && flags.Test(eStepOutAvoidNoDebug))
|
2014-08-06 09:49:59 +08:00
|
|
|
|| (operation == eFrameCompareYounger && flags.Test(eStepInAvoidNoDebug))
|
|
|
|
|| (operation == eFrameCompareSameParent && flags.Test(eStepInAvoidNoDebug)))
|
2014-03-13 10:47:14 +08:00
|
|
|
{
|
|
|
|
if (!frame->HasDebugInformation())
|
|
|
|
{
|
|
|
|
if (log)
|
|
|
|
log->Printf ("Stepping out of frame with no debug info");
|
|
|
|
|
|
|
|
should_stop_here = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-18 07:03:34 +08:00
|
|
|
// Always avoid code with line number 0.
|
|
|
|
// FIXME: At present the ShouldStop and the StepFromHere calculate this independently. If this ever
|
|
|
|
// becomes expensive (this one isn't) we can try to have this set a state that the StepFromHere can use.
|
|
|
|
if (frame)
|
|
|
|
{
|
|
|
|
SymbolContext sc;
|
|
|
|
sc = frame->GetSymbolContext (eSymbolContextLineEntry);
|
|
|
|
if (sc.line_entry.line == 0)
|
|
|
|
should_stop_here = false;
|
|
|
|
}
|
|
|
|
|
2014-03-13 10:47:14 +08:00
|
|
|
return should_stop_here;
|
|
|
|
}
|
|
|
|
|
|
|
|
ThreadPlanSP
|
|
|
|
ThreadPlanShouldStopHere::DefaultStepFromHereCallback (ThreadPlan *current_plan,
|
|
|
|
Flags &flags,
|
|
|
|
FrameComparison operation,
|
|
|
|
void *baton)
|
|
|
|
{
|
2014-03-18 07:03:34 +08:00
|
|
|
const bool stop_others = false;
|
|
|
|
const size_t frame_index = 0;
|
|
|
|
ThreadPlanSP return_plan_sp;
|
|
|
|
// If we are stepping through code at line number 0, then we need to step over this range. Otherwise
|
|
|
|
// we will step out.
|
2016-05-07 07:44:10 +08:00
|
|
|
Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP));
|
|
|
|
|
2014-03-18 07:03:34 +08:00
|
|
|
StackFrame *frame = current_plan->GetThread().GetStackFrameAtIndex(0).get();
|
|
|
|
if (!frame)
|
2013-07-19 05:48:26 +08:00
|
|
|
return return_plan_sp;
|
2014-03-18 07:03:34 +08:00
|
|
|
SymbolContext sc;
|
2016-05-07 07:44:10 +08:00
|
|
|
sc = frame->GetSymbolContext (eSymbolContextLineEntry|eSymbolContextSymbol);
|
2016-05-20 06:22:57 +08:00
|
|
|
|
2014-03-18 07:03:34 +08:00
|
|
|
if (sc.line_entry.line == 0)
|
|
|
|
{
|
|
|
|
AddressRange range = sc.line_entry.range;
|
2016-05-07 07:44:10 +08:00
|
|
|
|
2016-05-20 06:22:57 +08:00
|
|
|
// If the whole function is marked line 0 just step out, that's easier & faster than continuing
|
2016-05-07 07:44:10 +08:00
|
|
|
// to step through it.
|
2016-05-20 06:22:57 +08:00
|
|
|
bool just_step_out = false;
|
|
|
|
if (sc.symbol && sc.symbol->ValueIsAddress())
|
|
|
|
{
|
|
|
|
Address symbol_end = sc.symbol->GetAddress();
|
|
|
|
symbol_end.Slide(sc.symbol->GetByteSize() - 1);
|
2016-05-07 07:44:10 +08:00
|
|
|
if (range.ContainsFileAddress(sc.symbol->GetAddress()) && range.ContainsFileAddress(symbol_end))
|
2016-05-20 06:22:57 +08:00
|
|
|
{
|
|
|
|
if (log)
|
|
|
|
log->Printf("Stopped in a function with only line 0 lines, just stepping out.");
|
2016-05-07 07:44:10 +08:00
|
|
|
just_step_out = true;
|
|
|
|
}
|
2016-05-20 06:22:57 +08:00
|
|
|
}
|
2016-05-07 07:44:10 +08:00
|
|
|
if (!just_step_out)
|
|
|
|
{
|
2016-05-20 06:22:57 +08:00
|
|
|
if (log)
|
|
|
|
log->Printf ("ThreadPlanShouldStopHere::DefaultStepFromHereCallback Queueing StepInRange plan to step through line 0 code.");
|
2016-05-07 07:44:10 +08:00
|
|
|
|
2016-05-20 06:22:57 +08:00
|
|
|
return_plan_sp = current_plan->GetThread().QueueThreadPlanForStepInRange(false,
|
|
|
|
range,
|
|
|
|
sc,
|
|
|
|
NULL,
|
|
|
|
eOnlyDuringStepping,
|
|
|
|
eLazyBoolCalculate,
|
|
|
|
eLazyBoolNo);
|
2016-05-07 07:44:10 +08:00
|
|
|
}
|
2014-03-18 07:03:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!return_plan_sp)
|
2015-12-15 09:33:19 +08:00
|
|
|
return_plan_sp = current_plan->GetThread().QueueThreadPlanForStepOutNoShouldStop(false,
|
|
|
|
nullptr,
|
|
|
|
true,
|
|
|
|
stop_others,
|
|
|
|
eVoteNo,
|
|
|
|
eVoteNoOpinion,
|
2016-01-09 05:40:11 +08:00
|
|
|
frame_index,
|
|
|
|
true);
|
2014-03-18 07:03:34 +08:00
|
|
|
return return_plan_sp;
|
2014-03-13 10:47:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
ThreadPlanSP
|
|
|
|
ThreadPlanShouldStopHere::QueueStepOutFromHerePlan(lldb_private::Flags &flags, lldb::FrameComparison operation)
|
|
|
|
{
|
|
|
|
ThreadPlanSP return_plan_sp;
|
|
|
|
if (m_callbacks.step_from_here_callback)
|
|
|
|
{
|
|
|
|
return_plan_sp = m_callbacks.step_from_here_callback (m_owner, flags, operation, m_baton);
|
2011-12-03 09:52:59 +08:00
|
|
|
}
|
2014-03-13 10:47:14 +08:00
|
|
|
return return_plan_sp;
|
|
|
|
}
|
|
|
|
|
|
|
|
lldb::ThreadPlanSP
|
|
|
|
ThreadPlanShouldStopHere::CheckShouldStopHereAndQueueStepOut (lldb::FrameComparison operation)
|
|
|
|
{
|
|
|
|
if (!InvokeShouldStopHereCallback(operation))
|
|
|
|
return QueueStepOutFromHerePlan(m_flags, operation);
|
2010-06-09 00:52:24 +08:00
|
|
|
else
|
2013-07-19 05:48:26 +08:00
|
|
|
return ThreadPlanSP();
|
2011-01-26 07:55:37 +08:00
|
|
|
}
|