//===-- SBFrame.cpp ---------------------------------------------*- C++ -*-===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "lldb/API/SBFrame.h" #include #include #include "lldb/lldb-types.h" #include "lldb/Core/Address.h" #include "lldb/Core/ConstString.h" #include "lldb/Core/Log.h" #include "lldb/Core/Stream.h" #include "lldb/Core/StreamFile.h" #include "lldb/Core/ValueObjectRegister.h" #include "lldb/Core/ValueObjectVariable.h" #include "lldb/Expression/ClangUserExpression.h" #include "lldb/Symbol/Block.h" #include "lldb/Symbol/SymbolContext.h" #include "lldb/Symbol/VariableList.h" #include "lldb/Symbol/Variable.h" #include "lldb/Target/ExecutionContext.h" #include "lldb/Target/Target.h" #include "lldb/Target/Process.h" #include "lldb/Target/RegisterContext.h" #include "lldb/Target/StackFrame.h" #include "lldb/Target/Thread.h" #include "lldb/API/SBDebugger.h" #include "lldb/API/SBValue.h" #include "lldb/API/SBAddress.h" #include "lldb/API/SBStream.h" #include "lldb/API/SBSymbolContext.h" #include "lldb/API/SBThread.h" using namespace lldb; using namespace lldb_private; SBFrame::SBFrame () : m_opaque_sp () { } SBFrame::SBFrame (const lldb::StackFrameSP &lldb_object_sp) : m_opaque_sp (lldb_object_sp) { Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) { SBStream sstr; GetDescription (sstr); log->Printf ("SBFrame::SBFrame (sp=%p) => SBFrame(%p): %s", lldb_object_sp.get(), m_opaque_sp.get(), sstr.GetData()); } } SBFrame::~SBFrame() { } void SBFrame::SetFrame (const lldb::StackFrameSP &lldb_object_sp) { void *old_ptr = m_opaque_sp.get(); m_opaque_sp = lldb_object_sp; Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) { log->Printf ("SBFrame(%p)::SetFrame(sp=%p) := SBFrame(%p)", old_ptr, lldb_object_sp.get(), m_opaque_sp.get()); } } bool SBFrame::IsValid() const { return (m_opaque_sp.get() != NULL); } SBSymbolContext SBFrame::GetSymbolContext (uint32_t resolve_scope) const { SBSymbolContext sb_sym_ctx; if (m_opaque_sp) sb_sym_ctx.SetSymbolContext(&m_opaque_sp->GetSymbolContext (resolve_scope)); Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::GetSymbolContext (resolve_scope=0x%8.8x) => SBSymbolContext(%p)", m_opaque_sp.get(), resolve_scope, sb_sym_ctx.get()); return sb_sym_ctx; } SBModule SBFrame::GetModule () const { SBModule sb_module (m_opaque_sp->GetSymbolContext (eSymbolContextModule).module_sp); Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::GetModule () => SBModule(%p)", m_opaque_sp.get(), sb_module.get()); return sb_module; } SBCompileUnit SBFrame::GetCompileUnit () const { SBCompileUnit sb_comp_unit(m_opaque_sp->GetSymbolContext (eSymbolContextCompUnit).comp_unit); Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::GetModule () => SBCompileUnit(%p)", m_opaque_sp.get(), sb_comp_unit.get()); return sb_comp_unit; } SBFunction SBFrame::GetFunction () const { SBFunction sb_function(m_opaque_sp->GetSymbolContext (eSymbolContextFunction).function); Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::GetFunction () => SBFunction(%p)", m_opaque_sp.get(), sb_function.get()); return sb_function; } SBSymbol SBFrame::GetSymbol () const { SBSymbol sb_symbol(m_opaque_sp->GetSymbolContext (eSymbolContextSymbol).symbol); Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::GetSymbol () => SBSymbol(%p)", m_opaque_sp.get(), sb_symbol.get()); return sb_symbol; } SBBlock SBFrame::GetBlock () const { SBBlock sb_block(m_opaque_sp->GetSymbolContext (eSymbolContextBlock).block); Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::GetBlock () => SBBlock(%p)", m_opaque_sp.get(), sb_block.get()); return sb_block; } SBBlock SBFrame::GetFrameBlock () const { SBBlock sb_block(m_opaque_sp->GetFrameBlock ()); Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::GetFrameBlock () => SBBlock(%p)", m_opaque_sp.get(), sb_block.get()); return sb_block; } SBLineEntry SBFrame::GetLineEntry () const { SBLineEntry sb_line_entry(&m_opaque_sp->GetSymbolContext (eSymbolContextLineEntry).line_entry); Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::GetLineEntry () => SBLineEntry(%p)", m_opaque_sp.get(), sb_line_entry.get()); return sb_line_entry; } uint32_t SBFrame::GetFrameID () const { uint32_t frame_idx = m_opaque_sp ? m_opaque_sp->GetFrameIndex () : UINT32_MAX; Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::GetFrameID () => %u", m_opaque_sp.get(), frame_idx); return frame_idx; } lldb::addr_t SBFrame::GetPC () const { lldb::addr_t addr = LLDB_INVALID_ADDRESS; if (m_opaque_sp) addr = m_opaque_sp->GetFrameCodeAddress().GetLoadAddress (&m_opaque_sp->GetThread().GetProcess().GetTarget()); Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::GetPC () => %0xllx", m_opaque_sp.get(), addr); return addr; } bool SBFrame::SetPC (lldb::addr_t new_pc) { bool ret_val = false; if (m_opaque_sp) ret_val = m_opaque_sp->GetRegisterContext()->SetPC (new_pc); Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::SetPC (new_pc=0x%llx) => %i", m_opaque_sp.get(), new_pc, ret_val); return ret_val; } lldb::addr_t SBFrame::GetSP () const { addr_t addr = LLDB_INVALID_ADDRESS; if (m_opaque_sp) addr = m_opaque_sp->GetRegisterContext()->GetSP(); Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::GetSP () => %0xllx", m_opaque_sp.get(), addr); return addr; } lldb::addr_t SBFrame::GetFP () const { lldb::addr_t addr = LLDB_INVALID_ADDRESS; if (m_opaque_sp) addr = m_opaque_sp->GetRegisterContext()->GetFP(); Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::GetFP () => %0xllx", m_opaque_sp.get(), addr); return addr; } SBAddress SBFrame::GetPCAddress () const { SBAddress sb_addr; if (m_opaque_sp) sb_addr.SetAddress (&m_opaque_sp->GetFrameCodeAddress()); Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::GetPCAddress () => SBAddress(%p)", m_opaque_sp.get(), sb_addr.get()); return sb_addr; } void SBFrame::Clear() { m_opaque_sp.reset(); } SBValue SBFrame::LookupVar (const char *var_name) { lldb::VariableSP var_sp; if (IsValid ()) { lldb_private::VariableList variable_list; SBSymbolContext sc = GetSymbolContext (eSymbolContextEverything); SBBlock block = sc.GetBlock(); if (block.IsValid()) block.AppendVariables (true, true, &variable_list); const uint32_t num_variables = variable_list.GetSize(); bool found = false; for (uint32_t i = 0; i < num_variables && !found; ++i) { var_sp = variable_list.GetVariableAtIndex(i); if (var_sp && (var_sp.get()->GetName() == lldb_private::ConstString(var_name))) found = true; } if (!found) var_sp.reset(); } SBValue sb_value; if (var_sp) *sb_value = ValueObjectSP (new ValueObjectVariable (var_sp)); Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::LookupVar (name=\"%s\") => SBValue(%p)", m_opaque_sp.get(), var_name, sb_value.get()); return sb_value; } SBValue SBFrame::LookupVarInScope (const char *var_name, const char *scope) { lldb::VariableSP var_sp; if (IsValid()) { std::string scope_str = scope; lldb::ValueType var_scope = eValueTypeInvalid; // Convert scope_str to be all lowercase; std::transform (scope_str.begin(), scope_str.end(), scope_str.begin(), ::tolower); if (scope_str.compare ("global") == 0) var_scope = eValueTypeVariableGlobal; else if (scope_str.compare ("local") == 0) var_scope = eValueTypeVariableLocal; else if (scope_str.compare ("parameter") == 0) var_scope = eValueTypeVariableArgument; if (var_scope != eValueTypeInvalid) { lldb_private::VariableList variable_list; SBSymbolContext sc = GetSymbolContext (eSymbolContextEverything); SBBlock block = sc.GetBlock(); if (block.IsValid()) block.AppendVariables (true, true, &variable_list); const uint32_t num_variables = variable_list.GetSize(); bool found = false; for (uint32_t i = 0; i < num_variables && !found; ++i) { var_sp = variable_list.GetVariableAtIndex(i); if (var_sp && (var_sp.get()->GetName() == lldb_private::ConstString(var_name)) && var_sp.get()->GetScope() == var_scope) found = true; } if (!found) var_sp.reset(); } } SBValue sb_value; if (var_sp) *sb_value = ValueObjectSP (new ValueObjectVariable (var_sp)); Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::LookupVarInScope (name=\"%s\", scope=%s) => SBValue(%p)", m_opaque_sp.get(), var_name, scope, sb_value.get()); return sb_value; } bool SBFrame::operator == (const SBFrame &rhs) const { return m_opaque_sp.get() == rhs.m_opaque_sp.get(); } bool SBFrame::operator != (const SBFrame &rhs) const { return m_opaque_sp.get() != rhs.m_opaque_sp.get(); } lldb_private::StackFrame * SBFrame::operator->() const { return m_opaque_sp.get(); } lldb_private::StackFrame * SBFrame::get() const { return m_opaque_sp.get(); } SBThread SBFrame::GetThread () const { Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); //if (log) // log->Printf ("SBFrame::GetThread ()"); SBThread sb_thread (m_opaque_sp->GetThread().GetSP()); if (log) { SBStream sstr; sb_thread.GetDescription (sstr); log->Printf ("SBFrame(%p)::GetThread () => SBThread(%p): %s", m_opaque_sp.get(), sb_thread.get(), sstr.GetData()); } return sb_thread; } const char * SBFrame::Disassemble () const { const char *disassembly = NULL; if (m_opaque_sp) disassembly = m_opaque_sp->Disassemble(); Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::Disassemble () => %s", m_opaque_sp.get(), disassembly); return disassembly; } SBValueList SBFrame::GetVariables (bool arguments, bool locals, bool statics, bool in_scope_only) { Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); if (log) log->Printf ("SBFrame(%p)::GetVariables (arguments=%i, locals=%i, statics=%i, in_scope_only=%i)", m_opaque_sp.get(), arguments, locals, statics, in_scope_only); SBValueList value_list; if (m_opaque_sp) { size_t i; VariableList *variable_list = m_opaque_sp->GetVariableList(true); if (variable_list) { const size_t num_variables = variable_list->GetSize(); if (num_variables) { for (i = 0; i < num_variables; ++i) { VariableSP variable_sp (variable_list->GetVariableAtIndex(i)); if (variable_sp) { bool add_variable = false; switch (variable_sp->GetScope()) { case eValueTypeVariableGlobal: case eValueTypeVariableStatic: add_variable = statics; break; case eValueTypeVariableArgument: add_variable = arguments; break; case eValueTypeVariableLocal: add_variable = locals; break; default: break; } if (add_variable) { if (in_scope_only && !variable_sp->IsInScope(m_opaque_sp.get())) continue; value_list.Append(m_opaque_sp->GetValueObjectForFrameVariable (variable_sp)); } } } } } } if (log) { log->Printf ("SBFrame(%p)::GetVariables (...) => SBValueList(%p)", m_opaque_sp.get(), value_list.get()); } return value_list; } lldb::SBValueList SBFrame::GetRegisters () { Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); SBValueList value_list; if (m_opaque_sp) { RegisterContext *reg_ctx = m_opaque_sp->GetRegisterContext(); if (reg_ctx) { const uint32_t num_sets = reg_ctx->GetRegisterSetCount(); for (uint32_t set_idx = 0; set_idx < num_sets; ++set_idx) { value_list.Append(ValueObjectSP (new ValueObjectRegisterSet (NULL, reg_ctx, set_idx))); } } } if (log) log->Printf ("SBFrame(%p)::Registers () => SBValueList(%p)", m_opaque_sp.get(), value_list.get()); return value_list; } bool SBFrame::GetDescription (SBStream &description) { if (m_opaque_sp) { SBLineEntry line_entry = GetLineEntry (); SBFileSpec file_spec = line_entry.GetFileSpec (); uint32_t line = line_entry.GetLine (); description.Printf("SBFrame: idx = %u ('%s', %s, line %d)", m_opaque_sp->GetFrameIndex(), GetFunction().GetName(), file_spec.GetFilename(), line); } else description.Printf ("No value"); return true; } lldb::SBValue SBFrame::EvaluateExpression (const char *expr) { Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API); lldb::SBValue expr_result_value; if (log) log->Printf ("SBFrame(%p)::EvaluateExpression (expr=\"%s\")...", m_opaque_sp.get(), expr); if (m_opaque_sp) { ExecutionContext exe_ctx; m_opaque_sp->CalculateExecutionContext (exe_ctx); const char *prefix = NULL; if (exe_ctx.target) prefix = exe_ctx.target->GetExpressionPrefixContentsAsCString(); *expr_result_value = ClangUserExpression::Evaluate (exe_ctx, expr, prefix); } if (log) log->Printf ("SBFrame(%p)::EvaluateExpression (expr=\"%s\") => SBValue(%p)", m_opaque_sp.get(), expr_result_value.get()); return expr_result_value; }