blob: 6d99aeb42a407b71aef5cb71238bee6d5005b7a6 [file] [log] [blame] [edit]
//===----------------------------------------------------------------------===//
//
// 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 "MsvcStl.h"
#include "lldb/DataFormatters/FormattersHelpers.h"
#include "lldb/Utility/ConstString.h"
#include "lldb/ValueObject/ValueObject.h"
#include <optional>
using namespace lldb;
using namespace lldb_private;
using namespace lldb_private::formatters;
namespace lldb_private::formatters {
class MsvcStlSpanSyntheticFrontEnd : public SyntheticChildrenFrontEnd {
public:
MsvcStlSpanSyntheticFrontEnd(lldb::ValueObjectSP valobj_sp);
~MsvcStlSpanSyntheticFrontEnd() override = default;
llvm::Expected<uint32_t> CalculateNumChildren() override {
return m_num_elements;
}
lldb::ValueObjectSP GetChildAtIndex(uint32_t idx) override;
lldb::ChildCacheState Update() override;
llvm::Expected<size_t> GetIndexOfChildWithName(ConstString name) override;
private:
ValueObject *m_start = nullptr; ///< First element of span. Held, not owned.
CompilerType m_element_type{}; ///< Type of span elements.
size_t m_num_elements = 0; ///< Number of elements in span.
uint32_t m_element_size = 0; ///< Size in bytes of each span element.
};
lldb_private::formatters::MsvcStlSpanSyntheticFrontEnd::
MsvcStlSpanSyntheticFrontEnd(lldb::ValueObjectSP valobj_sp)
: SyntheticChildrenFrontEnd(*valobj_sp) {
if (valobj_sp)
Update();
}
lldb::ValueObjectSP
lldb_private::formatters::MsvcStlSpanSyntheticFrontEnd::GetChildAtIndex(
uint32_t idx) {
if (!m_start)
return {};
uint64_t offset = idx * m_element_size;
offset = offset + m_start->GetValueAsUnsigned(0);
StreamString name;
name.Printf("[%" PRIu64 "]", (uint64_t)idx);
return CreateValueObjectFromAddress(name.GetString(), offset,
m_backend.GetExecutionContextRef(),
m_element_type);
}
lldb::ChildCacheState
lldb_private::formatters::MsvcStlSpanSyntheticFrontEnd::Update() {
m_start = nullptr;
m_element_type = CompilerType();
m_num_elements = 0;
m_element_size = 0;
ValueObjectSP data_sp = m_backend.GetChildMemberWithName("_Mydata");
if (!data_sp)
return lldb::ChildCacheState::eRefetch;
m_element_type = data_sp->GetCompilerType().GetPointeeType();
// Get element size.
llvm::Expected<uint64_t> size_or_err = m_element_type.GetByteSize(nullptr);
if (!size_or_err) {
LLDB_LOG_ERRORV(GetLog(LLDBLog::DataFormatters), size_or_err.takeError(),
"{0}");
return lldb::ChildCacheState::eRefetch;
}
m_element_size = *size_or_err;
// Get data.
if (m_element_size > 0)
m_start = data_sp.get();
// Get number of elements.
if (auto size_sp = m_backend.GetChildMemberWithName("_Mysize"))
m_num_elements = size_sp->GetValueAsUnsigned(0);
else if (auto field =
m_backend.GetCompilerType()
.GetDirectBaseClassAtIndex(0, nullptr) // _Span_extent_type
.GetStaticFieldWithName("_Mysize"))
m_num_elements = field.GetConstantValue().ULongLong(0);
return lldb::ChildCacheState::eRefetch;
}
llvm::Expected<size_t>
lldb_private::formatters::MsvcStlSpanSyntheticFrontEnd::GetIndexOfChildWithName(
ConstString name) {
if (!m_start)
return llvm::createStringError("Type has no child named '%s'",
name.AsCString());
auto optional_idx = formatters::ExtractIndexFromString(name.GetCString());
if (!optional_idx)
return llvm::createStringError("Type has no child named '%s'",
name.AsCString());
return *optional_idx;
}
bool IsMsvcStlSpan(ValueObject &valobj) {
if (auto valobj_sp = valobj.GetNonSyntheticValue())
return valobj_sp->GetChildMemberWithName("_Mydata") != nullptr;
return false;
}
lldb_private::SyntheticChildrenFrontEnd *
MsvcStlSpanSyntheticFrontEndCreator(CXXSyntheticChildren *,
lldb::ValueObjectSP valobj_sp) {
if (!valobj_sp)
return nullptr;
return new MsvcStlSpanSyntheticFrontEnd(valobj_sp);
}
} // namespace lldb_private::formatters