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 
1093a64300SDaniel Malea #include "lldb/lldb-python.h"
1193a64300SDaniel Malea 
1206e827ccSJim Ingham #include "lldb/Target/ThreadPlan.h"
1306e827ccSJim Ingham 
1406e827ccSJim Ingham // C Includes
158c9e5383SSean Callanan #include <string.h>
1606e827ccSJim Ingham // C++ Includes
1706e827ccSJim Ingham // Other libraries and framework includes
1806e827ccSJim Ingham // Project includes
198c9e5383SSean Callanan #include "lldb/Core/ArchSpec.h"
208c9e5383SSean Callanan #include "lldb/Core/DataBufferHeap.h"
21*a78bd7ffSZachary Turner #include "lldb/Core/DataExtractor.h"
2206e827ccSJim Ingham #include "lldb/Core/Debugger.h"
238c9e5383SSean Callanan #include "lldb/Core/Disassembler.h"
2406e827ccSJim Ingham #include "lldb/Core/Log.h"
251f746071SGreg Clayton #include "lldb/Core/Module.h"
2606e827ccSJim Ingham #include "lldb/Core/State.h"
2744d93782SGreg Clayton #include "lldb/Core/StreamFile.h"
288c9e5383SSean Callanan #include "lldb/Core/Value.h"
29*a78bd7ffSZachary Turner #include "lldb/Symbol/ClangASTContext.h"
308c9e5383SSean Callanan #include "lldb/Symbol/TypeList.h"
3132abc6edSZachary Turner #include "lldb/Target/ABI.h"
3206e827ccSJim Ingham #include "lldb/Target/RegisterContext.h"
3306e827ccSJim Ingham #include "lldb/Target/Thread.h"
3406e827ccSJim Ingham #include "lldb/Target/Process.h"
35d5944cd1SGreg Clayton #include "lldb/Target/SectionLoadList.h"
3606e827ccSJim Ingham #include "lldb/Target/Target.h"
3706e827ccSJim Ingham 
3806e827ccSJim Ingham using namespace lldb;
3906e827ccSJim Ingham using namespace lldb_private;
4006e827ccSJim Ingham 
418c9e5383SSean Callanan #pragma mark ThreadPlanTracer
428c9e5383SSean Callanan 
4306e827ccSJim Ingham ThreadPlanTracer::ThreadPlanTracer (Thread &thread, lldb::StreamSP &stream_sp) :
4471c21d18SStephen Wilson     m_thread (thread),
4506e827ccSJim Ingham     m_single_step(true),
4606e827ccSJim Ingham     m_enabled (false),
4706e827ccSJim Ingham     m_stream_sp (stream_sp)
4806e827ccSJim Ingham {
4906e827ccSJim Ingham }
5006e827ccSJim Ingham 
5106e827ccSJim Ingham ThreadPlanTracer::ThreadPlanTracer (Thread &thread) :
5271c21d18SStephen Wilson     m_thread (thread),
5306e827ccSJim Ingham     m_single_step(true),
5406e827ccSJim Ingham     m_enabled (false),
5506e827ccSJim Ingham     m_stream_sp ()
5606e827ccSJim Ingham {
5706e827ccSJim Ingham }
5806e827ccSJim Ingham 
5906e827ccSJim Ingham Stream *
6006e827ccSJim Ingham ThreadPlanTracer::GetLogStream ()
6106e827ccSJim Ingham {
6206e827ccSJim Ingham 
6306e827ccSJim Ingham     if (m_stream_sp.get())
6406e827ccSJim Ingham         return m_stream_sp.get();
6506e827ccSJim Ingham     else
661ac04c30SGreg Clayton     {
671ac04c30SGreg Clayton         TargetSP target_sp (m_thread.CalculateTarget());
681ac04c30SGreg Clayton         if (target_sp)
6944d93782SGreg Clayton             return target_sp->GetDebugger().GetOutputFile().get();
701ac04c30SGreg Clayton     }
711ac04c30SGreg Clayton     return NULL;
7206e827ccSJim Ingham }
7306e827ccSJim Ingham 
7406e827ccSJim Ingham void
7506e827ccSJim Ingham ThreadPlanTracer::Log()
7606e827ccSJim Ingham {
7706e827ccSJim Ingham     SymbolContext sc;
7806e827ccSJim Ingham     bool show_frame_index = false;
7906e827ccSJim Ingham     bool show_fullpaths = false;
8006e827ccSJim Ingham 
8115356e7fSCaroline Tice     Stream *stream = GetLogStream();
8228eb5711SJim Ingham     if (stream)
8328eb5711SJim Ingham     {
8415356e7fSCaroline Tice         m_thread.GetStackFrameAtIndex(0)->Dump (stream, show_frame_index, show_fullpaths);
8515356e7fSCaroline Tice         stream->Printf("\n");
8615356e7fSCaroline Tice         stream->Flush();
8728eb5711SJim Ingham     }
8815356e7fSCaroline Tice 
8906e827ccSJim Ingham }
9006e827ccSJim Ingham 
9106e827ccSJim Ingham bool
9206e827ccSJim Ingham ThreadPlanTracer::TracerExplainsStop ()
9306e827ccSJim Ingham {
9406e827ccSJim Ingham     if (m_enabled && m_single_step)
9506e827ccSJim Ingham     {
9606e827ccSJim Ingham         lldb::StopInfoSP stop_info = m_thread.GetStopInfo();
9706e827ccSJim Ingham         if (stop_info->GetStopReason() == eStopReasonTrace)
9806e827ccSJim Ingham             return true;
9906e827ccSJim Ingham         else
10006e827ccSJim Ingham             return false;
10106e827ccSJim Ingham     }
10206e827ccSJim Ingham     else
10306e827ccSJim Ingham         return false;
10406e827ccSJim Ingham }
1058c9e5383SSean Callanan 
1068c9e5383SSean Callanan #pragma mark ThreadPlanAssemblyTracer
1078c9e5383SSean Callanan 
1088c9e5383SSean Callanan ThreadPlanAssemblyTracer::ThreadPlanAssemblyTracer (Thread &thread, lldb::StreamSP &stream_sp) :
1098c9e5383SSean Callanan     ThreadPlanTracer (thread, stream_sp),
1107e6d4e5aSSean Callanan     m_disassembler_sp (),
1117e9b1fd0SGreg Clayton     m_intptr_type (),
1127e9b1fd0SGreg Clayton     m_register_values ()
1138c9e5383SSean Callanan {
114a80ef359SJim Ingham }
115a80ef359SJim Ingham 
116a80ef359SJim Ingham ThreadPlanAssemblyTracer::ThreadPlanAssemblyTracer (Thread &thread) :
117a80ef359SJim Ingham     ThreadPlanTracer (thread),
1187e6d4e5aSSean Callanan     m_disassembler_sp (),
1197e9b1fd0SGreg Clayton     m_intptr_type (),
1207e9b1fd0SGreg Clayton     m_register_values ()
121a80ef359SJim Ingham {
122a80ef359SJim Ingham }
123a80ef359SJim Ingham 
1247e9b1fd0SGreg Clayton Disassembler *
1257e9b1fd0SGreg Clayton ThreadPlanAssemblyTracer::GetDisassembler ()
126a80ef359SJim Ingham {
1277e6d4e5aSSean Callanan     if (m_disassembler_sp.get() == NULL)
1280f063ba6SJim Ingham         m_disassembler_sp = Disassembler::FindPlugin(m_thread.GetProcess()->GetTarget().GetArchitecture(), NULL, NULL);
1297e6d4e5aSSean Callanan     return m_disassembler_sp.get();
1307e9b1fd0SGreg Clayton }
1318c9e5383SSean Callanan 
1327e9b1fd0SGreg Clayton TypeFromUser
1337e9b1fd0SGreg Clayton ThreadPlanAssemblyTracer::GetIntPointerType()
1347e9b1fd0SGreg Clayton {
1357e9b1fd0SGreg Clayton     if (!m_intptr_type.IsValid ())
1367e9b1fd0SGreg Clayton     {
1371ac04c30SGreg Clayton         TargetSP target_sp (m_thread.CalculateTarget());
1381ac04c30SGreg Clayton         if (target_sp)
1391ac04c30SGreg Clayton         {
1401ac04c30SGreg Clayton             Module *exe_module = target_sp->GetExecutableModulePointer();
1418c9e5383SSean Callanan 
142aa149cbdSGreg Clayton             if (exe_module)
1438c9e5383SSean Callanan             {
14457ee3067SGreg Clayton                 m_intptr_type = TypeFromUser(exe_module->GetClangASTContext().GetBuiltinTypeForEncodingAndBitSize(eEncodingUint, target_sp->GetArchitecture().GetAddressByteSize() * 8));
1458c9e5383SSean Callanan             }
1468c9e5383SSean Callanan         }
1471ac04c30SGreg Clayton     }
1487e9b1fd0SGreg Clayton     return m_intptr_type;
1497e9b1fd0SGreg Clayton }
1507e9b1fd0SGreg Clayton 
1517e9b1fd0SGreg Clayton 
1528c9e5383SSean Callanan 
1538c9e5383SSean Callanan ThreadPlanAssemblyTracer::~ThreadPlanAssemblyTracer()
1548c9e5383SSean Callanan {
1558c9e5383SSean Callanan }
1568c9e5383SSean Callanan 
1575ccbd294SGreg Clayton void
1585ccbd294SGreg Clayton ThreadPlanAssemblyTracer::TracingStarted ()
1598c9e5383SSean Callanan {
1605ccbd294SGreg Clayton     RegisterContext *reg_ctx = m_thread.GetRegisterContext().get();
1618c9e5383SSean Callanan 
1628c9e5383SSean Callanan     if (m_register_values.size() == 0)
163cd482e35SGreg Clayton         m_register_values.resize (reg_ctx->GetRegisterCount());
1648c9e5383SSean Callanan }
1658c9e5383SSean Callanan 
1665ccbd294SGreg Clayton void
1675ccbd294SGreg Clayton ThreadPlanAssemblyTracer::TracingEnded ()
1688c9e5383SSean Callanan {
169cd482e35SGreg Clayton     m_register_values.clear();
1708c9e5383SSean Callanan }
1718c9e5383SSean Callanan 
1725ccbd294SGreg Clayton void
1735ccbd294SGreg Clayton ThreadPlanAssemblyTracer::Log ()
1748c9e5383SSean Callanan {
1758c9e5383SSean Callanan     Stream *stream = GetLogStream ();
1768c9e5383SSean Callanan 
1778c9e5383SSean Callanan     if (!stream)
1788c9e5383SSean Callanan         return;
1798c9e5383SSean Callanan 
1805ccbd294SGreg Clayton     RegisterContext *reg_ctx = m_thread.GetRegisterContext().get();
1818c9e5383SSean Callanan 
1828c9e5383SSean Callanan     lldb::addr_t pc = reg_ctx->GetPC();
1831ac04c30SGreg Clayton     ProcessSP process_sp (m_thread.GetProcess());
1848c9e5383SSean Callanan     Address pc_addr;
1858c9e5383SSean Callanan     bool addr_valid = false;
1867e9b1fd0SGreg Clayton     uint8_t buffer[16] = {0}; // Must be big enough for any single instruction
1871ac04c30SGreg Clayton     addr_valid = process_sp->GetTarget().GetSectionLoadList().ResolveLoadAddress (pc, pc_addr);
1888c9e5383SSean Callanan 
189cd482e35SGreg Clayton     pc_addr.Dump(stream, &m_thread, Address::DumpStyleResolvedDescription, Address::DumpStyleModuleWithFileAddress);
1906cffdd2fSJim Ingham     stream->PutCString (" ");
1918c9e5383SSean Callanan 
1927e9b1fd0SGreg Clayton     Disassembler *disassembler = GetDisassembler();
1937e9b1fd0SGreg Clayton     if (disassembler)
1948c9e5383SSean Callanan     {
1958c9e5383SSean Callanan         Error err;
1961ac04c30SGreg Clayton         process_sp->ReadMemory(pc, buffer, sizeof(buffer), err);
1978c9e5383SSean Callanan 
1988c9e5383SSean Callanan         if (err.Success())
1998c9e5383SSean Callanan         {
2007e9b1fd0SGreg Clayton             DataExtractor extractor(buffer, sizeof(buffer),
2011ac04c30SGreg Clayton                                     process_sp->GetByteOrder(),
2021ac04c30SGreg Clayton                                     process_sp->GetAddressByteSize());
2038c9e5383SSean Callanan 
2043faf47c4SGreg Clayton 			bool data_from_file = false;
2058c9e5383SSean Callanan             if (addr_valid)
2063faf47c4SGreg Clayton                 disassembler->DecodeInstructions (pc_addr, extractor, 0, 1, false, data_from_file);
2078c9e5383SSean Callanan             else
2083faf47c4SGreg Clayton                 disassembler->DecodeInstructions (Address (pc), extractor, 0, 1, false, data_from_file);
2098c9e5383SSean Callanan 
2107e9b1fd0SGreg Clayton             InstructionList &instruction_list = disassembler->GetInstructionList();
211357132ebSGreg Clayton             const uint32_t max_opcode_byte_size = instruction_list.GetMaxOpcocdeByteSize();
2128c9e5383SSean Callanan 
2138c9e5383SSean Callanan             if (instruction_list.GetSize())
2148c9e5383SSean Callanan             {
215cd482e35SGreg Clayton                 const bool show_bytes = true;
216cd482e35SGreg Clayton                 const bool show_address = true;
2178c9e5383SSean Callanan                 Instruction *instruction = instruction_list.GetInstructionAtIndex(0).get();
218554f68d3SGreg Clayton                 const FormatEntity::Entry *disassemble_format = m_thread.GetProcess()->GetTarget().GetDebugger().GetDisassemblyFormat();
219cd482e35SGreg Clayton                 instruction->Dump (stream,
220357132ebSGreg Clayton                                    max_opcode_byte_size,
221cd482e35SGreg Clayton                                    show_address,
222cd482e35SGreg Clayton                                    show_bytes,
223aff1b357SJason Molenda                                    NULL,
224aff1b357SJason Molenda                                    NULL,
225aff1b357SJason Molenda                                    NULL,
226c980fa92SJason Molenda                                    disassemble_format,
227c980fa92SJason Molenda                                    0);
2288c9e5383SSean Callanan             }
2298c9e5383SSean Callanan         }
2308c9e5383SSean Callanan     }
2318c9e5383SSean Callanan 
2321ac04c30SGreg Clayton     const ABI *abi = process_sp->GetABI().get();
2337e9b1fd0SGreg Clayton     TypeFromUser intptr_type = GetIntPointerType();
2347e9b1fd0SGreg Clayton 
2357e9b1fd0SGreg Clayton     if (abi && intptr_type.IsValid())
2368c9e5383SSean Callanan     {
2378c9e5383SSean Callanan         ValueList value_list;
2388c9e5383SSean Callanan         const int num_args = 1;
2398c9e5383SSean Callanan 
2408c9e5383SSean Callanan         for (int arg_index = 0; arg_index < num_args; ++arg_index)
2418c9e5383SSean Callanan         {
2428c9e5383SSean Callanan             Value value;
2438c9e5383SSean Callanan             value.SetValueType (Value::eValueTypeScalar);
24457ee3067SGreg Clayton //            value.SetContext (Value::eContextTypeClangType, intptr_type.GetOpaqueQualType());
24557ee3067SGreg Clayton             value.SetClangType (intptr_type);
2468c9e5383SSean Callanan             value_list.PushValue (value);
2478c9e5383SSean Callanan         }
2488c9e5383SSean Callanan 
2497e9b1fd0SGreg Clayton         if (abi->GetArgumentValues (m_thread, value_list))
2508c9e5383SSean Callanan         {
2518c9e5383SSean Callanan             for (int arg_index = 0; arg_index < num_args; ++arg_index)
2528c9e5383SSean Callanan             {
253cd482e35SGreg Clayton                 stream->Printf("\n\targ[%d]=%llx", arg_index, value_list.GetValueAtIndex(arg_index)->GetScalar().ULongLong());
2548c9e5383SSean Callanan 
2558c9e5383SSean Callanan                 if (arg_index + 1 < num_args)
256cd482e35SGreg Clayton                     stream->PutCString (", ");
2578c9e5383SSean Callanan             }
2588c9e5383SSean Callanan         }
2598c9e5383SSean Callanan     }
2608c9e5383SSean Callanan 
2618c9e5383SSean Callanan 
262cd482e35SGreg Clayton     RegisterValue reg_value;
263cd482e35SGreg Clayton     for (uint32_t reg_num = 0, num_registers = reg_ctx->GetRegisterCount();
264cd482e35SGreg Clayton          reg_num < num_registers;
265cd482e35SGreg Clayton          ++reg_num)
2668c9e5383SSean Callanan     {
267cd482e35SGreg Clayton         const RegisterInfo *reg_info = reg_ctx->GetRegisterInfoAtIndex(reg_num);
268cd482e35SGreg Clayton         if (reg_ctx->ReadRegister (reg_info, reg_value))
2698c9e5383SSean Callanan         {
270cd482e35SGreg Clayton             assert (reg_num < m_register_values.size());
271cd482e35SGreg Clayton             if (m_register_values[reg_num].GetType() == RegisterValue::eTypeInvalid ||
272cd482e35SGreg Clayton                 reg_value != m_register_values[reg_num])
273cd482e35SGreg Clayton             {
274cd482e35SGreg Clayton                 if (reg_value.GetType() != RegisterValue::eTypeInvalid)
275cd482e35SGreg Clayton                 {
276cd482e35SGreg Clayton                     stream->PutCString ("\n\t");
277cd482e35SGreg Clayton                     reg_value.Dump(stream, reg_info, true, false, eFormatDefault);
2788c9e5383SSean Callanan                 }
2798c9e5383SSean Callanan             }
280cd482e35SGreg Clayton             m_register_values[reg_num] = reg_value;
281cd482e35SGreg Clayton         }
282cd482e35SGreg Clayton     }
283cd482e35SGreg Clayton     stream->EOL();
28415356e7fSCaroline Tice     stream->Flush();
2858c9e5383SSean Callanan }
286