| //===-- LibCxxUnorderedMap.cpp --------------------------------------------===// |
| // |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| // See https://llvm.org/LICENSE.txt for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "LibCxx.h" |
| |
| #include "Plugins/TypeSystem/Clang/TypeSystemClang.h" |
| #include "lldb/DataFormatters/FormattersHelpers.h" |
| #include "lldb/Target/Target.h" |
| #include "lldb/Utility/ConstString.h" |
| #include "lldb/Utility/DataBufferHeap.h" |
| #include "lldb/Utility/Endian.h" |
| #include "lldb/Utility/Status.h" |
| #include "lldb/Utility/Stream.h" |
| #include "lldb/ValueObject/ValueObject.h" |
| #include "lldb/ValueObject/ValueObjectConstResult.h" |
| #include "llvm/ADT/StringRef.h" |
| #include "llvm/Support/Error.h" |
| |
| using namespace lldb; |
| using namespace lldb_private; |
| using namespace lldb_private::formatters; |
| |
| namespace lldb_private { |
| namespace formatters { |
| class LibcxxStdUnorderedMapSyntheticFrontEnd |
| : public SyntheticChildrenFrontEnd { |
| public: |
| LibcxxStdUnorderedMapSyntheticFrontEnd(lldb::ValueObjectSP valobj_sp); |
| |
| ~LibcxxStdUnorderedMapSyntheticFrontEnd() override = default; |
| |
| llvm::Expected<uint32_t> CalculateNumChildren() override; |
| |
| lldb::ValueObjectSP GetChildAtIndex(uint32_t idx) override; |
| |
| lldb::ChildCacheState Update() override; |
| |
| llvm::Expected<size_t> GetIndexOfChildWithName(ConstString name) override; |
| |
| private: |
| CompilerType GetNodeType(); |
| CompilerType GetElementType(CompilerType table_type); |
| llvm::Expected<size_t> CalculateNumChildrenImpl(ValueObject &table); |
| |
| CompilerType m_element_type; |
| CompilerType m_node_type; |
| ValueObject *m_tree = nullptr; |
| size_t m_num_elements = 0; |
| ValueObject *m_next_element = nullptr; |
| std::vector<ValueObject *> m_elements_cache; |
| }; |
| |
| class LibCxxUnorderedMapIteratorSyntheticFrontEnd |
| : public SyntheticChildrenFrontEnd { |
| public: |
| LibCxxUnorderedMapIteratorSyntheticFrontEnd(lldb::ValueObjectSP valobj_sp); |
| |
| ~LibCxxUnorderedMapIteratorSyntheticFrontEnd() override = default; |
| |
| llvm::Expected<uint32_t> CalculateNumChildren() override; |
| |
| lldb::ValueObjectSP GetChildAtIndex(uint32_t idx) override; |
| |
| lldb::ChildCacheState Update() override; |
| |
| llvm::Expected<size_t> GetIndexOfChildWithName(ConstString name) override; |
| |
| private: |
| lldb::ValueObjectSP m_pair_sp; ///< ValueObject for the key/value pair |
| ///< that the iterator currently points |
| ///< to. |
| }; |
| |
| } // namespace formatters |
| } // namespace lldb_private |
| |
| lldb_private::formatters::LibcxxStdUnorderedMapSyntheticFrontEnd:: |
| LibcxxStdUnorderedMapSyntheticFrontEnd(lldb::ValueObjectSP valobj_sp) |
| : SyntheticChildrenFrontEnd(*valobj_sp), m_element_type(), |
| m_elements_cache() { |
| if (valobj_sp) |
| Update(); |
| } |
| |
| llvm::Expected<uint32_t> lldb_private::formatters:: |
| LibcxxStdUnorderedMapSyntheticFrontEnd::CalculateNumChildren() { |
| return m_num_elements; |
| } |
| |
| static bool isUnorderedMap(ConstString type_name) { |
| return isStdTemplate(type_name, "unordered_map") || |
| isStdTemplate(type_name, "unordered_multimap"); |
| } |
| |
| CompilerType lldb_private::formatters::LibcxxStdUnorderedMapSyntheticFrontEnd:: |
| GetElementType(CompilerType table_type) { |
| auto element_type = |
| table_type.GetDirectNestedTypeWithName("value_type").GetTypedefedType(); |
| |
| // In newer unordered_map layouts, the std::pair element type isn't wrapped |
| // in any helper types. So return it directly. |
| if (isStdTemplate(element_type.GetTypeName(), "pair")) |
| return element_type; |
| |
| // This synthetic provider is used for both unordered_(multi)map and |
| // unordered_(multi)set. For older unordered_map layouts, the element type has |
| // an additional type layer, an internal struct (`__hash_value_type`) that |
| // wraps a std::pair. Peel away the internal wrapper type - whose structure is |
| // of no value to users, to expose the std::pair. This matches the structure |
| // returned by the std::map synthetic provider. |
| CompilerType backend_type = m_backend.GetCompilerType(); |
| if (backend_type.IsPointerOrReferenceType()) |
| backend_type = backend_type.GetPointeeType(); |
| |
| if (isUnorderedMap(backend_type.GetCanonicalType().GetTypeName())) { |
| std::string name; |
| CompilerType field_type = |
| element_type.GetFieldAtIndex(0, name, nullptr, nullptr, nullptr); |
| CompilerType actual_type = field_type.GetTypedefedType(); |
| if (isStdTemplate(actual_type.GetTypeName(), "pair")) |
| return actual_type; |
| } |
| |
| return element_type; |
| } |
| |
| CompilerType lldb_private::formatters::LibcxxStdUnorderedMapSyntheticFrontEnd:: |
| GetNodeType() { |
| auto table_sp = m_backend.GetChildMemberWithName("__table_"); |
| if (!table_sp) |
| return {}; |
| |
| auto [node_sp, is_compressed_pair] = |
| GetValueOrOldCompressedPair(*table_sp, "__first_node_", "__p1_"); |
| if (is_compressed_pair) |
| node_sp = GetFirstValueOfLibCXXCompressedPair(*node_sp); |
| |
| if (!node_sp) |
| return {}; |
| |
| return node_sp->GetCompilerType().GetTypeTemplateArgument(0).GetPointeeType(); |
| } |
| |
| lldb::ValueObjectSP lldb_private::formatters:: |
| LibcxxStdUnorderedMapSyntheticFrontEnd::GetChildAtIndex(uint32_t idx) { |
| if (idx >= CalculateNumChildrenIgnoringErrors()) |
| return lldb::ValueObjectSP(); |
| if (m_tree == nullptr) |
| return lldb::ValueObjectSP(); |
| |
| while (idx >= m_elements_cache.size()) { |
| if (m_next_element == nullptr) |
| return lldb::ValueObjectSP(); |
| |
| Status error; |
| ValueObjectSP node_sp = m_next_element->Dereference(error); |
| if (!node_sp || error.Fail()) |
| return lldb::ValueObjectSP(); |
| |
| ValueObjectSP value_sp = node_sp->GetChildMemberWithName("__value_"); |
| ValueObjectSP hash_sp = node_sp->GetChildMemberWithName("__hash_"); |
| if (!hash_sp || !value_sp) { |
| node_sp = m_next_element->Cast(m_node_type.GetPointerType()) |
| ->Dereference(error); |
| if (!node_sp || error.Fail()) |
| return nullptr; |
| |
| hash_sp = node_sp->GetChildMemberWithName("__hash_"); |
| if (!hash_sp) |
| return nullptr; |
| |
| value_sp = node_sp->GetChildMemberWithName("__value_"); |
| if (!value_sp) { |
| // clang-format off |
| // Since D101206 (ba79fb2e1f), libc++ wraps the `__value_` in an |
| // anonymous union. |
| // Child 0: __hash_node_base base class |
| // Child 1: __hash_ |
| // Child 2: anonymous union |
| // clang-format on |
| auto anon_union_sp = node_sp->GetChildAtIndex(2); |
| if (!anon_union_sp) |
| return nullptr; |
| |
| value_sp = anon_union_sp->GetChildMemberWithName("__value_"); |
| if (!value_sp) |
| return nullptr; |
| } |
| } |
| m_elements_cache.push_back(value_sp.get()); |
| m_next_element = node_sp->GetChildMemberWithName("__next_").get(); |
| if (!m_next_element || m_next_element->GetValueAsUnsigned(0) == 0) |
| m_next_element = nullptr; |
| } |
| |
| ValueObject *val_hash = m_elements_cache[idx]; |
| if (!val_hash) |
| return lldb::ValueObjectSP(); |
| StreamString stream; |
| stream.Printf("[%" PRIu64 "]", (uint64_t)idx); |
| DataExtractor data; |
| Status error; |
| val_hash->GetData(data, error); |
| if (error.Fail()) |
| return lldb::ValueObjectSP(); |
| const bool thread_and_frame_only_if_stopped = true; |
| ExecutionContext exe_ctx = |
| val_hash->GetExecutionContextRef().Lock(thread_and_frame_only_if_stopped); |
| return CreateValueObjectFromData(stream.GetString(), data, exe_ctx, |
| m_element_type); |
| } |
| |
| llvm::Expected<size_t> |
| lldb_private::formatters::LibcxxStdUnorderedMapSyntheticFrontEnd:: |
| CalculateNumChildrenImpl(ValueObject &table) { |
| auto [size_sp, is_compressed_pair] = |
| GetValueOrOldCompressedPair(table, "__size_", "__p2_"); |
| if (!is_compressed_pair && size_sp) |
| return size_sp->GetValueAsUnsigned(0); |
| |
| if (!is_compressed_pair) |
| return llvm::createStringError("Unsupported std::unordered_map layout."); |
| |
| ValueObjectSP num_elements_sp = GetFirstValueOfLibCXXCompressedPair(*size_sp); |
| |
| if (!num_elements_sp) |
| return llvm::createStringError( |
| "Unexpected std::unordered_map layout: failed to retrieve first member " |
| "in old __compressed_pair layout."); |
| |
| return num_elements_sp->GetValueAsUnsigned(0); |
| } |
| |
| static ValueObjectSP GetTreePointer(ValueObject &table) { |
| auto [tree_sp, is_compressed_pair] = |
| GetValueOrOldCompressedPair(table, "__first_node_", "__p1_"); |
| if (is_compressed_pair) |
| tree_sp = GetFirstValueOfLibCXXCompressedPair(*tree_sp); |
| |
| if (!tree_sp) |
| return nullptr; |
| |
| return tree_sp->GetChildMemberWithName("__next_"); |
| } |
| |
| lldb::ChildCacheState |
| lldb_private::formatters::LibcxxStdUnorderedMapSyntheticFrontEnd::Update() { |
| m_num_elements = 0; |
| m_next_element = nullptr; |
| m_elements_cache.clear(); |
| ValueObjectSP table_sp = m_backend.GetChildMemberWithName("__table_"); |
| if (!table_sp) |
| return lldb::ChildCacheState::eRefetch; |
| |
| m_node_type = GetNodeType(); |
| if (!m_node_type) |
| return lldb::ChildCacheState::eRefetch; |
| |
| m_element_type = GetElementType(table_sp->GetCompilerType()); |
| if (!m_element_type) |
| return lldb::ChildCacheState::eRefetch; |
| |
| ValueObjectSP tree_sp = GetTreePointer(*table_sp); |
| if (!tree_sp) |
| return lldb::ChildCacheState::eRefetch; |
| |
| m_tree = tree_sp.get(); |
| |
| if (auto num_elems_or_err = CalculateNumChildrenImpl(*table_sp)) |
| m_num_elements = *num_elems_or_err; |
| else { |
| LLDB_LOG_ERRORV(GetLog(LLDBLog::DataFormatters), |
| num_elems_or_err.takeError(), "{0}"); |
| return lldb::ChildCacheState::eRefetch; |
| } |
| |
| if (m_num_elements > 0) |
| m_next_element = m_tree; |
| |
| return lldb::ChildCacheState::eRefetch; |
| } |
| |
| llvm::Expected<size_t> |
| lldb_private::formatters::LibcxxStdUnorderedMapSyntheticFrontEnd:: |
| GetIndexOfChildWithName(ConstString name) { |
| auto optional_idx = formatters::ExtractIndexFromString(name.GetCString()); |
| if (!optional_idx) { |
| return llvm::createStringError("Type has no child named '%s'", |
| name.AsCString()); |
| } |
| return *optional_idx; |
| } |
| |
| SyntheticChildrenFrontEnd * |
| lldb_private::formatters::LibcxxStdUnorderedMapSyntheticFrontEndCreator( |
| CXXSyntheticChildren *, lldb::ValueObjectSP valobj_sp) { |
| return (valobj_sp ? new LibcxxStdUnorderedMapSyntheticFrontEnd(valobj_sp) |
| : nullptr); |
| } |
| |
| lldb_private::formatters::LibCxxUnorderedMapIteratorSyntheticFrontEnd:: |
| LibCxxUnorderedMapIteratorSyntheticFrontEnd(lldb::ValueObjectSP valobj_sp) |
| : SyntheticChildrenFrontEnd(*valobj_sp) { |
| if (valobj_sp) |
| Update(); |
| } |
| |
| lldb::ChildCacheState lldb_private::formatters:: |
| LibCxxUnorderedMapIteratorSyntheticFrontEnd::Update() { |
| m_pair_sp.reset(); |
| |
| ValueObjectSP valobj_sp = m_backend.GetSP(); |
| if (!valobj_sp) |
| return lldb::ChildCacheState::eRefetch; |
| |
| TargetSP target_sp(valobj_sp->GetTargetSP()); |
| |
| if (!target_sp) |
| return lldb::ChildCacheState::eRefetch; |
| |
| // Get the unordered_map::iterator |
| // m_backend is an 'unordered_map::iterator', aka a |
| // '__hash_map_iterator<__hash_table::iterator>' |
| // |
| // __hash_map_iterator::__i_ is a __hash_table::iterator (aka |
| // __hash_iterator<__node_pointer>) |
| auto hash_iter_sp = valobj_sp->GetChildMemberWithName("__i_"); |
| if (!hash_iter_sp) |
| return lldb::ChildCacheState::eRefetch; |
| |
| // Type is '__hash_iterator<__node_pointer>' |
| auto hash_iter_type = hash_iter_sp->GetCompilerType(); |
| if (!hash_iter_type.IsValid()) |
| return lldb::ChildCacheState::eRefetch; |
| |
| // Type is '__node_pointer' |
| auto node_pointer_type = hash_iter_type.GetTypeTemplateArgument(0); |
| if (!node_pointer_type.IsValid()) |
| return lldb::ChildCacheState::eRefetch; |
| |
| // Cast the __hash_iterator to a __node_pointer (which stores our key/value |
| // pair) |
| auto hash_node_sp = hash_iter_sp->Cast(node_pointer_type); |
| if (!hash_node_sp) |
| return lldb::ChildCacheState::eRefetch; |
| |
| auto key_value_sp = hash_node_sp->GetChildMemberWithName("__value_"); |
| if (!key_value_sp) { |
| // clang-format off |
| // Since D101206 (ba79fb2e1f), libc++ wraps the `__value_` in an |
| // anonymous union. |
| // Child 0: __hash_node_base base class |
| // Child 1: __hash_ |
| // Child 2: anonymous union |
| // clang-format on |
| auto anon_union_sp = hash_node_sp->GetChildAtIndex(2); |
| if (!anon_union_sp) |
| return lldb::ChildCacheState::eRefetch; |
| |
| key_value_sp = anon_union_sp->GetChildMemberWithName("__value_"); |
| if (!key_value_sp) |
| return lldb::ChildCacheState::eRefetch; |
| } |
| |
| // Create the synthetic child, which is a pair where the key and value can be |
| // retrieved by querying the synthetic frontend for |
| // GetIndexOfChildWithName("first") and GetIndexOfChildWithName("second") |
| // respectively. |
| // |
| // std::unordered_map stores the actual key/value pair in |
| // __hash_value_type::__cc_ (or previously __cc). |
| auto potential_child_sp = key_value_sp->Clone(ConstString("pair")); |
| if (potential_child_sp) |
| if (potential_child_sp->GetNumChildrenIgnoringErrors() == 1) |
| if (auto child0_sp = potential_child_sp->GetChildAtIndex(0); |
| child0_sp->GetName() == "__cc_" || child0_sp->GetName() == "__cc") |
| potential_child_sp = child0_sp->Clone(ConstString("pair")); |
| |
| m_pair_sp = potential_child_sp; |
| |
| return lldb::ChildCacheState::eRefetch; |
| } |
| |
| llvm::Expected<uint32_t> lldb_private::formatters:: |
| LibCxxUnorderedMapIteratorSyntheticFrontEnd::CalculateNumChildren() { |
| return 2; |
| } |
| |
| lldb::ValueObjectSP lldb_private::formatters:: |
| LibCxxUnorderedMapIteratorSyntheticFrontEnd::GetChildAtIndex(uint32_t idx) { |
| if (m_pair_sp) |
| return m_pair_sp->GetChildAtIndex(idx); |
| return lldb::ValueObjectSP(); |
| } |
| |
| llvm::Expected<size_t> |
| lldb_private::formatters::LibCxxUnorderedMapIteratorSyntheticFrontEnd:: |
| GetIndexOfChildWithName(ConstString name) { |
| if (name == "first") |
| return 0; |
| if (name == "second") |
| return 1; |
| return llvm::createStringError("Type has no child named '%s'", |
| name.AsCString()); |
| } |
| |
| SyntheticChildrenFrontEnd * |
| lldb_private::formatters::LibCxxUnorderedMapIteratorSyntheticFrontEndCreator( |
| CXXSyntheticChildren *, lldb::ValueObjectSP valobj_sp) { |
| return (valobj_sp ? new LibCxxUnorderedMapIteratorSyntheticFrontEnd(valobj_sp) |
| : nullptr); |
| } |