From a7acc8315a7fe2e8ff7425b06acd5871e92e3594 Mon Sep 17 00:00:00 2001 From: Peter Johnson Date: Fri, 25 Oct 2024 14:21:27 -0700 Subject: [PATCH] [wpiutil] DynamicStruct: Store StructDescriptor by value --- .../main/native/cpp/struct/DynamicStruct.cpp | 53 +++++++++---------- .../native/include/wpi/struct/DynamicStruct.h | 4 +- .../native/cpp/struct/DynamicStructTest.cpp | 1 + 3 files changed, 26 insertions(+), 32 deletions(-) diff --git a/wpiutil/src/main/native/cpp/struct/DynamicStruct.cpp b/wpiutil/src/main/native/cpp/struct/DynamicStruct.cpp index b2e667b5ad3..28e61dacd17 100644 --- a/wpiutil/src/main/native/cpp/struct/DynamicStruct.cpp +++ b/wpiutil/src/main/native/cpp/struct/DynamicStruct.cpp @@ -5,7 +5,6 @@ #include "wpi/struct/DynamicStruct.h" #include -#include #include #include @@ -195,14 +194,12 @@ const StructDescriptor* StructDescriptorDatabase::Add(std::string_view name, } // turn parsed schema into descriptors - auto& theStruct = m_structs[name]; - if (!theStruct) { - theStruct = std::make_unique( - name, StructDescriptor::private_init{}); - } - theStruct->m_schema = schema; - theStruct->m_fields.clear(); - theStruct->m_fields.reserve(parsed.declarations.size()); + auto& theStruct = + m_structs.try_emplace(name, name, StructDescriptor::private_init{}) + .first->second; + theStruct.m_schema = schema; + theStruct.m_fields.clear(); + theStruct.m_fields.reserve(parsed.declarations.size()); bool isValid = true; for (auto&& decl : parsed.declarations) { auto type = TypeStringToType(decl.typeString); @@ -251,43 +248,41 @@ const StructDescriptor* StructDescriptorDatabase::Add(std::string_view name, } // cross-reference struct, creating a placeholder if necessary - auto& aStruct = m_structs[decl.typeString]; - if (!aStruct) { - aStruct = std::make_unique( - decl.typeString, StructDescriptor::private_init{}); - } + auto& aStruct = m_structs + .try_emplace(decl.typeString, decl.typeString, + StructDescriptor::private_init{}) + .first->second; // if the struct isn't valid, we can't be valid either - if (aStruct->IsValid()) { - size = aStruct->GetSize(); + if (aStruct.IsValid()) { + size = aStruct.GetSize(); } else { isValid = false; } // add to cross-references for when the struct does become valid - aStruct->m_references.emplace_back(theStruct.get()); - structDesc = aStruct.get(); + aStruct.m_references.emplace_back(&theStruct); + structDesc = &aStruct; } // create field - if (!theStruct->m_fieldsByName - .insert({decl.name, theStruct->m_fields.size()}) + if (!theStruct.m_fieldsByName.insert({decl.name, theStruct.m_fields.size()}) .second) { *err = fmt::format("duplicate field {}", decl.name); [[unlikely]] return nullptr; } - theStruct->m_fields.emplace_back(theStruct.get(), decl.name, type, size, - decl.arraySize, decl.bitWidth, - std::move(decl.enumValues), structDesc, - StructFieldDescriptor::private_init{}); + theStruct.m_fields.emplace_back(&theStruct, decl.name, type, size, + decl.arraySize, decl.bitWidth, + std::move(decl.enumValues), structDesc, + StructFieldDescriptor::private_init{}); } - theStruct->m_valid = isValid; + theStruct.m_valid = isValid; if (isValid) { // we have all the info needed, so calculate field offset & shift wpi::SmallVector stack; - auto err2 = theStruct->CalculateOffsets(stack); + auto err2 = theStruct.CalculateOffsets(stack); if (!err2.empty()) { *err = std::move(err2); [[unlikely]] return nullptr; @@ -295,7 +290,7 @@ const StructDescriptor* StructDescriptorDatabase::Add(std::string_view name, } else { // check for circular reference wpi::SmallVector stack; - if (!theStruct->CheckCircular(stack)) { + if (!theStruct.CheckCircular(stack)) { wpi::SmallString<128> buf; wpi::raw_svector_ostream os{buf}; for (auto&& elem : stack) { @@ -309,7 +304,7 @@ const StructDescriptor* StructDescriptorDatabase::Add(std::string_view name, } } - return theStruct.get(); + return &theStruct; } const StructDescriptor* StructDescriptorDatabase::Find( @@ -318,7 +313,7 @@ const StructDescriptor* StructDescriptorDatabase::Find( if (it == m_structs.end()) { return nullptr; } - return it->second.get(); + return &it->second; } uint64_t DynamicStruct::GetFieldImpl(const StructFieldDescriptor* field, diff --git a/wpiutil/src/main/native/include/wpi/struct/DynamicStruct.h b/wpiutil/src/main/native/include/wpi/struct/DynamicStruct.h index f49db2514ec..e807ab7e44c 100644 --- a/wpiutil/src/main/native/include/wpi/struct/DynamicStruct.h +++ b/wpiutil/src/main/native/include/wpi/struct/DynamicStruct.h @@ -7,14 +7,12 @@ #include #include -#include #include #include #include #include #include -#include "wpi/MathExtras.h" #include "wpi/StringMap.h" #include "wpi/bit.h" @@ -357,7 +355,7 @@ class StructDescriptorDatabase { const StructDescriptor* Find(std::string_view name) const; private: - StringMap> m_structs; + StringMap m_structs; }; /** diff --git a/wpiutil/src/test/native/cpp/struct/DynamicStructTest.cpp b/wpiutil/src/test/native/cpp/struct/DynamicStructTest.cpp index 9741667be06..d326263a227 100644 --- a/wpiutil/src/test/native/cpp/struct/DynamicStructTest.cpp +++ b/wpiutil/src/test/native/cpp/struct/DynamicStructTest.cpp @@ -6,6 +6,7 @@ #include +#include #include #include