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" 19*288bdf9cSGreg Clayton #include "lldb/Core/ValueObjectVariable.h" 2030fdc8d8SChris Lattner #include "lldb/Symbol/Function.h" 21*288bdf9cSGreg Clayton #include "lldb/Symbol/VariableList.h" 2230fdc8d8SChris Lattner #include "lldb/Target/ExecutionContext.h" 2330fdc8d8SChris Lattner #include "lldb/Target/Process.h" 2430fdc8d8SChris Lattner #include "lldb/Target/RegisterContext.h" 2530fdc8d8SChris Lattner #include "lldb/Target/Target.h" 2630fdc8d8SChris Lattner #include "lldb/Target/Thread.h" 2730fdc8d8SChris Lattner 2830fdc8d8SChris Lattner using namespace lldb; 2930fdc8d8SChris Lattner using namespace lldb_private; 3030fdc8d8SChris Lattner 3130fdc8d8SChris Lattner // The first bits in the flags are reserved for the SymbolContext::Scope bits 3230fdc8d8SChris Lattner // so we know if we have tried to look up information in our internal symbol 3330fdc8d8SChris Lattner // context (m_sc) already. 3459e8fc1cSGreg Clayton #define RESOLVED_FRAME_CODE_ADDR (uint32_t(eSymbolContextEverything + 1)) 3559e8fc1cSGreg Clayton #define RESOLVED_FRAME_ID_START_ADDR (RESOLVED_FRAME_CODE_ADDR << 1) 3659e8fc1cSGreg Clayton #define RESOLVED_FRAME_ID_SYMBOL_SCOPE (RESOLVED_FRAME_ID_START_ADDR << 1) 3759e8fc1cSGreg Clayton #define GOT_FRAME_BASE (RESOLVED_FRAME_ID_SYMBOL_SCOPE << 1) 3859e8fc1cSGreg Clayton #define RESOLVED_VARIABLES (GOT_FRAME_BASE << 1) 3930fdc8d8SChris Lattner 401b72fcb7SGreg Clayton StackFrame::StackFrame 411b72fcb7SGreg Clayton ( 421b72fcb7SGreg Clayton lldb::user_id_t frame_idx, 4359e8fc1cSGreg Clayton lldb::user_id_t unwind_frame_index, 441b72fcb7SGreg Clayton Thread &thread, 451b72fcb7SGreg Clayton lldb::addr_t cfa, 461b72fcb7SGreg Clayton lldb::addr_t pc, 471b72fcb7SGreg Clayton const SymbolContext *sc_ptr 481b72fcb7SGreg Clayton ) : 491b72fcb7SGreg Clayton m_frame_index (frame_idx), 5059e8fc1cSGreg Clayton m_unwind_frame_index (unwind_frame_index), 5130fdc8d8SChris Lattner m_thread (thread), 5230fdc8d8SChris Lattner m_reg_context_sp (), 5359e8fc1cSGreg Clayton m_id (LLDB_INVALID_ADDRESS, cfa, NULL), 5412fc3e0fSGreg Clayton m_frame_code_addr (NULL, pc), 5530fdc8d8SChris Lattner m_sc (), 5630fdc8d8SChris Lattner m_flags (), 5730fdc8d8SChris Lattner m_frame_base (), 5830fdc8d8SChris Lattner m_frame_base_error (), 5930fdc8d8SChris Lattner m_variable_list_sp (), 60*288bdf9cSGreg Clayton m_variable_list_value_objects () 6130fdc8d8SChris Lattner { 6230fdc8d8SChris Lattner if (sc_ptr != NULL) 631b72fcb7SGreg Clayton { 6430fdc8d8SChris Lattner m_sc = *sc_ptr; 651b72fcb7SGreg Clayton m_flags.Set(m_sc.GetResolvedMask ()); 661b72fcb7SGreg Clayton } 6730fdc8d8SChris Lattner } 6830fdc8d8SChris Lattner 691b72fcb7SGreg Clayton StackFrame::StackFrame 701b72fcb7SGreg Clayton ( 711b72fcb7SGreg Clayton lldb::user_id_t frame_idx, 7259e8fc1cSGreg Clayton lldb::user_id_t unwind_frame_index, 731b72fcb7SGreg Clayton Thread &thread, 741b72fcb7SGreg Clayton const RegisterContextSP ®_context_sp, 751b72fcb7SGreg Clayton lldb::addr_t cfa, 761b72fcb7SGreg Clayton lldb::addr_t pc, 771b72fcb7SGreg Clayton const SymbolContext *sc_ptr 781b72fcb7SGreg Clayton ) : 791b72fcb7SGreg Clayton m_frame_index (frame_idx), 8059e8fc1cSGreg Clayton m_unwind_frame_index (unwind_frame_index), 8130fdc8d8SChris Lattner m_thread (thread), 8230fdc8d8SChris Lattner m_reg_context_sp (reg_context_sp), 8359e8fc1cSGreg Clayton m_id (LLDB_INVALID_ADDRESS, cfa, NULL), 8412fc3e0fSGreg Clayton m_frame_code_addr (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 (), 90*288bdf9cSGreg Clayton m_variable_list_value_objects () 9130fdc8d8SChris Lattner { 9230fdc8d8SChris Lattner if (sc_ptr != NULL) 931b72fcb7SGreg Clayton { 9430fdc8d8SChris Lattner m_sc = *sc_ptr; 951b72fcb7SGreg Clayton m_flags.Set(m_sc.GetResolvedMask ()); 961b72fcb7SGreg Clayton } 971b72fcb7SGreg Clayton 981b72fcb7SGreg Clayton if (reg_context_sp && !m_sc.target_sp) 991b72fcb7SGreg Clayton { 1001b72fcb7SGreg Clayton m_sc.target_sp = reg_context_sp->GetThread().GetProcess().GetTarget().GetSP(); 1011b72fcb7SGreg Clayton m_flags.Set (eSymbolContextTarget); 1021b72fcb7SGreg Clayton } 1031b72fcb7SGreg Clayton } 1041b72fcb7SGreg Clayton 1051b72fcb7SGreg Clayton StackFrame::StackFrame 1061b72fcb7SGreg Clayton ( 1071b72fcb7SGreg Clayton lldb::user_id_t frame_idx, 10859e8fc1cSGreg Clayton lldb::user_id_t unwind_frame_index, 1091b72fcb7SGreg Clayton Thread &thread, 1101b72fcb7SGreg Clayton const RegisterContextSP ®_context_sp, 1111b72fcb7SGreg Clayton lldb::addr_t cfa, 1121b72fcb7SGreg Clayton const Address& pc_addr, 1131b72fcb7SGreg Clayton const SymbolContext *sc_ptr 1141b72fcb7SGreg Clayton ) : 1151b72fcb7SGreg Clayton m_frame_index (frame_idx), 11659e8fc1cSGreg Clayton m_unwind_frame_index (unwind_frame_index), 1171b72fcb7SGreg Clayton m_thread (thread), 1181b72fcb7SGreg Clayton m_reg_context_sp (reg_context_sp), 11959e8fc1cSGreg Clayton m_id (LLDB_INVALID_ADDRESS, cfa, NULL), 12012fc3e0fSGreg Clayton m_frame_code_addr (pc_addr), 1211b72fcb7SGreg Clayton m_sc (), 1221b72fcb7SGreg Clayton m_flags (), 1231b72fcb7SGreg Clayton m_frame_base (), 1241b72fcb7SGreg Clayton m_frame_base_error (), 1251b72fcb7SGreg Clayton m_variable_list_sp (), 126*288bdf9cSGreg Clayton m_variable_list_value_objects () 1271b72fcb7SGreg Clayton { 1281b72fcb7SGreg Clayton if (sc_ptr != NULL) 1291b72fcb7SGreg Clayton { 1301b72fcb7SGreg Clayton m_sc = *sc_ptr; 1311b72fcb7SGreg Clayton m_flags.Set(m_sc.GetResolvedMask ()); 1321b72fcb7SGreg Clayton } 1331b72fcb7SGreg Clayton 1341b72fcb7SGreg Clayton if (m_sc.target_sp.get() == NULL && reg_context_sp) 1351b72fcb7SGreg Clayton { 1361b72fcb7SGreg Clayton m_sc.target_sp = reg_context_sp->GetThread().GetProcess().GetTarget().GetSP(); 1371b72fcb7SGreg Clayton m_flags.Set (eSymbolContextTarget); 1381b72fcb7SGreg Clayton } 1391b72fcb7SGreg Clayton 1401b72fcb7SGreg Clayton if (m_sc.module_sp.get() == NULL && pc_addr.GetSection()) 1411b72fcb7SGreg Clayton { 1421b72fcb7SGreg Clayton Module *pc_module = pc_addr.GetSection()->GetModule(); 1431b72fcb7SGreg Clayton if (pc_module) 1441b72fcb7SGreg Clayton { 1451b72fcb7SGreg Clayton m_sc.module_sp = pc_module->GetSP(); 1461b72fcb7SGreg Clayton m_flags.Set (eSymbolContextModule); 1471b72fcb7SGreg Clayton } 1481b72fcb7SGreg Clayton } 14930fdc8d8SChris Lattner } 15030fdc8d8SChris Lattner 15130fdc8d8SChris Lattner 15230fdc8d8SChris Lattner //---------------------------------------------------------------------- 15330fdc8d8SChris Lattner // Destructor 15430fdc8d8SChris Lattner //---------------------------------------------------------------------- 15530fdc8d8SChris Lattner StackFrame::~StackFrame() 15630fdc8d8SChris Lattner { 15730fdc8d8SChris Lattner } 15830fdc8d8SChris Lattner 15930fdc8d8SChris Lattner StackID& 16030fdc8d8SChris Lattner StackFrame::GetStackID() 16130fdc8d8SChris Lattner { 16212fc3e0fSGreg Clayton // Make sure we have resolved our stack ID's start PC before we give 16312fc3e0fSGreg Clayton // it out to any external clients. This allows us to not have to lookup 16412fc3e0fSGreg Clayton // this information if it is never asked for. 16559e8fc1cSGreg Clayton if (m_flags.IsClear(RESOLVED_FRAME_ID_START_ADDR)) 16630fdc8d8SChris Lattner { 16759e8fc1cSGreg Clayton m_flags.Set (RESOLVED_FRAME_ID_START_ADDR); 16830fdc8d8SChris Lattner 16959e8fc1cSGreg Clayton if (m_id.GetStartAddress() == LLDB_INVALID_ADDRESS) 17059e8fc1cSGreg Clayton { 17130fdc8d8SChris Lattner // Resolve our PC to section offset if we haven't alreday done so 17230fdc8d8SChris Lattner // and if we don't have a module. The resolved address section will 17330fdc8d8SChris Lattner // contain the module to which it belongs. 17459e8fc1cSGreg Clayton if (!m_sc.module_sp && m_flags.IsClear(RESOLVED_FRAME_CODE_ADDR)) 1759da7bd07SGreg Clayton GetFrameCodeAddress(); 17630fdc8d8SChris Lattner 1779da7bd07SGreg Clayton if (GetSymbolContext (eSymbolContextFunction).function) 17830fdc8d8SChris Lattner { 17912fc3e0fSGreg Clayton m_id.SetStartAddress (m_sc.function->GetAddressRange().GetBaseAddress().GetLoadAddress (&m_thread.GetProcess())); 18030fdc8d8SChris Lattner } 1819da7bd07SGreg Clayton else if (GetSymbolContext (eSymbolContextSymbol).symbol) 18230fdc8d8SChris Lattner { 18330fdc8d8SChris Lattner AddressRange *symbol_range_ptr = m_sc.symbol->GetAddressRangePtr(); 18430fdc8d8SChris Lattner if (symbol_range_ptr) 18512fc3e0fSGreg Clayton m_id.SetStartAddress(symbol_range_ptr->GetBaseAddress().GetLoadAddress (&m_thread.GetProcess())); 18630fdc8d8SChris Lattner } 18712fc3e0fSGreg Clayton 18812fc3e0fSGreg Clayton // We didn't find a function or symbol, just use the frame code address 18912fc3e0fSGreg Clayton // which will be the same as the PC in the frame. 19012fc3e0fSGreg Clayton if (m_id.GetStartAddress() == LLDB_INVALID_ADDRESS) 19112fc3e0fSGreg Clayton m_id.SetStartAddress (m_frame_code_addr.GetLoadAddress (&m_thread.GetProcess())); 19230fdc8d8SChris Lattner } 19359e8fc1cSGreg Clayton } 19459e8fc1cSGreg Clayton 19559e8fc1cSGreg Clayton if (m_flags.IsClear (RESOLVED_FRAME_ID_SYMBOL_SCOPE)) 19659e8fc1cSGreg Clayton { 19759e8fc1cSGreg Clayton if (m_id.GetSymbolContextScope ()) 19859e8fc1cSGreg Clayton { 19959e8fc1cSGreg Clayton m_flags.Set (RESOLVED_FRAME_ID_SYMBOL_SCOPE); 20059e8fc1cSGreg Clayton } 20159e8fc1cSGreg Clayton else 20259e8fc1cSGreg Clayton { 20359e8fc1cSGreg Clayton GetSymbolContext (eSymbolContextFunction | eSymbolContextBlock); 20459e8fc1cSGreg Clayton 20559e8fc1cSGreg Clayton if (m_sc.block) 20659e8fc1cSGreg Clayton { 20759e8fc1cSGreg Clayton Block *inline_block = m_sc.block->GetContainingInlinedBlock(); 20859e8fc1cSGreg Clayton if (inline_block) 20959e8fc1cSGreg Clayton { 21059e8fc1cSGreg Clayton // Use the block with the inlined function info 21159e8fc1cSGreg Clayton // as the symbol context since we want this frame 21259e8fc1cSGreg Clayton // to have only the variables for the inlined function 21359e8fc1cSGreg Clayton SetSymbolContextScope (inline_block); 21459e8fc1cSGreg Clayton } 21559e8fc1cSGreg Clayton else 21659e8fc1cSGreg Clayton { 21759e8fc1cSGreg Clayton // This block is not inlined with means it has no 21859e8fc1cSGreg Clayton // inlined parents either, so we want to use the top 21959e8fc1cSGreg Clayton // most function block. 22059e8fc1cSGreg Clayton SetSymbolContextScope (&m_sc.function->GetBlock(false)); 22159e8fc1cSGreg Clayton } 22259e8fc1cSGreg Clayton } 22359e8fc1cSGreg Clayton else 22459e8fc1cSGreg Clayton { 22559e8fc1cSGreg Clayton // The current stack frame doesn't have a block. Check to see 22659e8fc1cSGreg Clayton // if it has a symbol. If it does we will use this as the 22759e8fc1cSGreg Clayton // symbol scope. It is ok if "m_sc.symbol" is NULL below as 22859e8fc1cSGreg Clayton // it will set the symbol context to NULL and set the 22959e8fc1cSGreg Clayton // RESOLVED_FRAME_ID_SYMBOL_SCOPE flag bit. 23059e8fc1cSGreg Clayton GetSymbolContext (eSymbolContextSymbol); 23159e8fc1cSGreg Clayton SetSymbolContextScope (m_sc.symbol); 23259e8fc1cSGreg Clayton } 23359e8fc1cSGreg Clayton } 23459e8fc1cSGreg Clayton } 23530fdc8d8SChris Lattner return m_id; 23630fdc8d8SChris Lattner } 23730fdc8d8SChris Lattner 23859e8fc1cSGreg Clayton void 23959e8fc1cSGreg Clayton StackFrame::SetSymbolContextScope (SymbolContextScope *symbol_scope) 24059e8fc1cSGreg Clayton { 24159e8fc1cSGreg Clayton m_flags.Set (RESOLVED_FRAME_ID_SYMBOL_SCOPE); 24259e8fc1cSGreg Clayton m_id.SetSymbolContextScope (symbol_scope); 24359e8fc1cSGreg Clayton } 24459e8fc1cSGreg Clayton 24530fdc8d8SChris Lattner Address& 2469da7bd07SGreg Clayton StackFrame::GetFrameCodeAddress() 24730fdc8d8SChris Lattner { 24859e8fc1cSGreg Clayton if (m_flags.IsClear(RESOLVED_FRAME_CODE_ADDR) && !m_frame_code_addr.IsSectionOffset()) 24930fdc8d8SChris Lattner { 25059e8fc1cSGreg Clayton m_flags.Set (RESOLVED_FRAME_CODE_ADDR); 25130fdc8d8SChris Lattner 25230fdc8d8SChris Lattner // Resolve the PC into a temporary address because if ResolveLoadAddress 25330fdc8d8SChris Lattner // fails to resolve the address, it will clear the address object... 25430fdc8d8SChris Lattner Address resolved_pc; 25512fc3e0fSGreg Clayton if (m_thread.GetProcess().ResolveLoadAddress(m_frame_code_addr.GetOffset(), resolved_pc)) 25630fdc8d8SChris Lattner { 25712fc3e0fSGreg Clayton m_frame_code_addr = resolved_pc; 25812fc3e0fSGreg Clayton const Section *section = m_frame_code_addr.GetSection(); 25930fdc8d8SChris Lattner if (section) 26030fdc8d8SChris Lattner { 26130fdc8d8SChris Lattner Module *module = section->GetModule(); 26230fdc8d8SChris Lattner if (module) 26330fdc8d8SChris Lattner { 26430fdc8d8SChris Lattner m_sc.module_sp = module->GetSP(); 26530fdc8d8SChris Lattner if (m_sc.module_sp) 26630fdc8d8SChris Lattner m_flags.Set(eSymbolContextModule); 26730fdc8d8SChris Lattner } 26830fdc8d8SChris Lattner } 26930fdc8d8SChris Lattner } 27030fdc8d8SChris Lattner } 27112fc3e0fSGreg Clayton return m_frame_code_addr; 27230fdc8d8SChris Lattner } 27330fdc8d8SChris Lattner 27430fdc8d8SChris Lattner void 27530fdc8d8SChris Lattner StackFrame::ChangePC (addr_t pc) 27630fdc8d8SChris Lattner { 27712fc3e0fSGreg Clayton m_frame_code_addr.SetOffset(pc); 27812fc3e0fSGreg Clayton m_frame_code_addr.SetSection(NULL); 27930fdc8d8SChris Lattner m_sc.Clear(); 28030fdc8d8SChris Lattner m_flags.SetAllFlagBits(0); 28130fdc8d8SChris Lattner m_thread.ClearStackFrames (); 28230fdc8d8SChris Lattner } 28330fdc8d8SChris Lattner 28430fdc8d8SChris Lattner const char * 28530fdc8d8SChris Lattner StackFrame::Disassemble () 28630fdc8d8SChris Lattner { 28730fdc8d8SChris Lattner if (m_disassembly.GetSize() == 0) 28830fdc8d8SChris Lattner { 28930fdc8d8SChris Lattner ExecutionContext exe_ctx; 29030fdc8d8SChris Lattner Calculate(exe_ctx); 2916611103cSGreg Clayton Target &target = m_thread.GetProcess().GetTarget(); 2926611103cSGreg Clayton Disassembler::Disassemble (target.GetDebugger(), 2936611103cSGreg Clayton target.GetArchitecture(), 29430fdc8d8SChris Lattner exe_ctx, 29530fdc8d8SChris Lattner 0, 296dda4f7b5SGreg Clayton false, 29730fdc8d8SChris Lattner m_disassembly); 29830fdc8d8SChris Lattner if (m_disassembly.GetSize() == 0) 29930fdc8d8SChris Lattner return NULL; 30030fdc8d8SChris Lattner } 30130fdc8d8SChris Lattner return m_disassembly.GetData(); 30230fdc8d8SChris Lattner } 30330fdc8d8SChris Lattner 30430fdc8d8SChris Lattner //---------------------------------------------------------------------- 30530fdc8d8SChris Lattner // Get the symbol context if we already haven't done so by resolving the 30630fdc8d8SChris Lattner // PC address as much as possible. This way when we pass around a 30730fdc8d8SChris Lattner // StackFrame object, everyone will have as much information as 30830fdc8d8SChris Lattner // possible and no one will ever have to look things up manually. 30930fdc8d8SChris Lattner //---------------------------------------------------------------------- 31030fdc8d8SChris Lattner const SymbolContext& 31130fdc8d8SChris Lattner StackFrame::GetSymbolContext (uint32_t resolve_scope) 31230fdc8d8SChris Lattner { 31330fdc8d8SChris Lattner // Copy our internal symbol context into "sc". 31430fdc8d8SChris Lattner if ((m_flags.GetAllFlagBits() & resolve_scope) != resolve_scope) 31530fdc8d8SChris Lattner { 31630fdc8d8SChris Lattner // Resolve our PC to section offset if we haven't alreday done so 31730fdc8d8SChris Lattner // and if we don't have a module. The resolved address section will 31830fdc8d8SChris Lattner // contain the module to which it belongs 31959e8fc1cSGreg Clayton if (!m_sc.module_sp && m_flags.IsClear(RESOLVED_FRAME_CODE_ADDR)) 3209da7bd07SGreg Clayton GetFrameCodeAddress(); 32130fdc8d8SChris Lattner 32230fdc8d8SChris Lattner // If this is not frame zero, then we need to subtract 1 from the PC 32330fdc8d8SChris Lattner // value when doing address lookups since the PC will be on the 32430fdc8d8SChris Lattner // instruction following the function call instruction... 32530fdc8d8SChris Lattner 3269da7bd07SGreg Clayton Address lookup_addr(GetFrameCodeAddress()); 3271b72fcb7SGreg Clayton if (m_frame_index > 0 && lookup_addr.IsValid()) 32830fdc8d8SChris Lattner { 32930fdc8d8SChris Lattner addr_t offset = lookup_addr.GetOffset(); 33030fdc8d8SChris Lattner if (offset > 0) 33130fdc8d8SChris Lattner lookup_addr.SetOffset(offset - 1); 33230fdc8d8SChris Lattner } 33330fdc8d8SChris Lattner 3349da7bd07SGreg Clayton 3359da7bd07SGreg Clayton uint32_t resolved = 0; 33630fdc8d8SChris Lattner if (m_sc.module_sp) 33730fdc8d8SChris Lattner { 33830fdc8d8SChris Lattner // We have something in our stack frame symbol context, lets check 33930fdc8d8SChris Lattner // if we haven't already tried to lookup one of those things. If we 34030fdc8d8SChris Lattner // haven't then we will do the query. 3411b72fcb7SGreg Clayton 3421b72fcb7SGreg Clayton uint32_t actual_resolve_scope = 0; 3431b72fcb7SGreg Clayton 3441b72fcb7SGreg Clayton if (resolve_scope & eSymbolContextCompUnit) 3451b72fcb7SGreg Clayton { 3461b72fcb7SGreg Clayton if (m_flags.IsClear (eSymbolContextCompUnit)) 3471b72fcb7SGreg Clayton { 3481b72fcb7SGreg Clayton if (m_sc.comp_unit) 3499da7bd07SGreg Clayton resolved |= eSymbolContextCompUnit; 3501b72fcb7SGreg Clayton else 3511b72fcb7SGreg Clayton actual_resolve_scope |= eSymbolContextCompUnit; 3521b72fcb7SGreg Clayton } 3531b72fcb7SGreg Clayton } 3541b72fcb7SGreg Clayton 3551b72fcb7SGreg Clayton if (resolve_scope & eSymbolContextFunction) 3561b72fcb7SGreg Clayton { 3571b72fcb7SGreg Clayton if (m_flags.IsClear (eSymbolContextFunction)) 3581b72fcb7SGreg Clayton { 3591b72fcb7SGreg Clayton if (m_sc.function) 3609da7bd07SGreg Clayton resolved |= eSymbolContextFunction; 3611b72fcb7SGreg Clayton else 3621b72fcb7SGreg Clayton actual_resolve_scope |= eSymbolContextFunction; 3631b72fcb7SGreg Clayton } 3641b72fcb7SGreg Clayton } 3651b72fcb7SGreg Clayton 3661b72fcb7SGreg Clayton if (resolve_scope & eSymbolContextBlock) 3671b72fcb7SGreg Clayton { 3681b72fcb7SGreg Clayton if (m_flags.IsClear (eSymbolContextBlock)) 3691b72fcb7SGreg Clayton { 3701b72fcb7SGreg Clayton if (m_sc.block) 3719da7bd07SGreg Clayton resolved |= eSymbolContextBlock; 3721b72fcb7SGreg Clayton else 3731b72fcb7SGreg Clayton actual_resolve_scope |= eSymbolContextBlock; 3741b72fcb7SGreg Clayton } 3751b72fcb7SGreg Clayton } 3761b72fcb7SGreg Clayton 3771b72fcb7SGreg Clayton if (resolve_scope & eSymbolContextSymbol) 3781b72fcb7SGreg Clayton { 3791b72fcb7SGreg Clayton if (m_flags.IsClear (eSymbolContextSymbol)) 3801b72fcb7SGreg Clayton { 3811b72fcb7SGreg Clayton if (m_sc.symbol) 3829da7bd07SGreg Clayton resolved |= eSymbolContextSymbol; 3831b72fcb7SGreg Clayton else 3841b72fcb7SGreg Clayton actual_resolve_scope |= eSymbolContextSymbol; 3851b72fcb7SGreg Clayton } 3861b72fcb7SGreg Clayton } 3871b72fcb7SGreg Clayton 3881b72fcb7SGreg Clayton if (resolve_scope & eSymbolContextLineEntry) 3891b72fcb7SGreg Clayton { 3901b72fcb7SGreg Clayton if (m_flags.IsClear (eSymbolContextLineEntry)) 3911b72fcb7SGreg Clayton { 3921b72fcb7SGreg Clayton if (m_sc.line_entry.IsValid()) 3939da7bd07SGreg Clayton resolved |= eSymbolContextLineEntry; 3941b72fcb7SGreg Clayton else 3951b72fcb7SGreg Clayton actual_resolve_scope |= eSymbolContextLineEntry; 3961b72fcb7SGreg Clayton } 3971b72fcb7SGreg Clayton } 3981b72fcb7SGreg Clayton 3991b72fcb7SGreg Clayton if (actual_resolve_scope) 40030fdc8d8SChris Lattner { 40130fdc8d8SChris Lattner // We might be resolving less information than what is already 40230fdc8d8SChris Lattner // in our current symbol context so resolve into a temporary 40330fdc8d8SChris Lattner // symbol context "sc" so we don't clear out data we have 40430fdc8d8SChris Lattner // already found in "m_sc" 40530fdc8d8SChris Lattner SymbolContext sc; 40630fdc8d8SChris Lattner // Set flags that indicate what we have tried to resolve 4079da7bd07SGreg Clayton resolved |= m_sc.module_sp->ResolveSymbolContextForAddress (lookup_addr, actual_resolve_scope, sc); 4081b72fcb7SGreg Clayton // Only replace what we didn't already have as we may have 4091b72fcb7SGreg Clayton // information for an inlined function scope that won't match 4101b72fcb7SGreg Clayton // what a standard lookup by address would match 4119da7bd07SGreg Clayton if ((resolved & eSymbolContextCompUnit) && m_sc.comp_unit == NULL) 4129da7bd07SGreg Clayton m_sc.comp_unit = sc.comp_unit; 4139da7bd07SGreg Clayton if ((resolved & eSymbolContextFunction) && m_sc.function == NULL) 4149da7bd07SGreg Clayton m_sc.function = sc.function; 4159da7bd07SGreg Clayton if ((resolved & eSymbolContextBlock) && m_sc.block == NULL) 4169da7bd07SGreg Clayton m_sc.block = sc.block; 4179da7bd07SGreg Clayton if ((resolved & eSymbolContextSymbol) && m_sc.symbol == NULL) 4189da7bd07SGreg Clayton m_sc.symbol = sc.symbol; 4199da7bd07SGreg Clayton if ((resolved & eSymbolContextLineEntry) && !m_sc.line_entry.IsValid()) 4209da7bd07SGreg Clayton m_sc.line_entry = sc.line_entry; 4219da7bd07SGreg Clayton 42230fdc8d8SChris Lattner } 42330fdc8d8SChris Lattner } 42430fdc8d8SChris Lattner else 42530fdc8d8SChris Lattner { 42630fdc8d8SChris Lattner // If we don't have a module, then we can't have the compile unit, 42730fdc8d8SChris Lattner // function, block, line entry or symbol, so we can safely call 42830fdc8d8SChris Lattner // ResolveSymbolContextForAddress with our symbol context member m_sc. 4299da7bd07SGreg Clayton resolved |= m_thread.GetProcess().GetTarget().GetImages().ResolveSymbolContextForAddress (lookup_addr, resolve_scope, m_sc); 43030fdc8d8SChris Lattner } 43130fdc8d8SChris Lattner 43230fdc8d8SChris Lattner // If the target was requested add that: 43330fdc8d8SChris Lattner if (m_sc.target_sp.get() == NULL) 4349da7bd07SGreg Clayton { 43530fdc8d8SChris Lattner m_sc.target_sp = CalculateProcess()->GetTarget().GetSP(); 4369da7bd07SGreg Clayton if (m_sc.target_sp) 4379da7bd07SGreg Clayton resolved |= eSymbolContextTarget; 4389da7bd07SGreg Clayton } 43930fdc8d8SChris Lattner 44030fdc8d8SChris Lattner // Update our internal flags so we remember what we have tried to locate so 44130fdc8d8SChris Lattner // we don't have to keep trying when more calls to this function are made. 4429da7bd07SGreg Clayton // We might have dug up more information that was requested (for example 4439da7bd07SGreg Clayton // if we were asked to only get the block, we will have gotten the 4449da7bd07SGreg Clayton // compile unit, and function) so set any additional bits that we resolved 4459da7bd07SGreg Clayton m_flags.Set (resolve_scope | resolved); 44630fdc8d8SChris Lattner } 44730fdc8d8SChris Lattner 44830fdc8d8SChris Lattner // Return the symbol context with everything that was possible to resolve 44930fdc8d8SChris Lattner // resolved. 45030fdc8d8SChris Lattner return m_sc; 45130fdc8d8SChris Lattner } 45230fdc8d8SChris Lattner 45330fdc8d8SChris Lattner 45430fdc8d8SChris Lattner VariableList * 455*288bdf9cSGreg Clayton StackFrame::GetVariableList (bool get_file_globals) 45630fdc8d8SChris Lattner { 45730fdc8d8SChris Lattner if (m_flags.IsClear(RESOLVED_VARIABLES)) 45830fdc8d8SChris Lattner { 45930fdc8d8SChris Lattner m_flags.Set(RESOLVED_VARIABLES); 46030fdc8d8SChris Lattner 461*288bdf9cSGreg Clayton GetSymbolContext (eSymbolContextCompUnit | 462*288bdf9cSGreg Clayton eSymbolContextFunction | 463*288bdf9cSGreg Clayton eSymbolContextBlock); 464*288bdf9cSGreg Clayton 465*288bdf9cSGreg Clayton if (m_sc.block) 46630fdc8d8SChris Lattner { 46730fdc8d8SChris Lattner bool get_child_variables = true; 46830fdc8d8SChris Lattner bool can_create = true; 4690b76a2c2SGreg Clayton m_variable_list_sp = m_sc.function->GetBlock (can_create).GetVariableList (get_child_variables, can_create); 47030fdc8d8SChris Lattner } 471*288bdf9cSGreg Clayton 472*288bdf9cSGreg Clayton if (get_file_globals && m_sc.comp_unit) 473*288bdf9cSGreg Clayton { 474*288bdf9cSGreg Clayton VariableListSP global_variable_list_sp (m_sc.comp_unit->GetVariableList(true)); 475*288bdf9cSGreg Clayton if (m_variable_list_sp) 476*288bdf9cSGreg Clayton m_variable_list_sp->AddVariables (global_variable_list_sp.get()); 477*288bdf9cSGreg Clayton else 478*288bdf9cSGreg Clayton m_variable_list_sp = global_variable_list_sp; 479*288bdf9cSGreg Clayton } 48030fdc8d8SChris Lattner } 48130fdc8d8SChris Lattner return m_variable_list_sp.get(); 48230fdc8d8SChris Lattner } 48330fdc8d8SChris Lattner 48430fdc8d8SChris Lattner 48530fdc8d8SChris Lattner bool 48630fdc8d8SChris Lattner StackFrame::GetFrameBaseValue (Scalar &frame_base, Error *error_ptr) 48730fdc8d8SChris Lattner { 48830fdc8d8SChris Lattner if (m_flags.IsClear(GOT_FRAME_BASE)) 48930fdc8d8SChris Lattner { 49030fdc8d8SChris Lattner if (m_sc.function) 49130fdc8d8SChris Lattner { 49230fdc8d8SChris Lattner m_frame_base.Clear(); 49330fdc8d8SChris Lattner m_frame_base_error.Clear(); 49430fdc8d8SChris Lattner 49530fdc8d8SChris Lattner m_flags.Set(GOT_FRAME_BASE); 49630fdc8d8SChris Lattner ExecutionContext exe_ctx (&m_thread.GetProcess(), &m_thread, this); 49730fdc8d8SChris Lattner Value expr_value; 49830fdc8d8SChris Lattner if (m_sc.function->GetFrameBaseExpression().Evaluate(&exe_ctx, NULL, NULL, expr_value, &m_frame_base_error) < 0) 49930fdc8d8SChris Lattner { 50030fdc8d8SChris Lattner // We should really have an error if evaluate returns, but in case 50130fdc8d8SChris Lattner // we don't, lets set the error to something at least. 50230fdc8d8SChris Lattner if (m_frame_base_error.Success()) 50330fdc8d8SChris Lattner m_frame_base_error.SetErrorString("Evaluation of the frame base expression failed."); 50430fdc8d8SChris Lattner } 50530fdc8d8SChris Lattner else 50630fdc8d8SChris Lattner { 50730fdc8d8SChris Lattner m_frame_base = expr_value.ResolveValue(&exe_ctx, NULL); 50830fdc8d8SChris Lattner } 50930fdc8d8SChris Lattner } 51030fdc8d8SChris Lattner else 51130fdc8d8SChris Lattner { 51230fdc8d8SChris Lattner m_frame_base_error.SetErrorString ("No function in symbol context."); 51330fdc8d8SChris Lattner } 51430fdc8d8SChris Lattner } 51530fdc8d8SChris Lattner 51630fdc8d8SChris Lattner if (m_frame_base_error.Success()) 51730fdc8d8SChris Lattner frame_base = m_frame_base; 51830fdc8d8SChris Lattner 51930fdc8d8SChris Lattner if (error_ptr) 52030fdc8d8SChris Lattner *error_ptr = m_frame_base_error; 52130fdc8d8SChris Lattner return m_frame_base_error.Success(); 52230fdc8d8SChris Lattner } 52330fdc8d8SChris Lattner 52430fdc8d8SChris Lattner RegisterContext * 52530fdc8d8SChris Lattner StackFrame::GetRegisterContext () 52630fdc8d8SChris Lattner { 52730fdc8d8SChris Lattner if (m_reg_context_sp.get() == NULL) 52830fdc8d8SChris Lattner m_reg_context_sp.reset (m_thread.CreateRegisterContextForFrame (this)); 52930fdc8d8SChris Lattner return m_reg_context_sp.get(); 53030fdc8d8SChris Lattner } 53130fdc8d8SChris Lattner 53230fdc8d8SChris Lattner bool 53330fdc8d8SChris Lattner StackFrame::HasDebugInformation () 53430fdc8d8SChris Lattner { 53530fdc8d8SChris Lattner GetSymbolContext (eSymbolContextLineEntry); 53630fdc8d8SChris Lattner return m_sc.line_entry.IsValid(); 53730fdc8d8SChris Lattner } 53830fdc8d8SChris Lattner 539*288bdf9cSGreg Clayton 540*288bdf9cSGreg Clayton ValueObjectSP 541*288bdf9cSGreg Clayton StackFrame::GetValueObjectForFrameVariable (const VariableSP &variable_sp) 54230fdc8d8SChris Lattner { 543*288bdf9cSGreg Clayton ValueObjectSP valobj_sp; 544*288bdf9cSGreg Clayton VariableList *var_list = GetVariableList (true); 545*288bdf9cSGreg Clayton if (var_list) 546*288bdf9cSGreg Clayton { 547*288bdf9cSGreg Clayton // Make sure the variable is a frame variable 548*288bdf9cSGreg Clayton const uint32_t var_idx = var_list->FindIndexForVariable (variable_sp.get()); 549*288bdf9cSGreg Clayton const uint32_t num_variables = var_list->GetSize(); 550*288bdf9cSGreg Clayton if (var_idx < num_variables) 551*288bdf9cSGreg Clayton { 552*288bdf9cSGreg Clayton valobj_sp = m_variable_list_value_objects.GetValueObjectAtIndex (var_idx); 553*288bdf9cSGreg Clayton if (valobj_sp.get() == NULL) 554*288bdf9cSGreg Clayton { 555*288bdf9cSGreg Clayton if (m_variable_list_value_objects.GetSize() < num_variables) 556*288bdf9cSGreg Clayton m_variable_list_value_objects.Resize(num_variables); 557*288bdf9cSGreg Clayton valobj_sp.reset (new ValueObjectVariable (variable_sp)); 558*288bdf9cSGreg Clayton m_variable_list_value_objects.SetValueObjectAtIndex (var_idx, valobj_sp); 559*288bdf9cSGreg Clayton } 560*288bdf9cSGreg Clayton } 561*288bdf9cSGreg Clayton } 562*288bdf9cSGreg Clayton return valobj_sp; 563*288bdf9cSGreg Clayton } 564*288bdf9cSGreg Clayton 565*288bdf9cSGreg Clayton ValueObjectSP 566*288bdf9cSGreg Clayton StackFrame::TrackGlobalVariable (const VariableSP &variable_sp) 567*288bdf9cSGreg Clayton { 568*288bdf9cSGreg Clayton // Check to make sure we aren't already tracking this variable? 569*288bdf9cSGreg Clayton ValueObjectSP valobj_sp (GetValueObjectForFrameVariable (variable_sp)); 570*288bdf9cSGreg Clayton if (!valobj_sp) 571*288bdf9cSGreg Clayton { 572*288bdf9cSGreg Clayton // We aren't already tracking this global 573*288bdf9cSGreg Clayton VariableList *var_list = GetVariableList (true); 574*288bdf9cSGreg Clayton // If this frame has no variables, create a new list 575*288bdf9cSGreg Clayton if (var_list == NULL) 576*288bdf9cSGreg Clayton m_variable_list_sp.reset (new VariableList()); 577*288bdf9cSGreg Clayton 578*288bdf9cSGreg Clayton // Add the global/static variable to this frame 579*288bdf9cSGreg Clayton m_variable_list_sp->AddVariable (variable_sp); 580*288bdf9cSGreg Clayton 581*288bdf9cSGreg Clayton // Now make a value object for it so we can track its changes 582*288bdf9cSGreg Clayton valobj_sp = GetValueObjectForFrameVariable (variable_sp); 583*288bdf9cSGreg Clayton } 584*288bdf9cSGreg Clayton return valobj_sp; 58530fdc8d8SChris Lattner } 58630fdc8d8SChris Lattner 5876b8379c4SJim Ingham bool 5886b8379c4SJim Ingham StackFrame::IsInlined () 5896b8379c4SJim Ingham { 59059e8fc1cSGreg Clayton if (m_sc.block == NULL) 59159e8fc1cSGreg Clayton GetSymbolContext (eSymbolContextBlock); 59259e8fc1cSGreg Clayton if (m_sc.block) 59359e8fc1cSGreg Clayton return m_sc.block->GetContainingInlinedBlock() != NULL; 59459e8fc1cSGreg Clayton return false; 5956b8379c4SJim Ingham } 5966b8379c4SJim Ingham 59730fdc8d8SChris Lattner Target * 59830fdc8d8SChris Lattner StackFrame::CalculateTarget () 59930fdc8d8SChris Lattner { 60030fdc8d8SChris Lattner return m_thread.CalculateTarget(); 60130fdc8d8SChris Lattner } 60230fdc8d8SChris Lattner 60330fdc8d8SChris Lattner Process * 60430fdc8d8SChris Lattner StackFrame::CalculateProcess () 60530fdc8d8SChris Lattner { 60630fdc8d8SChris Lattner return m_thread.CalculateProcess(); 60730fdc8d8SChris Lattner } 60830fdc8d8SChris Lattner 60930fdc8d8SChris Lattner Thread * 61030fdc8d8SChris Lattner StackFrame::CalculateThread () 61130fdc8d8SChris Lattner { 61230fdc8d8SChris Lattner return &m_thread; 61330fdc8d8SChris Lattner } 61430fdc8d8SChris Lattner 61530fdc8d8SChris Lattner StackFrame * 61630fdc8d8SChris Lattner StackFrame::CalculateStackFrame () 61730fdc8d8SChris Lattner { 61830fdc8d8SChris Lattner return this; 61930fdc8d8SChris Lattner } 62030fdc8d8SChris Lattner 62130fdc8d8SChris Lattner 62230fdc8d8SChris Lattner void 62330fdc8d8SChris Lattner StackFrame::Calculate (ExecutionContext &exe_ctx) 62430fdc8d8SChris Lattner { 62530fdc8d8SChris Lattner m_thread.Calculate (exe_ctx); 62630fdc8d8SChris Lattner exe_ctx.frame = this; 62730fdc8d8SChris Lattner } 62830fdc8d8SChris Lattner 62930fdc8d8SChris Lattner void 63030fdc8d8SChris Lattner StackFrame::Dump (Stream *strm, bool show_frame_index) 63130fdc8d8SChris Lattner { 63230fdc8d8SChris Lattner if (strm == NULL) 63330fdc8d8SChris Lattner return; 63430fdc8d8SChris Lattner 63530fdc8d8SChris Lattner if (show_frame_index) 6361b72fcb7SGreg Clayton strm->Printf("frame #%u: ", m_frame_index); 6379da7bd07SGreg Clayton strm->Printf("0x%0*llx", m_thread.GetProcess().GetAddressByteSize() * 2, GetFrameCodeAddress().GetLoadAddress(&m_thread.GetProcess())); 6389da7bd07SGreg Clayton GetSymbolContext(eSymbolContextEverything); 63930fdc8d8SChris Lattner strm->PutCString(", where = "); 6401b72fcb7SGreg Clayton // TODO: need to get the 6411b72fcb7SGreg Clayton const bool show_module = true; 6421b72fcb7SGreg Clayton const bool show_inline = true; 6439da7bd07SGreg Clayton m_sc.DumpStopContext(strm, &m_thread.GetProcess(), GetFrameCodeAddress(), show_module, show_inline); 64430fdc8d8SChris Lattner } 64530fdc8d8SChris Lattner 6465082c5fdSGreg Clayton void 64759e8fc1cSGreg Clayton StackFrame::UpdateCurrentFrameFromPreviousFrame (StackFrame &prev_frame) 6485082c5fdSGreg Clayton { 64959e8fc1cSGreg Clayton assert (GetStackID() == prev_frame.GetStackID()); // TODO: remove this after some testing 65059e8fc1cSGreg Clayton m_variable_list_sp = prev_frame.m_variable_list_sp; 651*288bdf9cSGreg Clayton m_variable_list_value_objects.Swap (prev_frame.m_variable_list_value_objects); 65268275d5eSGreg Clayton if (!m_disassembly.GetString().empty()) 65368275d5eSGreg Clayton m_disassembly.GetString().swap (m_disassembly.GetString()); 6545082c5fdSGreg Clayton } 65568275d5eSGreg Clayton 65668275d5eSGreg Clayton 65759e8fc1cSGreg Clayton void 65859e8fc1cSGreg Clayton StackFrame::UpdatePreviousFrameFromCurrentFrame (StackFrame &curr_frame) 65959e8fc1cSGreg Clayton { 66059e8fc1cSGreg Clayton assert (GetStackID() == curr_frame.GetStackID()); // TODO: remove this after some testing 66159e8fc1cSGreg Clayton assert (&m_thread == &curr_frame.m_thread); 66259e8fc1cSGreg Clayton m_frame_index = curr_frame.m_frame_index; 66359e8fc1cSGreg Clayton m_unwind_frame_index = curr_frame.m_unwind_frame_index; 66459e8fc1cSGreg Clayton m_reg_context_sp = curr_frame.m_reg_context_sp; 66559e8fc1cSGreg Clayton m_frame_code_addr = curr_frame.m_frame_code_addr; 66659e8fc1cSGreg Clayton assert (m_sc.target_sp.get() == NULL || curr_frame.m_sc.target_sp.get() == NULL || m_sc.target_sp.get() == curr_frame.m_sc.target_sp.get()); 66759e8fc1cSGreg Clayton assert (m_sc.module_sp.get() == NULL || curr_frame.m_sc.module_sp.get() == NULL || m_sc.module_sp.get() == curr_frame.m_sc.module_sp.get()); 66859e8fc1cSGreg Clayton assert (m_sc.comp_unit == NULL || curr_frame.m_sc.comp_unit == NULL || m_sc.comp_unit == curr_frame.m_sc.comp_unit); 66959e8fc1cSGreg Clayton assert (m_sc.function == NULL || curr_frame.m_sc.function == NULL || m_sc.function == curr_frame.m_sc.function); 67059e8fc1cSGreg Clayton m_sc = curr_frame.m_sc; 67159e8fc1cSGreg Clayton m_flags.Clear(GOT_FRAME_BASE | eSymbolContextEverything); 67259e8fc1cSGreg Clayton m_flags.Set (m_sc.GetResolvedMask()); 67359e8fc1cSGreg Clayton m_frame_base.Clear(); 67459e8fc1cSGreg Clayton m_frame_base_error.Clear(); 67559e8fc1cSGreg Clayton } 67659e8fc1cSGreg Clayton 67759e8fc1cSGreg Clayton 678