106e827ccSJim Ingham //===-- ThreadPlan.cpp ------------------------------------------*- C++ -*-===//
206e827ccSJim Ingham //
306e827ccSJim Ingham //                     The LLVM Compiler Infrastructure
406e827ccSJim Ingham //
506e827ccSJim Ingham // This file is distributed under the University of Illinois Open Source
606e827ccSJim Ingham // License. See LICENSE.TXT for details.
706e827ccSJim Ingham //
806e827ccSJim Ingham //===----------------------------------------------------------------------===//
906e827ccSJim Ingham 
1006e827ccSJim Ingham #include "lldb/Target/ThreadPlan.h"
1106e827ccSJim Ingham 
1206e827ccSJim Ingham // C Includes
138c9e5383SSean Callanan #include <string.h>
1406e827ccSJim Ingham // C++ Includes
1506e827ccSJim Ingham // Other libraries and framework includes
1606e827ccSJim Ingham // Project includes
178c9e5383SSean Callanan #include "lldb/Core/ArchSpec.h"
188c9e5383SSean Callanan #include "lldb/Core/DataBufferHeap.h"
1906e827ccSJim Ingham #include "lldb/Core/Debugger.h"
208c9e5383SSean Callanan #include "lldb/Core/Disassembler.h"
2106e827ccSJim Ingham #include "lldb/Core/Log.h"
2206e827ccSJim Ingham #include "lldb/Core/State.h"
238c9e5383SSean Callanan #include "lldb/Core/Value.h"
248c9e5383SSean Callanan #include "lldb/Symbol/TypeList.h"
2506e827ccSJim Ingham #include "lldb/Target/RegisterContext.h"
2606e827ccSJim Ingham #include "lldb/Target/Thread.h"
2706e827ccSJim Ingham #include "lldb/Target/Process.h"
2806e827ccSJim Ingham #include "lldb/Target/Target.h"
2906e827ccSJim Ingham 
3006e827ccSJim Ingham using namespace lldb;
3106e827ccSJim Ingham using namespace lldb_private;
3206e827ccSJim Ingham 
338c9e5383SSean Callanan #pragma mark ThreadPlanTracer
348c9e5383SSean Callanan 
3506e827ccSJim Ingham ThreadPlanTracer::ThreadPlanTracer (Thread &thread, lldb::StreamSP &stream_sp) :
3606e827ccSJim Ingham     m_single_step(true),
3706e827ccSJim Ingham     m_enabled (false),
3806e827ccSJim Ingham     m_thread (thread),
3906e827ccSJim Ingham     m_stream_sp (stream_sp)
4006e827ccSJim Ingham {
4106e827ccSJim Ingham }
4206e827ccSJim Ingham 
4306e827ccSJim Ingham ThreadPlanTracer::ThreadPlanTracer (Thread &thread) :
4406e827ccSJim Ingham     m_single_step(true),
4506e827ccSJim Ingham     m_enabled (false),
4606e827ccSJim Ingham     m_thread (thread),
4706e827ccSJim Ingham     m_stream_sp ()
4806e827ccSJim Ingham {
4906e827ccSJim Ingham }
5006e827ccSJim Ingham 
5106e827ccSJim Ingham Stream *
5206e827ccSJim Ingham ThreadPlanTracer::GetLogStream ()
5306e827ccSJim Ingham {
5406e827ccSJim Ingham 
5506e827ccSJim Ingham     if (m_stream_sp.get())
5606e827ccSJim Ingham         return m_stream_sp.get();
5706e827ccSJim Ingham     else
5806e827ccSJim Ingham         return &(m_thread.GetProcess().GetTarget().GetDebugger().GetOutputStream());
5906e827ccSJim Ingham }
6006e827ccSJim Ingham 
6106e827ccSJim Ingham void
6206e827ccSJim Ingham ThreadPlanTracer::Log()
6306e827ccSJim Ingham {
6406e827ccSJim Ingham     SymbolContext sc;
6506e827ccSJim Ingham     bool show_frame_index = false;
6606e827ccSJim Ingham     bool show_fullpaths = false;
6706e827ccSJim Ingham 
6806e827ccSJim Ingham     m_thread.GetStackFrameAtIndex(0)->Dump (GetLogStream(), show_frame_index, show_fullpaths);
6906e827ccSJim Ingham     GetLogStream()->Printf("\n");
7006e827ccSJim Ingham }
7106e827ccSJim Ingham 
7206e827ccSJim Ingham bool
7306e827ccSJim Ingham ThreadPlanTracer::TracerExplainsStop ()
7406e827ccSJim Ingham {
7506e827ccSJim Ingham     if (m_enabled && m_single_step)
7606e827ccSJim Ingham     {
7706e827ccSJim Ingham         lldb::StopInfoSP stop_info = m_thread.GetStopInfo();
7806e827ccSJim Ingham         if (stop_info->GetStopReason() == eStopReasonTrace)
7906e827ccSJim Ingham             return true;
8006e827ccSJim Ingham         else
8106e827ccSJim Ingham             return false;
8206e827ccSJim Ingham     }
8306e827ccSJim Ingham     else
8406e827ccSJim Ingham         return false;
8506e827ccSJim Ingham }
868c9e5383SSean Callanan 
878c9e5383SSean Callanan #pragma mark ThreadPlanAssemblyTracer
888c9e5383SSean Callanan 
898c9e5383SSean Callanan ThreadPlanAssemblyTracer::ThreadPlanAssemblyTracer (Thread &thread, lldb::StreamSP &stream_sp) :
908c9e5383SSean Callanan     ThreadPlanTracer (thread, stream_sp),
918c9e5383SSean Callanan     m_process(thread.GetProcess()),
928c9e5383SSean Callanan     m_target(thread.GetProcess().GetTarget())
938c9e5383SSean Callanan {
94a80ef359SJim Ingham     InitializeTracer ();
95a80ef359SJim Ingham }
96a80ef359SJim Ingham 
97a80ef359SJim Ingham ThreadPlanAssemblyTracer::ThreadPlanAssemblyTracer (Thread &thread) :
98a80ef359SJim Ingham     ThreadPlanTracer (thread),
99a80ef359SJim Ingham     m_process(thread.GetProcess()),
100a80ef359SJim Ingham     m_target(thread.GetProcess().GetTarget())
101a80ef359SJim Ingham {
102a80ef359SJim Ingham     InitializeTracer ();
103a80ef359SJim Ingham }
104a80ef359SJim Ingham 
105a80ef359SJim Ingham void
106a80ef359SJim Ingham ThreadPlanAssemblyTracer::InitializeTracer()
107a80ef359SJim Ingham {
108a80ef359SJim Ingham     Process &process = m_thread.GetProcess();
1098c9e5383SSean Callanan     Target &target = process.GetTarget();
1108c9e5383SSean Callanan 
1118c9e5383SSean Callanan     ArchSpec arch(target.GetArchitecture());
1128c9e5383SSean Callanan 
113*1080edbcSGreg Clayton     m_disassembler = Disassembler::FindPlugin(arch, NULL);
1148c9e5383SSean Callanan 
1158c9e5383SSean Callanan     m_abi = process.GetABI();
1168c9e5383SSean Callanan 
1176beaaa68SGreg Clayton     ModuleSP exe_module_sp (target.GetExecutableModule());
1188c9e5383SSean Callanan 
1196beaaa68SGreg Clayton     if (exe_module_sp)
1208c9e5383SSean Callanan     {
1216beaaa68SGreg Clayton         m_intptr_type = TypeFromUser(exe_module_sp->GetClangASTContext().GetBuiltinTypeForEncodingAndBitSize(eEncodingUint, arch.GetAddressByteSize() * 8),
1226beaaa68SGreg Clayton                                      exe_module_sp->GetClangASTContext().getASTContext());
1238c9e5383SSean Callanan     }
1248c9e5383SSean Callanan 
1258c9e5383SSean Callanan     const unsigned int buf_size = 32;
1268c9e5383SSean Callanan 
1278c9e5383SSean Callanan     m_buffer_sp.reset(new DataBufferHeap(buf_size, 0));
1288c9e5383SSean Callanan }
1298c9e5383SSean Callanan 
1308c9e5383SSean Callanan ThreadPlanAssemblyTracer::~ThreadPlanAssemblyTracer()
1318c9e5383SSean Callanan {
1328c9e5383SSean Callanan }
1338c9e5383SSean Callanan 
1345ccbd294SGreg Clayton void
1355ccbd294SGreg Clayton ThreadPlanAssemblyTracer::TracingStarted ()
1368c9e5383SSean Callanan {
1375ccbd294SGreg Clayton     RegisterContext *reg_ctx = m_thread.GetRegisterContext().get();
1388c9e5383SSean Callanan 
1398c9e5383SSean Callanan     if (m_register_values.size() == 0)
1408c9e5383SSean Callanan     {
1418c9e5383SSean Callanan         for (uint32_t reg_index = 0, num_registers = reg_ctx->GetRegisterCount();
1428c9e5383SSean Callanan              reg_index < num_registers;
1438c9e5383SSean Callanan              ++reg_index)
1448c9e5383SSean Callanan             m_register_values.push_back(0);
1458c9e5383SSean Callanan     }
1468c9e5383SSean Callanan }
1478c9e5383SSean Callanan 
1485ccbd294SGreg Clayton void
1495ccbd294SGreg Clayton ThreadPlanAssemblyTracer::TracingEnded ()
1508c9e5383SSean Callanan {
1518c9e5383SSean Callanan     for (uint32_t reg_index = 0, num_registers = m_register_values.size();
1528c9e5383SSean Callanan          reg_index < num_registers;
1538c9e5383SSean Callanan          ++reg_index)
1548c9e5383SSean Callanan         m_register_values[reg_index] = 0;
1558c9e5383SSean Callanan }
1568c9e5383SSean Callanan 
1575ccbd294SGreg Clayton static void
1585ccbd294SGreg Clayton PadOutTo (StreamString &stream, int target)
1598c9e5383SSean Callanan {
1608c9e5383SSean Callanan     stream.Flush();
1618c9e5383SSean Callanan 
1628c9e5383SSean Callanan     int length = stream.GetString().length();
1638c9e5383SSean Callanan 
1648c9e5383SSean Callanan     if (length + 1 < target)
1655ccbd294SGreg Clayton         stream.Printf("%*s", target - (length + 1) + 1, "");
1668c9e5383SSean Callanan }
1678c9e5383SSean Callanan 
1685ccbd294SGreg Clayton void
1695ccbd294SGreg Clayton ThreadPlanAssemblyTracer::Log ()
1708c9e5383SSean Callanan {
1718c9e5383SSean Callanan     Stream *stream = GetLogStream ();
1728c9e5383SSean Callanan 
1738c9e5383SSean Callanan     if (!stream)
1748c9e5383SSean Callanan         return;
1758c9e5383SSean Callanan 
1765ccbd294SGreg Clayton     RegisterContext *reg_ctx = m_thread.GetRegisterContext().get();
1778c9e5383SSean Callanan 
1788c9e5383SSean Callanan     lldb::addr_t pc = reg_ctx->GetPC();
1798c9e5383SSean Callanan     Address pc_addr;
1808c9e5383SSean Callanan     bool addr_valid = false;
1818c9e5383SSean Callanan 
1828c9e5383SSean Callanan     StreamString desc;
1838c9e5383SSean Callanan 
1848c9e5383SSean Callanan     int desired_width = 0;
1858c9e5383SSean Callanan 
1868c9e5383SSean Callanan     addr_valid = m_process.GetTarget().GetSectionLoadList().ResolveLoadAddress (pc, pc_addr);
1878c9e5383SSean Callanan 
1888c9e5383SSean Callanan     pc_addr.Dump(&desc, &m_thread, Address::DumpStyleResolvedDescription, Address::DumpStyleModuleWithFileAddress);
1898c9e5383SSean Callanan 
1908c9e5383SSean Callanan     desired_width += 64;
1918c9e5383SSean Callanan     PadOutTo(desc, desired_width);
1928c9e5383SSean Callanan 
1938c9e5383SSean Callanan     if (m_disassembler)
1948c9e5383SSean Callanan     {
19572b77ebcSGreg Clayton         ::memset(m_buffer_sp->GetBytes(), 0, m_buffer_sp->GetByteSize());
1968c9e5383SSean Callanan 
1978c9e5383SSean Callanan         Error err;
1988c9e5383SSean Callanan         m_process.ReadMemory(pc, m_buffer_sp->GetBytes(), m_buffer_sp->GetByteSize(), err);
1998c9e5383SSean Callanan 
2008c9e5383SSean Callanan         if (err.Success())
2018c9e5383SSean Callanan         {
2028c9e5383SSean Callanan             DataExtractor extractor(m_buffer_sp,
2038c9e5383SSean Callanan                                     m_process.GetByteOrder(),
2048c9e5383SSean Callanan                                     m_process.GetAddressByteSize());
2058c9e5383SSean Callanan 
2068c9e5383SSean Callanan             if (addr_valid)
20737023b06SJim Ingham                 m_disassembler->DecodeInstructions (pc_addr, extractor, 0, 1, false);
2088c9e5383SSean Callanan             else
20937023b06SJim Ingham                 m_disassembler->DecodeInstructions (Address (NULL, pc), extractor, 0, 1, false);
2108c9e5383SSean Callanan 
2118c9e5383SSean Callanan             InstructionList &instruction_list = m_disassembler->GetInstructionList();
2128c9e5383SSean Callanan 
2138c9e5383SSean Callanan             if (instruction_list.GetSize())
2148c9e5383SSean Callanan             {
2158c9e5383SSean Callanan                 Instruction *instruction = instruction_list.GetInstructionAtIndex(0).get();
2168c9e5383SSean Callanan                 instruction->Dump (&desc,
2178c9e5383SSean Callanan                                    false,
218*1080edbcSGreg Clayton                                    false,
2198c9e5383SSean Callanan                                    NULL,
2208c9e5383SSean Callanan                                    true);
2218c9e5383SSean Callanan             }
2228c9e5383SSean Callanan         }
2238c9e5383SSean Callanan 
2248c9e5383SSean Callanan         desired_width += 32;
2258c9e5383SSean Callanan         PadOutTo(desc, desired_width);
2268c9e5383SSean Callanan     }
2278c9e5383SSean Callanan 
2288c9e5383SSean Callanan     if (m_abi && m_intptr_type.GetOpaqueQualType())
2298c9e5383SSean Callanan     {
2308c9e5383SSean Callanan         ValueList value_list;
2318c9e5383SSean Callanan         const int num_args = 1;
2328c9e5383SSean Callanan 
2338c9e5383SSean Callanan         for (int arg_index = 0; arg_index < num_args; ++arg_index)
2348c9e5383SSean Callanan         {
2358c9e5383SSean Callanan             Value value;
2368c9e5383SSean Callanan             value.SetValueType (Value::eValueTypeScalar);
237526e5afbSGreg Clayton             value.SetContext (Value::eContextTypeClangType, m_intptr_type.GetOpaqueQualType());
2388c9e5383SSean Callanan             value_list.PushValue (value);
2398c9e5383SSean Callanan         }
2408c9e5383SSean Callanan 
2418c9e5383SSean Callanan         if (m_abi->GetArgumentValues (m_thread, value_list))
2428c9e5383SSean Callanan         {
2438c9e5383SSean Callanan             for (int arg_index = 0; arg_index < num_args; ++arg_index)
2448c9e5383SSean Callanan             {
2458c9e5383SSean Callanan                 desc.Printf("arg[%d]=%llx", arg_index, value_list.GetValueAtIndex(arg_index)->GetScalar().ULongLong());
2468c9e5383SSean Callanan 
2478c9e5383SSean Callanan                 if (arg_index + 1 < num_args)
2488c9e5383SSean Callanan                     desc.Printf(", ");
2498c9e5383SSean Callanan             }
2508c9e5383SSean Callanan         }
2518c9e5383SSean Callanan     }
2528c9e5383SSean Callanan 
2538c9e5383SSean Callanan     desired_width += 20;
2548c9e5383SSean Callanan     PadOutTo(desc, desired_width);
2558c9e5383SSean Callanan 
2568c9e5383SSean Callanan     for (uint32_t reg_index = 0, num_registers = reg_ctx->GetRegisterCount();
2578c9e5383SSean Callanan          reg_index < num_registers;
2588c9e5383SSean Callanan          ++reg_index)
2598c9e5383SSean Callanan     {
2608c9e5383SSean Callanan         uint64_t reg_value = reg_ctx->ReadRegisterAsUnsigned(reg_index, 0x0);
2618c9e5383SSean Callanan 
2628c9e5383SSean Callanan         if (reg_value != m_register_values[reg_index])
2638c9e5383SSean Callanan         {
2648c9e5383SSean Callanan             desc.Printf ("%s:0x%llx->0x%llx ", reg_ctx->GetRegisterName(reg_index), m_register_values[reg_index], reg_value);
2658c9e5383SSean Callanan 
2668c9e5383SSean Callanan             m_register_values[reg_index] = reg_value;
2678c9e5383SSean Callanan         }
2688c9e5383SSean Callanan     }
2698c9e5383SSean Callanan 
270978e071fSJim Ingham     stream->Printf ("Single-step: %s\n", desc.GetString().c_str());
2718c9e5383SSean Callanan }
272