//===-- SBTypeSummary.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/SBTypeSummary.h" #include "lldb/API/SBStream.h" #include "lldb/API/SBValue.h" #include "lldb/DataFormatters/DataVisualization.h" #include "llvm/Support/Casting.h" using namespace lldb; using namespace lldb_private; SBTypeSummaryOptions::SBTypeSummaryOptions() { m_opaque_ap.reset(new TypeSummaryOptions()); } SBTypeSummaryOptions::SBTypeSummaryOptions( const lldb::SBTypeSummaryOptions &rhs) { if (rhs.m_opaque_ap) m_opaque_ap.reset(new TypeSummaryOptions(*rhs.m_opaque_ap)); else m_opaque_ap.reset(new TypeSummaryOptions()); } SBTypeSummaryOptions::~SBTypeSummaryOptions() {} bool SBTypeSummaryOptions::IsValid() { return m_opaque_ap.get(); } lldb::LanguageType SBTypeSummaryOptions::GetLanguage() { if (IsValid()) return m_opaque_ap->GetLanguage(); return lldb::eLanguageTypeUnknown; } lldb::TypeSummaryCapping SBTypeSummaryOptions::GetCapping() { if (IsValid()) return m_opaque_ap->GetCapping(); return eTypeSummaryCapped; } void SBTypeSummaryOptions::SetLanguage(lldb::LanguageType l) { if (IsValid()) m_opaque_ap->SetLanguage(l); } void SBTypeSummaryOptions::SetCapping(lldb::TypeSummaryCapping c) { if (IsValid()) m_opaque_ap->SetCapping(c); } lldb_private::TypeSummaryOptions *SBTypeSummaryOptions::operator->() { return m_opaque_ap.get(); } const lldb_private::TypeSummaryOptions *SBTypeSummaryOptions:: operator->() const { return m_opaque_ap.get(); } lldb_private::TypeSummaryOptions *SBTypeSummaryOptions::get() { return m_opaque_ap.get(); } lldb_private::TypeSummaryOptions &SBTypeSummaryOptions::ref() { return *m_opaque_ap; } const lldb_private::TypeSummaryOptions &SBTypeSummaryOptions::ref() const { return *m_opaque_ap; } SBTypeSummaryOptions::SBTypeSummaryOptions( const lldb_private::TypeSummaryOptions *lldb_object_ptr) { SetOptions(lldb_object_ptr); } void SBTypeSummaryOptions::SetOptions( const lldb_private::TypeSummaryOptions *lldb_object_ptr) { if (lldb_object_ptr) m_opaque_ap.reset(new TypeSummaryOptions(*lldb_object_ptr)); else m_opaque_ap.reset(new TypeSummaryOptions()); } SBTypeSummary::SBTypeSummary() : m_opaque_sp() {} SBTypeSummary SBTypeSummary::CreateWithSummaryString(const char *data, uint32_t options) { if (!data || data[0] == 0) return SBTypeSummary(); return SBTypeSummary( TypeSummaryImplSP(new StringSummaryFormat(options, data))); } SBTypeSummary SBTypeSummary::CreateWithFunctionName(const char *data, uint32_t options) { if (!data || data[0] == 0) return SBTypeSummary(); return SBTypeSummary( TypeSummaryImplSP(new ScriptSummaryFormat(options, data))); } SBTypeSummary SBTypeSummary::CreateWithScriptCode(const char *data, uint32_t options) { if (!data || data[0] == 0) return SBTypeSummary(); return SBTypeSummary( TypeSummaryImplSP(new ScriptSummaryFormat(options, "", data))); } SBTypeSummary SBTypeSummary::CreateWithCallback(FormatCallback cb, uint32_t options, const char *description) { SBTypeSummary retval; if (cb) { retval.SetSP(TypeSummaryImplSP(new CXXFunctionSummaryFormat( options, [cb](ValueObject &valobj, Stream &stm, const TypeSummaryOptions &opt) -> bool { SBStream stream; SBValue sb_value(valobj.GetSP()); SBTypeSummaryOptions options(&opt); if (!cb(sb_value, options, stream)) return false; stm.Write(stream.GetData(), stream.GetSize()); return true; }, description ? description : "callback summary formatter"))); } return retval; } SBTypeSummary::SBTypeSummary(const lldb::SBTypeSummary &rhs) : m_opaque_sp(rhs.m_opaque_sp) {} SBTypeSummary::~SBTypeSummary() {} bool SBTypeSummary::IsValid() const { return m_opaque_sp.get() != NULL; } bool SBTypeSummary::IsFunctionCode() { if (!IsValid()) return false; if (ScriptSummaryFormat *script_summary_ptr = llvm::dyn_cast(m_opaque_sp.get())) { const char *ftext = script_summary_ptr->GetPythonScript(); return (ftext && *ftext != 0); } return false; } bool SBTypeSummary::IsFunctionName() { if (!IsValid()) return false; if (ScriptSummaryFormat *script_summary_ptr = llvm::dyn_cast(m_opaque_sp.get())) { const char *ftext = script_summary_ptr->GetPythonScript(); return (!ftext || *ftext == 0); } return false; } bool SBTypeSummary::IsSummaryString() { if (!IsValid()) return false; return m_opaque_sp->GetKind() == TypeSummaryImpl::Kind::eSummaryString; } const char *SBTypeSummary::GetData() { if (!IsValid()) return NULL; if (ScriptSummaryFormat *script_summary_ptr = llvm::dyn_cast(m_opaque_sp.get())) { const char *fname = script_summary_ptr->GetFunctionName(); const char *ftext = script_summary_ptr->GetPythonScript(); if (ftext && *ftext) return ftext; return fname; } else if (StringSummaryFormat *string_summary_ptr = llvm::dyn_cast(m_opaque_sp.get())) return string_summary_ptr->GetSummaryString(); return nullptr; } uint32_t SBTypeSummary::GetOptions() { if (!IsValid()) return lldb::eTypeOptionNone; return m_opaque_sp->GetOptions(); } void SBTypeSummary::SetOptions(uint32_t value) { if (!CopyOnWrite_Impl()) return; m_opaque_sp->SetOptions(value); } void SBTypeSummary::SetSummaryString(const char *data) { if (!IsValid()) return; if (!llvm::isa(m_opaque_sp.get())) ChangeSummaryType(false); if (StringSummaryFormat *string_summary_ptr = llvm::dyn_cast(m_opaque_sp.get())) string_summary_ptr->SetSummaryString(data); } void SBTypeSummary::SetFunctionName(const char *data) { if (!IsValid()) return; if (!llvm::isa(m_opaque_sp.get())) ChangeSummaryType(true); if (ScriptSummaryFormat *script_summary_ptr = llvm::dyn_cast(m_opaque_sp.get())) script_summary_ptr->SetFunctionName(data); } void SBTypeSummary::SetFunctionCode(const char *data) { if (!IsValid()) return; if (!llvm::isa(m_opaque_sp.get())) ChangeSummaryType(true); if (ScriptSummaryFormat *script_summary_ptr = llvm::dyn_cast(m_opaque_sp.get())) script_summary_ptr->SetPythonScript(data); } bool SBTypeSummary::GetDescription(lldb::SBStream &description, lldb::DescriptionLevel description_level) { if (!CopyOnWrite_Impl()) return false; else { description.Printf("%s\n", m_opaque_sp->GetDescription().c_str()); return true; } } bool SBTypeSummary::DoesPrintValue(lldb::SBValue value) { if (!IsValid()) return false; lldb::ValueObjectSP value_sp = value.GetSP(); return m_opaque_sp->DoesPrintValue(value_sp.get()); } lldb::SBTypeSummary &SBTypeSummary::operator=(const lldb::SBTypeSummary &rhs) { if (this != &rhs) { m_opaque_sp = rhs.m_opaque_sp; } return *this; } bool SBTypeSummary::operator==(lldb::SBTypeSummary &rhs) { if (!IsValid()) return !rhs.IsValid(); return m_opaque_sp == rhs.m_opaque_sp; } bool SBTypeSummary::IsEqualTo(lldb::SBTypeSummary &rhs) { if (IsValid()) { // valid and invalid are different if (!rhs.IsValid()) return false; } else { // invalid and valid are different if (rhs.IsValid()) return false; else // both invalid are the same return true; } if (m_opaque_sp->GetKind() != rhs.m_opaque_sp->GetKind()) return false; switch (m_opaque_sp->GetKind()) { case TypeSummaryImpl::Kind::eCallback: return llvm::dyn_cast(m_opaque_sp.get()) == llvm::dyn_cast(rhs.m_opaque_sp.get()); case TypeSummaryImpl::Kind::eScript: if (IsFunctionCode() != rhs.IsFunctionCode()) return false; if (IsFunctionName() != rhs.IsFunctionName()) return false; return GetOptions() == rhs.GetOptions(); case TypeSummaryImpl::Kind::eSummaryString: if (IsSummaryString() != rhs.IsSummaryString()) return false; return GetOptions() == rhs.GetOptions(); case TypeSummaryImpl::Kind::eInternal: return (m_opaque_sp.get() == rhs.m_opaque_sp.get()); } return false; } bool SBTypeSummary::operator!=(lldb::SBTypeSummary &rhs) { if (!IsValid()) return !rhs.IsValid(); return m_opaque_sp != rhs.m_opaque_sp; } lldb::TypeSummaryImplSP SBTypeSummary::GetSP() { return m_opaque_sp; } void SBTypeSummary::SetSP(const lldb::TypeSummaryImplSP &typesummary_impl_sp) { m_opaque_sp = typesummary_impl_sp; } SBTypeSummary::SBTypeSummary(const lldb::TypeSummaryImplSP &typesummary_impl_sp) : m_opaque_sp(typesummary_impl_sp) {} bool SBTypeSummary::CopyOnWrite_Impl() { if (!IsValid()) return false; if (m_opaque_sp.unique()) return true; TypeSummaryImplSP new_sp; if (CXXFunctionSummaryFormat *current_summary_ptr = llvm::dyn_cast(m_opaque_sp.get())) { new_sp = TypeSummaryImplSP(new CXXFunctionSummaryFormat( GetOptions(), current_summary_ptr->m_impl, current_summary_ptr->m_description.c_str())); } else if (ScriptSummaryFormat *current_summary_ptr = llvm::dyn_cast(m_opaque_sp.get())) { new_sp = TypeSummaryImplSP(new ScriptSummaryFormat( GetOptions(), current_summary_ptr->GetFunctionName(), current_summary_ptr->GetPythonScript())); } else if (StringSummaryFormat *current_summary_ptr = llvm::dyn_cast(m_opaque_sp.get())) { new_sp = TypeSummaryImplSP(new StringSummaryFormat( GetOptions(), current_summary_ptr->GetSummaryString())); } SetSP(new_sp); return nullptr != new_sp.get(); } bool SBTypeSummary::ChangeSummaryType(bool want_script) { if (!IsValid()) return false; TypeSummaryImplSP new_sp; if (want_script == (m_opaque_sp->GetKind() == TypeSummaryImpl::Kind::eScript)) { if (m_opaque_sp->GetKind() == lldb_private::TypeSummaryImpl::Kind::eCallback && !want_script) new_sp = TypeSummaryImplSP(new StringSummaryFormat(GetOptions(), "")); else return CopyOnWrite_Impl(); } if (!new_sp) { if (want_script) new_sp = TypeSummaryImplSP(new ScriptSummaryFormat(GetOptions(), "", "")); else new_sp = TypeSummaryImplSP(new StringSummaryFormat(GetOptions(), "")); } SetSP(new_sp); return true; }