blob: 5e69792151c870a18f1cd768a49d75301bdb3c37 [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 "LibStdcpp.h"
#include "lldb/DataFormatters/FormattersHelpers.h"
#include "lldb/Utility/ConstString.h"
#include "lldb/ValueObject/ValueObject.h"
#include "llvm/ADT/APSInt.h"
#include "llvm/Support/Error.h"
#include <cstddef>
#include <optional>
using namespace lldb;
namespace lldb_private::formatters {
class LibStdcppSpanSyntheticFrontEnd : public SyntheticChildrenFrontEnd {
public:
LibStdcppSpanSyntheticFrontEnd(const lldb::ValueObjectSP &valobj_sp)
: SyntheticChildrenFrontEnd(*valobj_sp) {
if (valobj_sp)
Update();
}
~LibStdcppSpanSyntheticFrontEnd() override = default;
llvm::Expected<uint32_t> CalculateNumChildren() override {
return m_num_elements;
}
lldb::ValueObjectSP GetChildAtIndex(uint32_t idx) override {
if (!m_start)
return {};
uint64_t offset = (static_cast<uint64_t>(idx) * m_element_size);
offset += m_start->GetValueAsUnsigned(0);
const std::string name = llvm::formatv("[{0}]", idx);
return CreateValueObjectFromAddress(
name, offset, m_backend.GetExecutionContextRef(), m_element_type);
}
lldb::ChildCacheState Update() override {
const ValueObjectSP data_ptr = m_backend.GetChildMemberWithName("_M_ptr");
if (!data_ptr)
return lldb::ChildCacheState::eRefetch;
m_element_type = data_ptr->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::eReuse;
}
m_element_size = *size_or_err;
if (m_element_size > 0) {
m_start = data_ptr.get();
}
// Get number of elements.
if (const ValueObjectSP size_sp =
m_backend.GetChildAtNamePath({"_M_extent", "_M_extent_value"})) {
m_num_elements = size_sp->GetValueAsUnsigned(0);
} else if (const auto arg =
m_backend.GetCompilerType().GetIntegralTemplateArgument(1)) {
m_num_elements = arg->value.GetAPSInt().getLimitedValue();
}
return lldb::ChildCacheState::eReuse;
}
llvm::Expected<size_t> GetIndexOfChildWithName(ConstString name) override {
if (!m_start)
return llvm::createStringError(
llvm::formatv("Type has no child named {0}", name.GetStringRef()));
auto optional_idx = formatters::ExtractIndexFromString(name.GetCString());
if (!optional_idx) {
return llvm::createStringError(
llvm::formatv("Type has no child named {0}", name.GetStringRef()));
}
return *optional_idx;
}
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.
};
SyntheticChildrenFrontEnd *
LibStdcppSpanSyntheticFrontEndCreator(CXXSyntheticChildren * /*unused*/,
lldb::ValueObjectSP valobj_sp) {
if (!valobj_sp)
return nullptr;
const CompilerType type = valobj_sp->GetCompilerType();
if (!type || type.GetNumTemplateArguments() != 2)
return nullptr;
return new LibStdcppSpanSyntheticFrontEnd(valobj_sp);
}
} // namespace lldb_private::formatters