130fdc8d8SChris Lattner //===-- StackFrame.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 #include "lldb/Target/StackFrame.h"
1130fdc8d8SChris Lattner 
1230fdc8d8SChris Lattner // C Includes
1330fdc8d8SChris Lattner // C++ Includes
1430fdc8d8SChris Lattner // Other libraries and framework includes
1530fdc8d8SChris Lattner // Project includes
1630fdc8d8SChris Lattner #include "lldb/Core/Module.h"
1730fdc8d8SChris Lattner #include "lldb/Core/Disassembler.h"
1830fdc8d8SChris Lattner #include "lldb/Core/Value.h"
1930fdc8d8SChris Lattner #include "lldb/Symbol/Function.h"
2030fdc8d8SChris Lattner #include "lldb/Target/ExecutionContext.h"
2130fdc8d8SChris Lattner #include "lldb/Target/Process.h"
2230fdc8d8SChris Lattner #include "lldb/Target/RegisterContext.h"
2330fdc8d8SChris Lattner #include "lldb/Target/Target.h"
2430fdc8d8SChris Lattner #include "lldb/Target/Thread.h"
2530fdc8d8SChris Lattner 
2630fdc8d8SChris Lattner using namespace lldb;
2730fdc8d8SChris Lattner using namespace lldb_private;
2830fdc8d8SChris Lattner 
2930fdc8d8SChris Lattner // The first bits in the flags are reserved for the SymbolContext::Scope bits
3030fdc8d8SChris Lattner // so we know if we have tried to look up information in our internal symbol
3130fdc8d8SChris Lattner // context (m_sc) already.
3230fdc8d8SChris Lattner #define RESOLVED_PC_SO_ADDR (uint32_t(eSymbolContextEverything + 1))
3330fdc8d8SChris Lattner #define RESOLVED_FRAME_ID   (RESOLVED_PC_SO_ADDR << 1)
3430fdc8d8SChris Lattner #define GOT_FRAME_BASE      (RESOLVED_FRAME_ID << 1)
3530fdc8d8SChris Lattner #define FRAME_IS_OBSOLETE   (GOT_FRAME_BASE << 1)
3630fdc8d8SChris Lattner #define RESOLVED_VARIABLES  (FRAME_IS_OBSOLETE << 1)
3730fdc8d8SChris Lattner 
38*1b72fcb7SGreg Clayton StackFrame::StackFrame
39*1b72fcb7SGreg Clayton (
40*1b72fcb7SGreg Clayton     lldb::user_id_t frame_idx,
41*1b72fcb7SGreg Clayton     lldb::user_id_t concrete_frame_index,
42*1b72fcb7SGreg Clayton     Thread &thread,
43*1b72fcb7SGreg Clayton     lldb::addr_t cfa,
44*1b72fcb7SGreg Clayton     uint32_t inline_height,
45*1b72fcb7SGreg Clayton     lldb::addr_t pc,
46*1b72fcb7SGreg Clayton     const SymbolContext *sc_ptr
47*1b72fcb7SGreg Clayton ) :
48*1b72fcb7SGreg Clayton     m_frame_index (frame_idx),
49*1b72fcb7SGreg Clayton     m_concrete_frame_index (concrete_frame_index),
5030fdc8d8SChris Lattner     m_thread (thread),
5130fdc8d8SChris Lattner     m_reg_context_sp (),
52*1b72fcb7SGreg Clayton     m_id (cfa, inline_height),
5330fdc8d8SChris Lattner     m_pc (NULL, pc),
5430fdc8d8SChris Lattner     m_sc (),
5530fdc8d8SChris Lattner     m_flags (),
5630fdc8d8SChris Lattner     m_frame_base (),
5730fdc8d8SChris Lattner     m_frame_base_error (),
5830fdc8d8SChris Lattner     m_variable_list_sp (),
5930fdc8d8SChris Lattner     m_value_object_list ()
6030fdc8d8SChris Lattner {
6130fdc8d8SChris Lattner     if (sc_ptr != NULL)
62*1b72fcb7SGreg Clayton     {
6330fdc8d8SChris Lattner         m_sc = *sc_ptr;
64*1b72fcb7SGreg Clayton         m_flags.Set(m_sc.GetResolvedMask ());
65*1b72fcb7SGreg Clayton     }
6630fdc8d8SChris Lattner }
6730fdc8d8SChris Lattner 
68*1b72fcb7SGreg Clayton StackFrame::StackFrame
69*1b72fcb7SGreg Clayton (
70*1b72fcb7SGreg Clayton     lldb::user_id_t frame_idx,
71*1b72fcb7SGreg Clayton     lldb::user_id_t concrete_frame_index,
72*1b72fcb7SGreg Clayton     Thread &thread,
73*1b72fcb7SGreg Clayton     const RegisterContextSP &reg_context_sp,
74*1b72fcb7SGreg Clayton     lldb::addr_t cfa,
75*1b72fcb7SGreg Clayton     uint32_t inline_height,
76*1b72fcb7SGreg Clayton     lldb::addr_t pc,
77*1b72fcb7SGreg Clayton     const SymbolContext *sc_ptr
78*1b72fcb7SGreg Clayton ) :
79*1b72fcb7SGreg Clayton     m_frame_index (frame_idx),
80*1b72fcb7SGreg Clayton     m_concrete_frame_index (concrete_frame_index),
8130fdc8d8SChris Lattner     m_thread (thread),
8230fdc8d8SChris Lattner     m_reg_context_sp (reg_context_sp),
83*1b72fcb7SGreg Clayton     m_id (cfa, inline_height),
8430fdc8d8SChris Lattner     m_pc (NULL, pc),
8530fdc8d8SChris Lattner     m_sc (),
8630fdc8d8SChris Lattner     m_flags (),
8730fdc8d8SChris Lattner     m_frame_base (),
8830fdc8d8SChris Lattner     m_frame_base_error (),
8930fdc8d8SChris Lattner     m_variable_list_sp (),
9030fdc8d8SChris Lattner     m_value_object_list ()
9130fdc8d8SChris Lattner {
9230fdc8d8SChris Lattner     if (sc_ptr != NULL)
93*1b72fcb7SGreg Clayton     {
9430fdc8d8SChris Lattner         m_sc = *sc_ptr;
95*1b72fcb7SGreg Clayton         m_flags.Set(m_sc.GetResolvedMask ());
96*1b72fcb7SGreg Clayton     }
97*1b72fcb7SGreg Clayton 
98*1b72fcb7SGreg Clayton     if (reg_context_sp && !m_sc.target_sp)
99*1b72fcb7SGreg Clayton     {
100*1b72fcb7SGreg Clayton         m_sc.target_sp = reg_context_sp->GetThread().GetProcess().GetTarget().GetSP();
101*1b72fcb7SGreg Clayton         m_flags.Set (eSymbolContextTarget);
102*1b72fcb7SGreg Clayton     }
103*1b72fcb7SGreg Clayton }
104*1b72fcb7SGreg Clayton 
105*1b72fcb7SGreg Clayton StackFrame::StackFrame
106*1b72fcb7SGreg Clayton (
107*1b72fcb7SGreg Clayton     lldb::user_id_t frame_idx,
108*1b72fcb7SGreg Clayton     lldb::user_id_t concrete_frame_index,
109*1b72fcb7SGreg Clayton     Thread &thread,
110*1b72fcb7SGreg Clayton     const RegisterContextSP &reg_context_sp,
111*1b72fcb7SGreg Clayton     lldb::addr_t cfa,
112*1b72fcb7SGreg Clayton     uint32_t inline_height,
113*1b72fcb7SGreg Clayton     const Address& pc_addr,
114*1b72fcb7SGreg Clayton     const SymbolContext *sc_ptr
115*1b72fcb7SGreg Clayton ) :
116*1b72fcb7SGreg Clayton     m_frame_index (frame_idx),
117*1b72fcb7SGreg Clayton     m_concrete_frame_index (concrete_frame_index),
118*1b72fcb7SGreg Clayton     m_thread (thread),
119*1b72fcb7SGreg Clayton     m_reg_context_sp (reg_context_sp),
120*1b72fcb7SGreg Clayton     m_id (cfa, inline_height),
121*1b72fcb7SGreg Clayton     m_pc (pc_addr),
122*1b72fcb7SGreg Clayton     m_sc (),
123*1b72fcb7SGreg Clayton     m_flags (),
124*1b72fcb7SGreg Clayton     m_frame_base (),
125*1b72fcb7SGreg Clayton     m_frame_base_error (),
126*1b72fcb7SGreg Clayton     m_variable_list_sp (),
127*1b72fcb7SGreg Clayton     m_value_object_list ()
128*1b72fcb7SGreg Clayton {
129*1b72fcb7SGreg Clayton     if (sc_ptr != NULL)
130*1b72fcb7SGreg Clayton     {
131*1b72fcb7SGreg Clayton         m_sc = *sc_ptr;
132*1b72fcb7SGreg Clayton         m_flags.Set(m_sc.GetResolvedMask ());
133*1b72fcb7SGreg Clayton     }
134*1b72fcb7SGreg Clayton 
135*1b72fcb7SGreg Clayton     if (m_sc.target_sp.get() == NULL && reg_context_sp)
136*1b72fcb7SGreg Clayton     {
137*1b72fcb7SGreg Clayton         m_sc.target_sp = reg_context_sp->GetThread().GetProcess().GetTarget().GetSP();
138*1b72fcb7SGreg Clayton         m_flags.Set (eSymbolContextTarget);
139*1b72fcb7SGreg Clayton     }
140*1b72fcb7SGreg Clayton 
141*1b72fcb7SGreg Clayton     if (m_sc.module_sp.get() == NULL && pc_addr.GetSection())
142*1b72fcb7SGreg Clayton     {
143*1b72fcb7SGreg Clayton         Module *pc_module = pc_addr.GetSection()->GetModule();
144*1b72fcb7SGreg Clayton         if (pc_module)
145*1b72fcb7SGreg Clayton         {
146*1b72fcb7SGreg Clayton             m_sc.module_sp = pc_module->GetSP();
147*1b72fcb7SGreg Clayton             m_flags.Set (eSymbolContextModule);
148*1b72fcb7SGreg Clayton         }
149*1b72fcb7SGreg Clayton     }
15030fdc8d8SChris Lattner }
15130fdc8d8SChris Lattner 
15230fdc8d8SChris Lattner 
15330fdc8d8SChris Lattner //----------------------------------------------------------------------
15430fdc8d8SChris Lattner // Destructor
15530fdc8d8SChris Lattner //----------------------------------------------------------------------
15630fdc8d8SChris Lattner StackFrame::~StackFrame()
15730fdc8d8SChris Lattner {
15830fdc8d8SChris Lattner }
15930fdc8d8SChris Lattner 
16030fdc8d8SChris Lattner StackID&
16130fdc8d8SChris Lattner StackFrame::GetStackID()
16230fdc8d8SChris Lattner {
16330fdc8d8SChris Lattner     // Make sure we have resolved our stack ID's address range before we give
16430fdc8d8SChris Lattner     // it out to any external clients
16530fdc8d8SChris Lattner     if (m_id.GetStartAddress().IsValid() == 0 && m_flags.IsClear(RESOLVED_FRAME_ID))
16630fdc8d8SChris Lattner     {
16730fdc8d8SChris Lattner         m_flags.Set (RESOLVED_FRAME_ID);
16830fdc8d8SChris Lattner 
16930fdc8d8SChris Lattner         // Resolve our PC to section offset if we haven't alreday done so
17030fdc8d8SChris Lattner         // and if we don't have a module. The resolved address section will
17130fdc8d8SChris Lattner         // contain the module to which it belongs.
17230fdc8d8SChris Lattner         if (!m_sc.module_sp && m_flags.IsClear(RESOLVED_PC_SO_ADDR))
17330fdc8d8SChris Lattner             GetPC();
17430fdc8d8SChris Lattner 
17530fdc8d8SChris Lattner         const uint32_t resolve_scope = eSymbolContextModule |
17630fdc8d8SChris Lattner                                        eSymbolContextCompUnit |
17730fdc8d8SChris Lattner                                        eSymbolContextFunction;
17830fdc8d8SChris Lattner 
17930fdc8d8SChris Lattner         if (m_sc.module_sp)
18030fdc8d8SChris Lattner         {
18130fdc8d8SChris Lattner             if (m_sc.module_sp->ResolveSymbolContextForAddress (GetPC(), resolve_scope, m_sc) & eSymbolContextFunction)
18230fdc8d8SChris Lattner             {
18330fdc8d8SChris Lattner                 assert (m_sc.function);
18430fdc8d8SChris Lattner                 m_id.SetStartAddress(m_sc.function->GetAddressRange().GetBaseAddress());
18530fdc8d8SChris Lattner             }
18630fdc8d8SChris Lattner             else if (m_sc.module_sp->ResolveSymbolContextForAddress (GetPC(), resolve_scope, m_sc) & eSymbolContextSymbol)
18730fdc8d8SChris Lattner             {
18830fdc8d8SChris Lattner                 assert (m_sc.symbol);
18930fdc8d8SChris Lattner                 AddressRange *symbol_range_ptr = m_sc.symbol->GetAddressRangePtr();
19030fdc8d8SChris Lattner                 if (symbol_range_ptr)
19130fdc8d8SChris Lattner                     m_id.SetStartAddress(symbol_range_ptr->GetBaseAddress());
19230fdc8d8SChris Lattner             }
19330fdc8d8SChris Lattner         }
19430fdc8d8SChris Lattner //      else if (m_sc.target != NULL)
19530fdc8d8SChris Lattner //      {
19630fdc8d8SChris Lattner //          if (m_sc.target->GetImages().ResolveSymbolContextForAddress (GetPC(), resolve_scope, m_sc) & eSymbolContextFunction)
19730fdc8d8SChris Lattner //          {
19830fdc8d8SChris Lattner //              assert (m_sc.function);
19930fdc8d8SChris Lattner //              m_id.GetAddressRange() = m_sc.function->GetAddressRange();
20030fdc8d8SChris Lattner //          }
20130fdc8d8SChris Lattner //          else if (m_sc.target->GetImages().ResolveSymbolContextForAddress (GetPC(), resolve_scope, m_sc) & eSymbolContextSymbol)
20230fdc8d8SChris Lattner //          {
20330fdc8d8SChris Lattner //              assert (m_sc.symbol);
20430fdc8d8SChris Lattner //              AddressRange *symbol_range_ptr = m_sc.symbol->GetAddressRange();
20530fdc8d8SChris Lattner //              if (symbol_range_ptr)
20630fdc8d8SChris Lattner //                  m_id.GetAddressRange() = *symbol_range_ptr;
20730fdc8d8SChris Lattner //          }
20830fdc8d8SChris Lattner //      }
20930fdc8d8SChris Lattner     }
21030fdc8d8SChris Lattner     return m_id;
21130fdc8d8SChris Lattner }
21230fdc8d8SChris Lattner 
21330fdc8d8SChris Lattner Address&
21430fdc8d8SChris Lattner StackFrame::GetPC()
21530fdc8d8SChris Lattner {
21630fdc8d8SChris Lattner     if (m_flags.IsClear(RESOLVED_PC_SO_ADDR) && !m_pc.IsSectionOffset())
21730fdc8d8SChris Lattner     {
21830fdc8d8SChris Lattner         m_flags.Set (RESOLVED_PC_SO_ADDR);
21930fdc8d8SChris Lattner 
22030fdc8d8SChris Lattner         // Resolve the PC into a temporary address because if ResolveLoadAddress
22130fdc8d8SChris Lattner         // fails to resolve the address, it will clear the address object...
22230fdc8d8SChris Lattner         Address resolved_pc;
22330fdc8d8SChris Lattner         if (m_thread.GetProcess().ResolveLoadAddress(m_pc.GetOffset(), resolved_pc))
22430fdc8d8SChris Lattner         {
22530fdc8d8SChris Lattner             m_pc = resolved_pc;
22630fdc8d8SChris Lattner             const Section *section = m_pc.GetSection();
22730fdc8d8SChris Lattner             if (section)
22830fdc8d8SChris Lattner             {
22930fdc8d8SChris Lattner                 Module *module = section->GetModule();
23030fdc8d8SChris Lattner                 if (module)
23130fdc8d8SChris Lattner                 {
23230fdc8d8SChris Lattner                     m_sc.module_sp = module->GetSP();
23330fdc8d8SChris Lattner                     if (m_sc.module_sp)
23430fdc8d8SChris Lattner                         m_flags.Set(eSymbolContextModule);
23530fdc8d8SChris Lattner                 }
23630fdc8d8SChris Lattner             }
23730fdc8d8SChris Lattner         }
23830fdc8d8SChris Lattner     }
23930fdc8d8SChris Lattner     return m_pc;
24030fdc8d8SChris Lattner }
24130fdc8d8SChris Lattner 
24230fdc8d8SChris Lattner void
24330fdc8d8SChris Lattner StackFrame::ChangePC (addr_t pc)
24430fdc8d8SChris Lattner {
24530fdc8d8SChris Lattner     m_pc.SetOffset(pc);
24630fdc8d8SChris Lattner     m_pc.SetSection(NULL);
24730fdc8d8SChris Lattner     m_sc.Clear();
24830fdc8d8SChris Lattner     m_flags.SetAllFlagBits(0);
24930fdc8d8SChris Lattner     m_thread.ClearStackFrames ();
25030fdc8d8SChris Lattner }
25130fdc8d8SChris Lattner 
25230fdc8d8SChris Lattner const char *
25330fdc8d8SChris Lattner StackFrame::Disassemble ()
25430fdc8d8SChris Lattner {
25530fdc8d8SChris Lattner     if (m_disassembly.GetSize() == 0)
25630fdc8d8SChris Lattner     {
25730fdc8d8SChris Lattner         ExecutionContext exe_ctx;
25830fdc8d8SChris Lattner         Calculate(exe_ctx);
2596611103cSGreg Clayton         Target &target = m_thread.GetProcess().GetTarget();
2606611103cSGreg Clayton         Disassembler::Disassemble (target.GetDebugger(),
2616611103cSGreg Clayton                                    target.GetArchitecture(),
26230fdc8d8SChris Lattner                                    exe_ctx,
26330fdc8d8SChris Lattner                                    0,
264dda4f7b5SGreg Clayton                                    false,
26530fdc8d8SChris Lattner                                    m_disassembly);
26630fdc8d8SChris Lattner         if (m_disassembly.GetSize() == 0)
26730fdc8d8SChris Lattner             return NULL;
26830fdc8d8SChris Lattner     }
26930fdc8d8SChris Lattner     return m_disassembly.GetData();
27030fdc8d8SChris Lattner }
27130fdc8d8SChris Lattner 
27230fdc8d8SChris Lattner //----------------------------------------------------------------------
27330fdc8d8SChris Lattner // Get the symbol context if we already haven't done so by resolving the
27430fdc8d8SChris Lattner // PC address as much as possible. This way when we pass around a
27530fdc8d8SChris Lattner // StackFrame object, everyone will have as much information as
27630fdc8d8SChris Lattner // possible and no one will ever have to look things up manually.
27730fdc8d8SChris Lattner //----------------------------------------------------------------------
27830fdc8d8SChris Lattner const SymbolContext&
27930fdc8d8SChris Lattner StackFrame::GetSymbolContext (uint32_t resolve_scope)
28030fdc8d8SChris Lattner {
28130fdc8d8SChris Lattner     // Copy our internal symbol context into "sc".
28230fdc8d8SChris Lattner 
28330fdc8d8SChris Lattner     if ((m_flags.GetAllFlagBits() & resolve_scope) != resolve_scope)
28430fdc8d8SChris Lattner     {
28530fdc8d8SChris Lattner         // Resolve our PC to section offset if we haven't alreday done so
28630fdc8d8SChris Lattner         // and if we don't have a module. The resolved address section will
28730fdc8d8SChris Lattner         // contain the module to which it belongs
28830fdc8d8SChris Lattner         if (!m_sc.module_sp && m_flags.IsClear(RESOLVED_PC_SO_ADDR))
28930fdc8d8SChris Lattner             GetPC();
29030fdc8d8SChris Lattner 
29130fdc8d8SChris Lattner         // If this is not frame zero, then we need to subtract 1 from the PC
29230fdc8d8SChris Lattner         // value when doing address lookups since the PC will be on the
29330fdc8d8SChris Lattner         // instruction following the function call instruction...
29430fdc8d8SChris Lattner 
29530fdc8d8SChris Lattner         Address lookup_addr(GetPC());
296*1b72fcb7SGreg Clayton         if (m_frame_index > 0 && lookup_addr.IsValid())
29730fdc8d8SChris Lattner         {
29830fdc8d8SChris Lattner             addr_t offset = lookup_addr.GetOffset();
29930fdc8d8SChris Lattner             if (offset > 0)
30030fdc8d8SChris Lattner                 lookup_addr.SetOffset(offset - 1);
30130fdc8d8SChris Lattner         }
30230fdc8d8SChris Lattner 
30330fdc8d8SChris Lattner         if (m_sc.module_sp)
30430fdc8d8SChris Lattner         {
30530fdc8d8SChris Lattner             // We have something in our stack frame symbol context, lets check
30630fdc8d8SChris Lattner             // if we haven't already tried to lookup one of those things. If we
30730fdc8d8SChris Lattner             // haven't then we will do the query.
308*1b72fcb7SGreg Clayton 
309*1b72fcb7SGreg Clayton             uint32_t actual_resolve_scope = 0;
310*1b72fcb7SGreg Clayton 
311*1b72fcb7SGreg Clayton             if (resolve_scope & eSymbolContextCompUnit)
312*1b72fcb7SGreg Clayton             {
313*1b72fcb7SGreg Clayton                 if (m_flags.IsClear (eSymbolContextCompUnit))
314*1b72fcb7SGreg Clayton                 {
315*1b72fcb7SGreg Clayton                     if (m_sc.comp_unit)
316*1b72fcb7SGreg Clayton                         m_flags.Set (eSymbolContextCompUnit);
317*1b72fcb7SGreg Clayton                     else
318*1b72fcb7SGreg Clayton                         actual_resolve_scope |= eSymbolContextCompUnit;
319*1b72fcb7SGreg Clayton                 }
320*1b72fcb7SGreg Clayton             }
321*1b72fcb7SGreg Clayton 
322*1b72fcb7SGreg Clayton             if (resolve_scope & eSymbolContextFunction)
323*1b72fcb7SGreg Clayton             {
324*1b72fcb7SGreg Clayton                 if (m_flags.IsClear (eSymbolContextFunction))
325*1b72fcb7SGreg Clayton                 {
326*1b72fcb7SGreg Clayton                     if (m_sc.function)
327*1b72fcb7SGreg Clayton                         m_flags.Set (eSymbolContextFunction);
328*1b72fcb7SGreg Clayton                     else
329*1b72fcb7SGreg Clayton                         actual_resolve_scope |= eSymbolContextFunction;
330*1b72fcb7SGreg Clayton                 }
331*1b72fcb7SGreg Clayton             }
332*1b72fcb7SGreg Clayton 
333*1b72fcb7SGreg Clayton             if (resolve_scope & eSymbolContextBlock)
334*1b72fcb7SGreg Clayton             {
335*1b72fcb7SGreg Clayton                 if (m_flags.IsClear (eSymbolContextBlock))
336*1b72fcb7SGreg Clayton                 {
337*1b72fcb7SGreg Clayton                     if (m_sc.block)
338*1b72fcb7SGreg Clayton                         m_flags.Set (eSymbolContextBlock);
339*1b72fcb7SGreg Clayton                     else
340*1b72fcb7SGreg Clayton                         actual_resolve_scope |= eSymbolContextBlock;
341*1b72fcb7SGreg Clayton                 }
342*1b72fcb7SGreg Clayton             }
343*1b72fcb7SGreg Clayton 
344*1b72fcb7SGreg Clayton             if (resolve_scope & eSymbolContextSymbol)
345*1b72fcb7SGreg Clayton             {
346*1b72fcb7SGreg Clayton                 if (m_flags.IsClear (eSymbolContextSymbol))
347*1b72fcb7SGreg Clayton                 {
348*1b72fcb7SGreg Clayton                     if (m_sc.symbol)
349*1b72fcb7SGreg Clayton                         m_flags.Set (eSymbolContextSymbol);
350*1b72fcb7SGreg Clayton                     else
351*1b72fcb7SGreg Clayton                         actual_resolve_scope |= eSymbolContextSymbol;
352*1b72fcb7SGreg Clayton                 }
353*1b72fcb7SGreg Clayton             }
354*1b72fcb7SGreg Clayton 
355*1b72fcb7SGreg Clayton             if (resolve_scope & eSymbolContextLineEntry)
356*1b72fcb7SGreg Clayton             {
357*1b72fcb7SGreg Clayton                 if (m_flags.IsClear (eSymbolContextLineEntry))
358*1b72fcb7SGreg Clayton                 {
359*1b72fcb7SGreg Clayton                     if (m_sc.line_entry.IsValid())
360*1b72fcb7SGreg Clayton                         m_flags.Set (eSymbolContextLineEntry);
361*1b72fcb7SGreg Clayton                     else
362*1b72fcb7SGreg Clayton                         actual_resolve_scope |= eSymbolContextLineEntry;
363*1b72fcb7SGreg Clayton                 }
364*1b72fcb7SGreg Clayton             }
365*1b72fcb7SGreg Clayton 
366*1b72fcb7SGreg Clayton             if (actual_resolve_scope)
36730fdc8d8SChris Lattner             {
36830fdc8d8SChris Lattner                 // We might be resolving less information than what is already
36930fdc8d8SChris Lattner                 // in our current symbol context so resolve into a temporary
37030fdc8d8SChris Lattner                 // symbol context "sc" so we don't clear out data we have
37130fdc8d8SChris Lattner                 // already found in "m_sc"
37230fdc8d8SChris Lattner                 SymbolContext sc;
37330fdc8d8SChris Lattner                 // Set flags that indicate what we have tried to resolve
374*1b72fcb7SGreg Clayton                 const uint32_t resolved = m_sc.module_sp->ResolveSymbolContextForAddress (lookup_addr, actual_resolve_scope, sc);
375*1b72fcb7SGreg Clayton                 // Only replace what we didn't already have as we may have
376*1b72fcb7SGreg Clayton                 // information for an inlined function scope that won't match
377*1b72fcb7SGreg Clayton                 // what a standard lookup by address would match
37830fdc8d8SChris Lattner                 if (resolved & eSymbolContextCompUnit)  m_sc.comp_unit  = sc.comp_unit;
37930fdc8d8SChris Lattner                 if (resolved & eSymbolContextFunction)  m_sc.function   = sc.function;
38030fdc8d8SChris Lattner                 if (resolved & eSymbolContextBlock)     m_sc.block      = sc.block;
38130fdc8d8SChris Lattner                 if (resolved & eSymbolContextSymbol)    m_sc.symbol     = sc.symbol;
38230fdc8d8SChris Lattner                 if (resolved & eSymbolContextLineEntry) m_sc.line_entry = sc.line_entry;
38330fdc8d8SChris Lattner             }
38430fdc8d8SChris Lattner         }
38530fdc8d8SChris Lattner         else
38630fdc8d8SChris Lattner         {
38730fdc8d8SChris Lattner             // If we don't have a module, then we can't have the compile unit,
38830fdc8d8SChris Lattner             // function, block, line entry or symbol, so we can safely call
38930fdc8d8SChris Lattner             // ResolveSymbolContextForAddress with our symbol context member m_sc.
39030fdc8d8SChris Lattner             m_thread.GetProcess().GetTarget().GetImages().ResolveSymbolContextForAddress (lookup_addr, resolve_scope, m_sc);
39130fdc8d8SChris Lattner         }
39230fdc8d8SChris Lattner 
39330fdc8d8SChris Lattner         // If the target was requested add that:
39430fdc8d8SChris Lattner         if (m_sc.target_sp.get() == NULL)
39530fdc8d8SChris Lattner             m_sc.target_sp = CalculateProcess()->GetTarget().GetSP();
39630fdc8d8SChris Lattner 
39730fdc8d8SChris Lattner         // Update our internal flags so we remember what we have tried to locate so
39830fdc8d8SChris Lattner         // we don't have to keep trying when more calls to this function are made.
39930fdc8d8SChris Lattner         m_flags.Set(resolve_scope);
40030fdc8d8SChris Lattner     }
40130fdc8d8SChris Lattner 
40230fdc8d8SChris Lattner     // Return the symbol context with everything that was possible to resolve
40330fdc8d8SChris Lattner     // resolved.
40430fdc8d8SChris Lattner     return m_sc;
40530fdc8d8SChris Lattner }
40630fdc8d8SChris Lattner 
40730fdc8d8SChris Lattner 
40830fdc8d8SChris Lattner VariableList *
40930fdc8d8SChris Lattner StackFrame::GetVariableList ()
41030fdc8d8SChris Lattner {
41130fdc8d8SChris Lattner     if (m_flags.IsClear(RESOLVED_VARIABLES))
41230fdc8d8SChris Lattner     {
41330fdc8d8SChris Lattner         m_flags.Set(RESOLVED_VARIABLES);
41430fdc8d8SChris Lattner 
41530fdc8d8SChris Lattner         GetSymbolContext(eSymbolContextFunction);
41630fdc8d8SChris Lattner         if (m_sc.function)
41730fdc8d8SChris Lattner         {
41830fdc8d8SChris Lattner             bool get_child_variables = true;
41930fdc8d8SChris Lattner             bool can_create = true;
4200b76a2c2SGreg Clayton             m_variable_list_sp = m_sc.function->GetBlock (can_create).GetVariableList (get_child_variables, can_create);
42130fdc8d8SChris Lattner         }
42230fdc8d8SChris Lattner     }
42330fdc8d8SChris Lattner     return m_variable_list_sp.get();
42430fdc8d8SChris Lattner }
42530fdc8d8SChris Lattner 
42630fdc8d8SChris Lattner 
42730fdc8d8SChris Lattner bool
42830fdc8d8SChris Lattner StackFrame::GetFrameBaseValue (Scalar &frame_base, Error *error_ptr)
42930fdc8d8SChris Lattner {
43030fdc8d8SChris Lattner     if (m_flags.IsClear(GOT_FRAME_BASE))
43130fdc8d8SChris Lattner     {
43230fdc8d8SChris Lattner         if (m_sc.function)
43330fdc8d8SChris Lattner         {
43430fdc8d8SChris Lattner             m_frame_base.Clear();
43530fdc8d8SChris Lattner             m_frame_base_error.Clear();
43630fdc8d8SChris Lattner 
43730fdc8d8SChris Lattner             m_flags.Set(GOT_FRAME_BASE);
43830fdc8d8SChris Lattner             ExecutionContext exe_ctx (&m_thread.GetProcess(), &m_thread, this);
43930fdc8d8SChris Lattner             Value expr_value;
44030fdc8d8SChris Lattner             if (m_sc.function->GetFrameBaseExpression().Evaluate(&exe_ctx, NULL, NULL, expr_value, &m_frame_base_error) < 0)
44130fdc8d8SChris Lattner             {
44230fdc8d8SChris Lattner                 // We should really have an error if evaluate returns, but in case
44330fdc8d8SChris Lattner                 // we don't, lets set the error to something at least.
44430fdc8d8SChris Lattner                 if (m_frame_base_error.Success())
44530fdc8d8SChris Lattner                     m_frame_base_error.SetErrorString("Evaluation of the frame base expression failed.");
44630fdc8d8SChris Lattner             }
44730fdc8d8SChris Lattner             else
44830fdc8d8SChris Lattner             {
44930fdc8d8SChris Lattner                 m_frame_base = expr_value.ResolveValue(&exe_ctx, NULL);
45030fdc8d8SChris Lattner             }
45130fdc8d8SChris Lattner         }
45230fdc8d8SChris Lattner         else
45330fdc8d8SChris Lattner         {
45430fdc8d8SChris Lattner             m_frame_base_error.SetErrorString ("No function in symbol context.");
45530fdc8d8SChris Lattner         }
45630fdc8d8SChris Lattner     }
45730fdc8d8SChris Lattner 
45830fdc8d8SChris Lattner     if (m_frame_base_error.Success())
45930fdc8d8SChris Lattner         frame_base = m_frame_base;
46030fdc8d8SChris Lattner 
46130fdc8d8SChris Lattner     if (error_ptr)
46230fdc8d8SChris Lattner         *error_ptr = m_frame_base_error;
46330fdc8d8SChris Lattner     return m_frame_base_error.Success();
46430fdc8d8SChris Lattner }
46530fdc8d8SChris Lattner 
46630fdc8d8SChris Lattner RegisterContext *
46730fdc8d8SChris Lattner StackFrame::GetRegisterContext ()
46830fdc8d8SChris Lattner {
46930fdc8d8SChris Lattner     if (m_reg_context_sp.get() == NULL)
47030fdc8d8SChris Lattner         m_reg_context_sp.reset (m_thread.CreateRegisterContextForFrame (this));
47130fdc8d8SChris Lattner     return m_reg_context_sp.get();
47230fdc8d8SChris Lattner }
47330fdc8d8SChris Lattner 
47430fdc8d8SChris Lattner bool
47530fdc8d8SChris Lattner StackFrame::HasDebugInformation ()
47630fdc8d8SChris Lattner {
47730fdc8d8SChris Lattner     GetSymbolContext(eSymbolContextLineEntry);
47830fdc8d8SChris Lattner     return m_sc.line_entry.IsValid();
47930fdc8d8SChris Lattner }
48030fdc8d8SChris Lattner 
48130fdc8d8SChris Lattner ValueObjectList &
48230fdc8d8SChris Lattner StackFrame::GetValueObjectList()
48330fdc8d8SChris Lattner {
48430fdc8d8SChris Lattner     return m_value_object_list;
48530fdc8d8SChris Lattner }
48630fdc8d8SChris Lattner 
48730fdc8d8SChris Lattner 
48830fdc8d8SChris Lattner Target *
48930fdc8d8SChris Lattner StackFrame::CalculateTarget ()
49030fdc8d8SChris Lattner {
49130fdc8d8SChris Lattner     return m_thread.CalculateTarget();
49230fdc8d8SChris Lattner }
49330fdc8d8SChris Lattner 
49430fdc8d8SChris Lattner Process *
49530fdc8d8SChris Lattner StackFrame::CalculateProcess ()
49630fdc8d8SChris Lattner {
49730fdc8d8SChris Lattner     return m_thread.CalculateProcess();
49830fdc8d8SChris Lattner }
49930fdc8d8SChris Lattner 
50030fdc8d8SChris Lattner Thread *
50130fdc8d8SChris Lattner StackFrame::CalculateThread ()
50230fdc8d8SChris Lattner {
50330fdc8d8SChris Lattner     return &m_thread;
50430fdc8d8SChris Lattner }
50530fdc8d8SChris Lattner 
50630fdc8d8SChris Lattner StackFrame *
50730fdc8d8SChris Lattner StackFrame::CalculateStackFrame ()
50830fdc8d8SChris Lattner {
50930fdc8d8SChris Lattner     return this;
51030fdc8d8SChris Lattner }
51130fdc8d8SChris Lattner 
51230fdc8d8SChris Lattner 
51330fdc8d8SChris Lattner void
51430fdc8d8SChris Lattner StackFrame::Calculate (ExecutionContext &exe_ctx)
51530fdc8d8SChris Lattner {
51630fdc8d8SChris Lattner     m_thread.Calculate (exe_ctx);
51730fdc8d8SChris Lattner     exe_ctx.frame = this;
51830fdc8d8SChris Lattner }
51930fdc8d8SChris Lattner 
52030fdc8d8SChris Lattner void
52130fdc8d8SChris Lattner StackFrame::Dump (Stream *strm, bool show_frame_index)
52230fdc8d8SChris Lattner {
52330fdc8d8SChris Lattner     if (strm == NULL)
52430fdc8d8SChris Lattner         return;
52530fdc8d8SChris Lattner 
52630fdc8d8SChris Lattner     if (show_frame_index)
527*1b72fcb7SGreg Clayton         strm->Printf("frame #%u: ", m_frame_index);
52830fdc8d8SChris Lattner     strm->Printf("pc = 0x%0*llx", m_thread.GetProcess().GetAddressByteSize() * 2, GetRegisterContext()->GetPC());
52930fdc8d8SChris Lattner     SymbolContext sc (GetSymbolContext(eSymbolContextEverything));
53030fdc8d8SChris Lattner     strm->PutCString(", where = ");
531*1b72fcb7SGreg Clayton     // TODO: need to get the
532*1b72fcb7SGreg Clayton     const bool show_module = true;
533*1b72fcb7SGreg Clayton     const bool show_inline = true;
534*1b72fcb7SGreg Clayton     sc.DumpStopContext(strm, &m_thread.GetProcess(), GetPC(), show_module, show_inline);
53530fdc8d8SChris Lattner }
53630fdc8d8SChris Lattner 
537