mirror of
https://github.com/intel/llvm.git
synced 2026-01-17 14:48:27 +08:00
value. This fixes problems, for instance, with the StepRange plans, where they know that they explained the stop because they were at their "run to here" breakpoint, then deleted that breakpoint, so when they got asked again, doh! I had done this for a couple of plans in an ad hoc fashion, this just formalizes it. Also add a "ResumeRequested" in Process so that the code in the completion handlers can tell the ShouldStop logic they want to resume rather than just directly resuming. That allows us to handle resuming in a more controlled fashion. Also, SetPublicState can take a "restarted" flag, so that it doesn't drop the run lock when the target was immediately restarted. --This line, and those below , will be ignored-- M test/lang/objc/objc-dynamic-value/TestObjCDynamicValue.py M include/lldb/Target/ThreadList.h M include/lldb/Target/ThreadPlanStepOut.h M include/lldb/Target/Thread.h M include/lldb/Target/ThreadPlanBase.h M include/lldb/Target/ThreadPlanStepThrough.h M include/lldb/Target/ThreadPlanStepInstruction.h M include/lldb/Target/ThreadPlanStepInRange.h M include/lldb/Target/ThreadPlanStepOverBreakpoint.h M include/lldb/Target/ThreadPlanStepUntil.h M include/lldb/Target/StopInfo.h M include/lldb/Target/Process.h M include/lldb/Target/ThreadPlanRunToAddress.h M include/lldb/Target/ThreadPlan.h M include/lldb/Target/ThreadPlanCallFunction.h M include/lldb/Target/ThreadPlanStepOverRange.h M source/Plugins/LanguageRuntime/ObjC/AppleObjCRuntime/AppleThreadPlanStepThroughObjCTrampoline.h M source/Plugins/LanguageRuntime/ObjC/AppleObjCRuntime/AppleThreadPlanStepThroughObjCTrampoline.cpp M source/Target/StopInfo.cpp M source/Target/Process.cpp M source/Target/ThreadPlanRunToAddress.cpp M source/Target/ThreadPlan.cpp M source/Target/ThreadPlanCallFunction.cpp M source/Target/ThreadPlanStepOverRange.cpp M source/Target/ThreadList.cpp M source/Target/ThreadPlanStepOut.cpp M source/Target/Thread.cpp M source/Target/ThreadPlanBase.cpp M source/Target/ThreadPlanStepThrough.cpp M source/Target/ThreadPlanStepInstruction.cpp M source/Target/ThreadPlanStepInRange.cpp M source/Target/ThreadPlanStepOverBreakpoint.cpp M source/Target/ThreadPlanStepUntil.cpp M lldb.xcodeproj/xcshareddata/xcschemes/Run Testsuite.xcscheme llvm-svn: 181381
269 lines
7.0 KiB
C++
269 lines
7.0 KiB
C++
//===-- ThreadPlanRunToAddress.cpp ------------------------------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "lldb/Target/ThreadPlanRunToAddress.h"
|
|
|
|
// C Includes
|
|
// C++ Includes
|
|
// Other libraries and framework includes
|
|
// Project includes
|
|
#include "lldb/lldb-private-log.h"
|
|
#include "lldb/Core/Log.h"
|
|
#include "lldb/Core/Stream.h"
|
|
#include "lldb/Target/Target.h"
|
|
#include "lldb/Target/Process.h"
|
|
#include "lldb/Target/Thread.h"
|
|
#include "lldb/Target/RegisterContext.h"
|
|
|
|
using namespace lldb;
|
|
using namespace lldb_private;
|
|
|
|
//----------------------------------------------------------------------
|
|
// ThreadPlanRunToAddress: Continue plan
|
|
//----------------------------------------------------------------------
|
|
|
|
ThreadPlanRunToAddress::ThreadPlanRunToAddress
|
|
(
|
|
Thread &thread,
|
|
Address &address,
|
|
bool stop_others
|
|
) :
|
|
ThreadPlan (ThreadPlan::eKindRunToAddress, "Run to address plan", thread, eVoteNoOpinion, eVoteNoOpinion),
|
|
m_stop_others (stop_others),
|
|
m_addresses (),
|
|
m_break_ids ()
|
|
{
|
|
m_addresses.push_back (address.GetOpcodeLoadAddress (m_thread.CalculateTarget().get()));
|
|
SetInitialBreakpoints();
|
|
}
|
|
|
|
ThreadPlanRunToAddress::ThreadPlanRunToAddress
|
|
(
|
|
Thread &thread,
|
|
lldb::addr_t address,
|
|
bool stop_others
|
|
) :
|
|
ThreadPlan (ThreadPlan::eKindRunToAddress, "Run to address plan", thread, eVoteNoOpinion, eVoteNoOpinion),
|
|
m_stop_others (stop_others),
|
|
m_addresses (),
|
|
m_break_ids ()
|
|
{
|
|
m_addresses.push_back(m_thread.CalculateTarget()->GetOpcodeLoadAddress(address));
|
|
SetInitialBreakpoints();
|
|
}
|
|
|
|
ThreadPlanRunToAddress::ThreadPlanRunToAddress
|
|
(
|
|
Thread &thread,
|
|
const std::vector<lldb::addr_t> &addresses,
|
|
bool stop_others
|
|
) :
|
|
ThreadPlan (ThreadPlan::eKindRunToAddress, "Run to address plan", thread, eVoteNoOpinion, eVoteNoOpinion),
|
|
m_stop_others (stop_others),
|
|
m_addresses (addresses),
|
|
m_break_ids ()
|
|
{
|
|
// Convert all addressses into opcode addresses to make sure we set
|
|
// breakpoints at the correct address.
|
|
Target &target = thread.GetProcess()->GetTarget();
|
|
std::vector<lldb::addr_t>::iterator pos, end = m_addresses.end();
|
|
for (pos = m_addresses.begin(); pos != end; ++pos)
|
|
*pos = target.GetOpcodeLoadAddress (*pos);
|
|
|
|
SetInitialBreakpoints();
|
|
}
|
|
|
|
void
|
|
ThreadPlanRunToAddress::SetInitialBreakpoints ()
|
|
{
|
|
size_t num_addresses = m_addresses.size();
|
|
m_break_ids.resize(num_addresses);
|
|
|
|
for (size_t i = 0; i < num_addresses; i++)
|
|
{
|
|
Breakpoint *breakpoint;
|
|
breakpoint = m_thread.CalculateTarget()->CreateBreakpoint (m_addresses[i], true).get();
|
|
if (breakpoint != NULL)
|
|
{
|
|
m_break_ids[i] = breakpoint->GetID();
|
|
breakpoint->SetThreadID(m_thread.GetID());
|
|
breakpoint->SetBreakpointKind("run-to-address");
|
|
}
|
|
}
|
|
}
|
|
|
|
ThreadPlanRunToAddress::~ThreadPlanRunToAddress ()
|
|
{
|
|
size_t num_break_ids = m_break_ids.size();
|
|
for (size_t i = 0; i < num_break_ids; i++)
|
|
{
|
|
m_thread.CalculateTarget()->RemoveBreakpointByID (m_break_ids[i]);
|
|
}
|
|
}
|
|
|
|
void
|
|
ThreadPlanRunToAddress::GetDescription (Stream *s, lldb::DescriptionLevel level)
|
|
{
|
|
size_t num_addresses = m_addresses.size();
|
|
|
|
if (level == lldb::eDescriptionLevelBrief)
|
|
{
|
|
if (num_addresses == 0)
|
|
{
|
|
s->Printf ("run to address with no addresses given.");
|
|
return;
|
|
}
|
|
else if (num_addresses == 1)
|
|
s->Printf ("run to address: ");
|
|
else
|
|
s->Printf ("run to addresses: ");
|
|
|
|
for (size_t i = 0; i < num_addresses; i++)
|
|
{
|
|
s->Address (m_addresses[i], sizeof (addr_t));
|
|
s->Printf(" ");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (num_addresses == 0)
|
|
{
|
|
s->Printf ("run to address with no addresses given.");
|
|
return;
|
|
}
|
|
else if (num_addresses == 1)
|
|
s->Printf ("Run to address: ");
|
|
else
|
|
{
|
|
s->Printf ("Run to addresses: ");
|
|
}
|
|
|
|
for (size_t i = 0; i < num_addresses; i++)
|
|
{
|
|
if (num_addresses > 1)
|
|
{
|
|
s->Printf("\n");
|
|
s->Indent();
|
|
}
|
|
|
|
s->Address(m_addresses[i], sizeof (addr_t));
|
|
s->Printf (" using breakpoint: %d - ", m_break_ids[i]);
|
|
Breakpoint *breakpoint = m_thread.CalculateTarget()->GetBreakpointByID (m_break_ids[i]).get();
|
|
if (breakpoint)
|
|
breakpoint->Dump (s);
|
|
else
|
|
s->Printf ("but the breakpoint has been deleted.");
|
|
}
|
|
}
|
|
}
|
|
|
|
bool
|
|
ThreadPlanRunToAddress::ValidatePlan (Stream *error)
|
|
{
|
|
// If we couldn't set the breakpoint for some reason, then this won't
|
|
// work.
|
|
bool all_bps_good = true;
|
|
size_t num_break_ids = m_break_ids.size();
|
|
|
|
for (size_t i = 0; i < num_break_ids; i++)
|
|
{
|
|
if (m_break_ids[i] == LLDB_INVALID_BREAK_ID)
|
|
{
|
|
all_bps_good = false;
|
|
if (error)
|
|
{
|
|
error->Printf ("Could not set breakpoint for address: ");
|
|
error->Address (m_addresses[i], sizeof (addr_t));
|
|
error->Printf ("\n");
|
|
}
|
|
}
|
|
}
|
|
return all_bps_good;
|
|
}
|
|
|
|
bool
|
|
ThreadPlanRunToAddress::DoPlanExplainsStop (Event *event_ptr)
|
|
{
|
|
return AtOurAddress();
|
|
}
|
|
|
|
bool
|
|
ThreadPlanRunToAddress::ShouldStop (Event *event_ptr)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
bool
|
|
ThreadPlanRunToAddress::StopOthers ()
|
|
{
|
|
return m_stop_others;
|
|
}
|
|
|
|
void
|
|
ThreadPlanRunToAddress::SetStopOthers (bool new_value)
|
|
{
|
|
m_stop_others = new_value;
|
|
}
|
|
|
|
StateType
|
|
ThreadPlanRunToAddress::GetPlanRunState ()
|
|
{
|
|
return eStateRunning;
|
|
}
|
|
|
|
bool
|
|
ThreadPlanRunToAddress::WillStop ()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
ThreadPlanRunToAddress::MischiefManaged ()
|
|
{
|
|
Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP));
|
|
|
|
if (AtOurAddress())
|
|
{
|
|
// Remove the breakpoint
|
|
size_t num_break_ids = m_break_ids.size();
|
|
|
|
for (size_t i = 0; i < num_break_ids; i++)
|
|
{
|
|
if (m_break_ids[i] != LLDB_INVALID_BREAK_ID)
|
|
{
|
|
m_thread.CalculateTarget()->RemoveBreakpointByID (m_break_ids[i]);
|
|
m_break_ids[i] = LLDB_INVALID_BREAK_ID;
|
|
}
|
|
}
|
|
if (log)
|
|
log->Printf("Completed run to address plan.");
|
|
ThreadPlan::MischiefManaged ();
|
|
return true;
|
|
}
|
|
else
|
|
return false;
|
|
}
|
|
|
|
bool
|
|
ThreadPlanRunToAddress::AtOurAddress ()
|
|
{
|
|
lldb::addr_t current_address = m_thread.GetRegisterContext()->GetPC();
|
|
bool found_it = false;
|
|
size_t num_addresses = m_addresses.size();
|
|
for (size_t i = 0; i < num_addresses; i++)
|
|
{
|
|
if (m_addresses[i] == current_address)
|
|
{
|
|
found_it = true;
|
|
break;
|
|
}
|
|
}
|
|
return found_it;
|
|
}
|