130fdc8d8SChris Lattner //===-- ThreadPlanStepInstruction.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 
1130fdc8d8SChris Lattner #include "lldb/Target/ThreadPlanStepInstruction.h"
1230fdc8d8SChris Lattner 
1330fdc8d8SChris Lattner // C Includes
1430fdc8d8SChris Lattner // C++ Includes
1530fdc8d8SChris Lattner // Other libraries and framework includes
1630fdc8d8SChris Lattner // Project includes
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"
21f4b47e15SGreg Clayton #include "lldb/Target/RegisterContext.h"
22f4b47e15SGreg Clayton #include "lldb/Target/RegisterContext.h"
23f4b47e15SGreg Clayton #include "lldb/Target/StopInfo.h"
24f4b47e15SGreg Clayton #include "lldb/Target/Target.h"
2530fdc8d8SChris Lattner 
2630fdc8d8SChris Lattner using namespace lldb;
2730fdc8d8SChris Lattner using namespace lldb_private;
2830fdc8d8SChris Lattner 
2930fdc8d8SChris Lattner //----------------------------------------------------------------------
3030fdc8d8SChris Lattner // ThreadPlanStepInstruction: Step over the current instruction
3130fdc8d8SChris Lattner //----------------------------------------------------------------------
3230fdc8d8SChris Lattner 
3330fdc8d8SChris Lattner ThreadPlanStepInstruction::ThreadPlanStepInstruction
3430fdc8d8SChris Lattner (
3530fdc8d8SChris Lattner     Thread &thread,
3630fdc8d8SChris Lattner     bool step_over,
3730fdc8d8SChris Lattner     bool stop_other_threads,
3830fdc8d8SChris Lattner     Vote stop_vote,
3930fdc8d8SChris Lattner     Vote run_vote
4030fdc8d8SChris Lattner ) :
41b01e742aSJim Ingham     ThreadPlan (ThreadPlan::eKindStepInstruction, "Step over single instruction", thread, stop_vote, run_vote),
4230fdc8d8SChris Lattner     m_instruction_addr (0),
431ee0d4f7SBenjamin Kramer     m_stop_other_threads (stop_other_threads),
446b35c86fSJim Ingham     m_step_over (step_over)
451ee0d4f7SBenjamin Kramer {
4630fdc8d8SChris Lattner     m_instruction_addr = m_thread.GetRegisterContext()->GetPC(0);
476b35c86fSJim Ingham     m_stack_id = m_thread.GetStackFrameAtIndex(0)->GetStackID();
48*886a3e2cSJim Ingham     StackFrameSP parent_frame_sp = m_thread.GetStackFrameAtIndex(1);
49*886a3e2cSJim Ingham     if (parent_frame_sp)
50*886a3e2cSJim Ingham         m_parent_frame_id = parent_frame_sp->GetStackID();
5130fdc8d8SChris Lattner }
5230fdc8d8SChris Lattner 
5330fdc8d8SChris Lattner ThreadPlanStepInstruction::~ThreadPlanStepInstruction ()
5430fdc8d8SChris Lattner {
5530fdc8d8SChris Lattner }
5630fdc8d8SChris Lattner 
5730fdc8d8SChris Lattner void
5830fdc8d8SChris Lattner ThreadPlanStepInstruction::GetDescription (Stream *s, lldb::DescriptionLevel level)
5930fdc8d8SChris Lattner {
6030fdc8d8SChris Lattner     if (level == lldb::eDescriptionLevelBrief)
6130fdc8d8SChris Lattner     {
6230fdc8d8SChris Lattner         if (m_step_over)
6330fdc8d8SChris Lattner             s->Printf ("instruction step over");
6430fdc8d8SChris Lattner         else
6530fdc8d8SChris Lattner             s->Printf ("instruction step into");
6630fdc8d8SChris Lattner     }
6730fdc8d8SChris Lattner     else
6830fdc8d8SChris Lattner     {
6930fdc8d8SChris Lattner         s->Printf ("Stepping one instruction past ");
7030fdc8d8SChris Lattner         s->Address(m_instruction_addr, sizeof (addr_t));
7130fdc8d8SChris Lattner         if (m_step_over)
7230fdc8d8SChris Lattner             s->Printf(" stepping over calls");
7330fdc8d8SChris Lattner         else
7430fdc8d8SChris Lattner             s->Printf(" stepping into calls");
7530fdc8d8SChris Lattner     }
7630fdc8d8SChris Lattner }
7730fdc8d8SChris Lattner 
7830fdc8d8SChris Lattner bool
7930fdc8d8SChris Lattner ThreadPlanStepInstruction::ValidatePlan (Stream *error)
8030fdc8d8SChris Lattner {
8130fdc8d8SChris Lattner     // Since we read the instruction we're stepping over from the thread,
8230fdc8d8SChris Lattner     // this plan will always work.
8330fdc8d8SChris Lattner     return true;
8430fdc8d8SChris Lattner }
8530fdc8d8SChris Lattner 
8630fdc8d8SChris Lattner bool
87221d51cfSJim Ingham ThreadPlanStepInstruction::DoPlanExplainsStop (Event *event_ptr)
8830fdc8d8SChris Lattner {
8960c4118cSJim Ingham     StopInfoSP stop_info_sp = GetPrivateStopInfo ();
90b15bfc75SJim Ingham     if (stop_info_sp)
9130fdc8d8SChris Lattner     {
92b15bfc75SJim Ingham         StopReason reason = stop_info_sp->GetStopReason();
9330fdc8d8SChris Lattner         if (reason == eStopReasonTrace || reason == eStopReasonNone)
9430fdc8d8SChris Lattner             return true;
9530fdc8d8SChris Lattner         else
9630fdc8d8SChris Lattner             return false;
9730fdc8d8SChris Lattner     }
9830fdc8d8SChris Lattner     return false;
9930fdc8d8SChris Lattner }
10030fdc8d8SChris Lattner 
10130fdc8d8SChris Lattner bool
10230fdc8d8SChris Lattner ThreadPlanStepInstruction::ShouldStop (Event *event_ptr)
10330fdc8d8SChris Lattner {
10430fdc8d8SChris Lattner     if (m_step_over)
10530fdc8d8SChris Lattner     {
1065160ce5cSGreg Clayton         Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP));
1076b35c86fSJim Ingham 
1086b35c86fSJim Ingham         StackID cur_frame_zero_id = m_thread.GetStackFrameAtIndex(0)->GetStackID();
1096b35c86fSJim Ingham 
1106b35c86fSJim Ingham         if (cur_frame_zero_id == m_stack_id || m_stack_id < cur_frame_zero_id)
11130fdc8d8SChris Lattner         {
11230fdc8d8SChris Lattner             if (m_thread.GetRegisterContext()->GetPC(0) != m_instruction_addr)
11330fdc8d8SChris Lattner             {
11430fdc8d8SChris Lattner                 SetPlanComplete();
11530fdc8d8SChris Lattner                 return true;
11630fdc8d8SChris Lattner             }
11730fdc8d8SChris Lattner             else
11830fdc8d8SChris Lattner                 return false;
11930fdc8d8SChris Lattner         }
12030fdc8d8SChris Lattner         else
12130fdc8d8SChris Lattner         {
12230fdc8d8SChris Lattner             // We've stepped in, step back out again:
12330fdc8d8SChris Lattner             StackFrame *return_frame = m_thread.GetStackFrameAtIndex(1).get();
12430fdc8d8SChris Lattner             if (return_frame)
12530fdc8d8SChris Lattner             {
126*886a3e2cSJim Ingham                 if (return_frame->GetStackID() != m_parent_frame_id)
127*886a3e2cSJim Ingham                 {
12830fdc8d8SChris Lattner                     if (log)
12930fdc8d8SChris Lattner                     {
13030fdc8d8SChris Lattner                         StreamString s;
13130fdc8d8SChris Lattner                         s.PutCString ("Stepped in to: ");
1329da7bd07SGreg Clayton                         addr_t stop_addr = m_thread.GetStackFrameAtIndex(0)->GetRegisterContext()->GetPC();
1331ac04c30SGreg Clayton                         s.Address (stop_addr, m_thread.CalculateTarget()->GetArchitecture().GetAddressByteSize());
13430fdc8d8SChris Lattner                         s.PutCString (" stepping out to: ");
1359da7bd07SGreg Clayton                         addr_t return_addr = return_frame->GetRegisterContext()->GetPC();
1361ac04c30SGreg Clayton                         s.Address (return_addr, m_thread.CalculateTarget()->GetArchitecture().GetAddressByteSize());
13730fdc8d8SChris Lattner                         log->Printf("%s.", s.GetData());
13830fdc8d8SChris Lattner                     }
1394a58e968SJim Ingham 
1404a58e968SJim Ingham                     // StepInstruction should probably have the tri-state RunMode, but for now it is safer to
1414a58e968SJim Ingham                     // run others.
1424a58e968SJim Ingham                     const bool stop_others = false;
14364e7ead1SJim Ingham                     m_thread.QueueThreadPlanForStepOut(false,
14464e7ead1SJim Ingham                                                        NULL,
14564e7ead1SJim Ingham                                                        true,
1464a58e968SJim Ingham                                                        stop_others,
14764e7ead1SJim Ingham                                                        eVoteNo,
14864e7ead1SJim Ingham                                                        eVoteNoOpinion,
14964e7ead1SJim Ingham                                                        0);
15030fdc8d8SChris Lattner                     return false;
15130fdc8d8SChris Lattner                 }
15230fdc8d8SChris Lattner                 else
15330fdc8d8SChris Lattner                 {
15430fdc8d8SChris Lattner                     if (log)
155*886a3e2cSJim Ingham                     {
156*886a3e2cSJim Ingham                         log->PutCString("The stack id we are stepping in changed, but our parent frame did not.  "
157*886a3e2cSJim Ingham                         "We are probably just confused about where we are, stopping.");
158*886a3e2cSJim Ingham                     }
159*886a3e2cSJim Ingham                     SetPlanComplete();
160*886a3e2cSJim Ingham                     return true;
161*886a3e2cSJim Ingham                 }
162*886a3e2cSJim Ingham             }
163*886a3e2cSJim Ingham             else
164*886a3e2cSJim Ingham             {
165*886a3e2cSJim Ingham                 if (log)
16630fdc8d8SChris Lattner                     log->Printf("Could not find previous frame, stopping.");
16730fdc8d8SChris Lattner                 SetPlanComplete();
16830fdc8d8SChris Lattner                 return true;
16930fdc8d8SChris Lattner             }
17030fdc8d8SChris Lattner 
17130fdc8d8SChris Lattner         }
17230fdc8d8SChris Lattner 
17330fdc8d8SChris Lattner     }
17430fdc8d8SChris Lattner     else
17530fdc8d8SChris Lattner     {
17630fdc8d8SChris Lattner         if (m_thread.GetRegisterContext()->GetPC(0) != m_instruction_addr)
17730fdc8d8SChris Lattner         {
17830fdc8d8SChris Lattner             SetPlanComplete();
17930fdc8d8SChris Lattner             return true;
18030fdc8d8SChris Lattner         }
18130fdc8d8SChris Lattner         else
18230fdc8d8SChris Lattner             return false;
18330fdc8d8SChris Lattner     }
18430fdc8d8SChris Lattner }
18530fdc8d8SChris Lattner 
18630fdc8d8SChris Lattner bool
18730fdc8d8SChris Lattner ThreadPlanStepInstruction::StopOthers ()
18830fdc8d8SChris Lattner {
18930fdc8d8SChris Lattner     return m_stop_other_threads;
19030fdc8d8SChris Lattner }
19130fdc8d8SChris Lattner 
19230fdc8d8SChris Lattner StateType
19306e827ccSJim Ingham ThreadPlanStepInstruction::GetPlanRunState ()
19430fdc8d8SChris Lattner {
19530fdc8d8SChris Lattner     return eStateStepping;
19630fdc8d8SChris Lattner }
19730fdc8d8SChris Lattner 
19830fdc8d8SChris Lattner bool
19930fdc8d8SChris Lattner ThreadPlanStepInstruction::WillStop ()
20030fdc8d8SChris Lattner {
20130fdc8d8SChris Lattner     return true;
20230fdc8d8SChris Lattner }
20330fdc8d8SChris Lattner 
20430fdc8d8SChris Lattner bool
20530fdc8d8SChris Lattner ThreadPlanStepInstruction::MischiefManaged ()
20630fdc8d8SChris Lattner {
20730fdc8d8SChris Lattner     if (IsPlanComplete())
20830fdc8d8SChris Lattner     {
2095160ce5cSGreg Clayton         Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_STEP));
21030fdc8d8SChris Lattner         if (log)
21130fdc8d8SChris Lattner             log->Printf("Completed single instruction step plan.");
21230fdc8d8SChris Lattner         ThreadPlan::MischiefManaged ();
21330fdc8d8SChris Lattner         return true;
21430fdc8d8SChris Lattner     }
21530fdc8d8SChris Lattner     else
21630fdc8d8SChris Lattner     {
21730fdc8d8SChris Lattner         return false;
21830fdc8d8SChris Lattner     }
21930fdc8d8SChris Lattner }
22030fdc8d8SChris Lattner 
221