tint::CloneContext: Use utils::Hashset and utils::Hashmap
For fewer heap allocations, faster lookups. Also use a generational counter to cache `ListTransforms` map lookups between clones. Reduces repeated map lookups in super-hot code. Change-Id: I34865bb39c756129be845b1eb6255ce436e275f7 Reviewed-on: https://dawn-review.googlesource.com/c/dawn/+/99280 Commit-Queue: Ben Clayton <bclayton@chromium.org> Kokoro: Kokoro <noreply+kokoro@google.com> Reviewed-by: Dan Sinclair <dsinclair@chromium.org>
This commit is contained in:
parent
32a1bce736
commit
18dc315ccb
|
@ -27,9 +27,6 @@ Cloneable::Cloneable() = default;
|
||||||
Cloneable::Cloneable(Cloneable&&) = default;
|
Cloneable::Cloneable(Cloneable&&) = default;
|
||||||
Cloneable::~Cloneable() = default;
|
Cloneable::~Cloneable() = default;
|
||||||
|
|
||||||
CloneContext::ListTransforms::ListTransforms() = default;
|
|
||||||
CloneContext::ListTransforms::~ListTransforms() = default;
|
|
||||||
|
|
||||||
CloneContext::CloneContext(ProgramBuilder* to, Program const* from, bool auto_clone_symbols)
|
CloneContext::CloneContext(ProgramBuilder* to, Program const* from, bool auto_clone_symbols)
|
||||||
: dst(to), src(from) {
|
: dst(to), src(from) {
|
||||||
if (auto_clone_symbols) {
|
if (auto_clone_symbols) {
|
||||||
|
@ -48,7 +45,7 @@ Symbol CloneContext::Clone(Symbol s) {
|
||||||
if (!src) {
|
if (!src) {
|
||||||
return s; // In-place clone
|
return s; // In-place clone
|
||||||
}
|
}
|
||||||
return utils::GetOrCreate(cloned_symbols_, s, [&]() -> Symbol {
|
return cloned_symbols_.GetOrCreate(s, [&]() -> Symbol {
|
||||||
if (symbol_transform_) {
|
if (symbol_transform_) {
|
||||||
return symbol_transform_(s);
|
return symbol_transform_(s);
|
||||||
}
|
}
|
||||||
|
@ -76,9 +73,8 @@ const tint::Cloneable* CloneContext::CloneCloneable(const Cloneable* object) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Was Replace() called for this object?
|
// Was Replace() called for this object?
|
||||||
auto it = replacements_.find(object);
|
if (auto* fn = replacements_.Find(object)) {
|
||||||
if (it != replacements_.end()) {
|
return (*fn)();
|
||||||
return it->second();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Attempt to clone using the registered replacer functions.
|
// Attempt to clone using the registered replacer functions.
|
||||||
|
|
|
@ -18,8 +18,6 @@
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <functional>
|
#include <functional>
|
||||||
#include <type_traits>
|
#include <type_traits>
|
||||||
#include <unordered_map>
|
|
||||||
#include <unordered_set>
|
|
||||||
#include <utility>
|
#include <utility>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
|
@ -28,6 +26,7 @@
|
||||||
#include "src/tint/program_id.h"
|
#include "src/tint/program_id.h"
|
||||||
#include "src/tint/symbol.h"
|
#include "src/tint/symbol.h"
|
||||||
#include "src/tint/traits.h"
|
#include "src/tint/traits.h"
|
||||||
|
#include "src/tint/utils/hashmap.h"
|
||||||
#include "src/tint/utils/vector.h"
|
#include "src/tint/utils/vector.h"
|
||||||
|
|
||||||
// Forward declarations
|
// Forward declarations
|
||||||
|
@ -201,56 +200,49 @@ class CloneContext {
|
||||||
void Clone(utils::Vector<T*, N>& to, const utils::Vector<T*, N>& from) {
|
void Clone(utils::Vector<T*, N>& to, const utils::Vector<T*, N>& from) {
|
||||||
to.Reserve(from.Length());
|
to.Reserve(from.Length());
|
||||||
|
|
||||||
auto list_transform_it = list_transforms_.find(&from);
|
auto transforms = list_transforms_.Find(&from);
|
||||||
if (list_transform_it != list_transforms_.end()) {
|
|
||||||
const auto& transforms = list_transform_it->second;
|
if (transforms) {
|
||||||
for (auto* o : transforms.insert_front_) {
|
for (auto* o : transforms->insert_front_) {
|
||||||
to.Push(CheckedCast<T>(o));
|
to.Push(CheckedCast<T>(o));
|
||||||
}
|
}
|
||||||
for (auto& el : from) {
|
for (auto& el : from) {
|
||||||
auto insert_before_it = transforms.insert_before_.find(el);
|
if (auto* insert_before = transforms->insert_before_.Find(el)) {
|
||||||
if (insert_before_it != transforms.insert_before_.end()) {
|
for (auto insert : *insert_before) {
|
||||||
for (auto insert : insert_before_it->second) {
|
|
||||||
to.Push(CheckedCast<T>(insert));
|
to.Push(CheckedCast<T>(insert));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (transforms.remove_.count(el) == 0) {
|
if (!transforms->remove_.Contains(el)) {
|
||||||
to.Push(Clone(el));
|
to.Push(Clone(el));
|
||||||
}
|
}
|
||||||
auto insert_after_it = transforms.insert_after_.find(el);
|
if (auto* insert_after = transforms->insert_after_.Find(el)) {
|
||||||
if (insert_after_it != transforms.insert_after_.end()) {
|
for (auto insert : *insert_after) {
|
||||||
for (auto insert : insert_after_it->second) {
|
|
||||||
to.Push(CheckedCast<T>(insert));
|
to.Push(CheckedCast<T>(insert));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (auto* o : transforms.insert_back_) {
|
for (auto* o : transforms->insert_back_) {
|
||||||
to.Push(CheckedCast<T>(o));
|
to.Push(CheckedCast<T>(o));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
for (auto& el : from) {
|
for (auto& el : from) {
|
||||||
to.Push(Clone(el));
|
to.Push(Clone(el));
|
||||||
|
|
||||||
// Clone(el) may have inserted after
|
// Clone(el) may have updated the transformation list, adding an `insert_after`
|
||||||
list_transform_it = list_transforms_.find(&from);
|
// transform for `from`.
|
||||||
if (list_transform_it != list_transforms_.end()) {
|
if (transforms) {
|
||||||
const auto& transforms = list_transform_it->second;
|
if (auto* insert_after = transforms->insert_after_.Find(el)) {
|
||||||
|
for (auto insert : *insert_after) {
|
||||||
auto insert_after_it = transforms.insert_after_.find(el);
|
|
||||||
if (insert_after_it != transforms.insert_after_.end()) {
|
|
||||||
for (auto insert : insert_after_it->second) {
|
|
||||||
to.Push(CheckedCast<T>(insert));
|
to.Push(CheckedCast<T>(insert));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Clone(el)s may have inserted back
|
// Clone(el) may have updated the transformation list, adding an `insert_back_`
|
||||||
list_transform_it = list_transforms_.find(&from);
|
// transform for `from`.
|
||||||
if (list_transform_it != list_transforms_.end()) {
|
if (transforms) {
|
||||||
const auto& transforms = list_transform_it->second;
|
for (auto* o : transforms->insert_back_) {
|
||||||
|
|
||||||
for (auto* o : transforms.insert_back_) {
|
|
||||||
to.Push(CheckedCast<T>(o));
|
to.Push(CheckedCast<T>(o));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -358,7 +350,7 @@ class CloneContext {
|
||||||
CloneContext& Replace(const WHAT* what, const WITH* with) {
|
CloneContext& Replace(const WHAT* what, const WITH* with) {
|
||||||
TINT_ASSERT_PROGRAM_IDS_EQUAL_IF_VALID(Clone, src, what);
|
TINT_ASSERT_PROGRAM_IDS_EQUAL_IF_VALID(Clone, src, what);
|
||||||
TINT_ASSERT_PROGRAM_IDS_EQUAL_IF_VALID(Clone, dst, with);
|
TINT_ASSERT_PROGRAM_IDS_EQUAL_IF_VALID(Clone, dst, with);
|
||||||
replacements_[what] = [with]() -> const Cloneable* { return with; };
|
replacements_.Add(what, [with]() -> const Cloneable* { return with; });
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -378,7 +370,7 @@ class CloneContext {
|
||||||
template <typename WHAT, typename WITH, typename = std::invoke_result_t<WITH>>
|
template <typename WHAT, typename WITH, typename = std::invoke_result_t<WITH>>
|
||||||
CloneContext& Replace(const WHAT* what, WITH&& with) {
|
CloneContext& Replace(const WHAT* what, WITH&& with) {
|
||||||
TINT_ASSERT_PROGRAM_IDS_EQUAL_IF_VALID(Clone, src, what);
|
TINT_ASSERT_PROGRAM_IDS_EQUAL_IF_VALID(Clone, src, what);
|
||||||
replacements_[what] = with;
|
replacements_.Add(what, with);
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -396,7 +388,7 @@ class CloneContext {
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
list_transforms_[&vector].remove_.emplace(object);
|
list_transforms_.Edit(&vector).remove_.Add(object);
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -408,9 +400,7 @@ class CloneContext {
|
||||||
template <typename T, size_t N, typename OBJECT>
|
template <typename T, size_t N, typename OBJECT>
|
||||||
CloneContext& InsertFront(const utils::Vector<T, N>& vector, OBJECT* object) {
|
CloneContext& InsertFront(const utils::Vector<T, N>& vector, OBJECT* object) {
|
||||||
TINT_ASSERT_PROGRAM_IDS_EQUAL_IF_VALID(Clone, dst, object);
|
TINT_ASSERT_PROGRAM_IDS_EQUAL_IF_VALID(Clone, dst, object);
|
||||||
auto& transforms = list_transforms_[&vector];
|
list_transforms_.Edit(&vector).insert_front_.Push(object);
|
||||||
auto& list = transforms.insert_front_;
|
|
||||||
list.Push(object);
|
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -422,9 +412,7 @@ class CloneContext {
|
||||||
template <typename T, size_t N, typename OBJECT>
|
template <typename T, size_t N, typename OBJECT>
|
||||||
CloneContext& InsertBack(const utils::Vector<T, N>& vector, OBJECT* object) {
|
CloneContext& InsertBack(const utils::Vector<T, N>& vector, OBJECT* object) {
|
||||||
TINT_ASSERT_PROGRAM_IDS_EQUAL_IF_VALID(Clone, dst, object);
|
TINT_ASSERT_PROGRAM_IDS_EQUAL_IF_VALID(Clone, dst, object);
|
||||||
auto& transforms = list_transforms_[&vector];
|
list_transforms_.Edit(&vector).insert_back_.Push(object);
|
||||||
auto& list = transforms.insert_back_;
|
|
||||||
list.Push(object);
|
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -446,9 +434,7 @@ class CloneContext {
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto& transforms = list_transforms_[&vector];
|
list_transforms_.Edit(&vector).insert_before_.GetOrZero(before).Push(object);
|
||||||
auto& list = transforms.insert_before_[before];
|
|
||||||
list.Push(object);
|
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -470,9 +456,7 @@ class CloneContext {
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto& transforms = list_transforms_[&vector];
|
list_transforms_.Edit(&vector).insert_after_.GetOrZero(after).Push(object);
|
||||||
auto& list = transforms.insert_after_[after];
|
|
||||||
list.Push(object);
|
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -502,6 +486,31 @@ class CloneContext {
|
||||||
std::function<const Cloneable*(const Cloneable*)> function;
|
std::function<const Cloneable*(const Cloneable*)> function;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/// A vector of const Cloneable*
|
||||||
|
using CloneableList = utils::Vector<const Cloneable*, 4>;
|
||||||
|
|
||||||
|
/// Transformations to be applied to a list (vector)
|
||||||
|
struct ListTransforms {
|
||||||
|
/// A map of object in #src to omit when cloned into #dst.
|
||||||
|
utils::Hashset<const Cloneable*, 4> remove_;
|
||||||
|
|
||||||
|
/// A list of objects in #dst to insert before any others when the vector is cloned.
|
||||||
|
CloneableList insert_front_;
|
||||||
|
|
||||||
|
/// A list of objects in #dst to insert after all others when the vector is cloned.
|
||||||
|
CloneableList insert_back_;
|
||||||
|
|
||||||
|
/// A map of object in #src to the list of cloned objects in #dst.
|
||||||
|
/// Clone(const utils::Vector<T*>& v) will use this to insert the map-value
|
||||||
|
/// list into the target vector before cloning and inserting the map-key.
|
||||||
|
utils::Hashmap<const Cloneable*, CloneableList, 4> insert_before_;
|
||||||
|
|
||||||
|
/// A map of object in #src to the list of cloned objects in #dst.
|
||||||
|
/// Clone(const utils::Vector<T*>& v) will use this to insert the map-value
|
||||||
|
/// list into the target vector after cloning and inserting the map-key.
|
||||||
|
utils::Hashmap<const Cloneable*, CloneableList, 4> insert_after_;
|
||||||
|
};
|
||||||
|
|
||||||
CloneContext(const CloneContext&) = delete;
|
CloneContext(const CloneContext&) = delete;
|
||||||
CloneContext& operator=(const CloneContext&) = delete;
|
CloneContext& operator=(const CloneContext&) = delete;
|
||||||
|
|
||||||
|
@ -530,50 +539,78 @@ class CloneContext {
|
||||||
/// @returns the diagnostic list of #dst
|
/// @returns the diagnostic list of #dst
|
||||||
diag::List& Diagnostics() const;
|
diag::List& Diagnostics() const;
|
||||||
|
|
||||||
/// A vector of const Cloneable*
|
/// VectorListTransforms is a map of utils::Vector pointer to transforms for that list
|
||||||
using CloneableList = utils::Vector<const Cloneable*, 4>;
|
struct VectorListTransforms {
|
||||||
|
/// An accessor to the VectorListTransforms map.
|
||||||
|
/// Index caches the last map lookup, and will only re-search the map if the transform map
|
||||||
|
/// was modified since the last lookup.
|
||||||
|
struct Index {
|
||||||
|
/// @returns true if the map now holds a value for the index
|
||||||
|
operator bool() {
|
||||||
|
Update();
|
||||||
|
return cached_;
|
||||||
|
}
|
||||||
|
|
||||||
/// Transformations to be applied to a list (vector)
|
/// @returns a pointer to the indexed map entry
|
||||||
struct ListTransforms {
|
const ListTransforms* operator->() {
|
||||||
/// Constructor
|
Update();
|
||||||
ListTransforms();
|
return cached_;
|
||||||
/// Destructor
|
}
|
||||||
~ListTransforms();
|
|
||||||
|
|
||||||
/// A map of object in #src to omit when cloned into #dst.
|
private:
|
||||||
std::unordered_set<const Cloneable*> remove_;
|
friend VectorListTransforms;
|
||||||
|
|
||||||
/// A list of objects in #dst to insert before any others when the vector is
|
Index(const void* list,
|
||||||
/// cloned.
|
VectorListTransforms& vlt,
|
||||||
CloneableList insert_front_;
|
uint32_t generation,
|
||||||
|
const ListTransforms* cached)
|
||||||
|
: list_(list), vlt_(vlt), generation_(generation), cached_(cached) {}
|
||||||
|
|
||||||
/// A list of objects in #dst to insert befor after any others when the
|
void Update() {
|
||||||
/// vector is cloned.
|
if (vlt_.generation_ != generation_) {
|
||||||
CloneableList insert_back_;
|
cached_ = vlt_.map_.Find(list_);
|
||||||
|
generation_ = vlt_.generation_;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// A map of object in #src to the list of cloned objects in #dst.
|
const void* list_;
|
||||||
/// Clone(const utils::Vector<T*>& v) will use this to insert the map-value
|
VectorListTransforms& vlt_;
|
||||||
/// list into the target vector before cloning and inserting the map-key.
|
uint32_t generation_;
|
||||||
std::unordered_map<const Cloneable*, CloneableList> insert_before_;
|
const ListTransforms* cached_;
|
||||||
|
|
||||||
/// A map of object in #src to the list of cloned objects in #dst.
|
|
||||||
/// Clone(const utils::Vector<T*>& v) will use this to insert the map-value
|
|
||||||
/// list into the target vector after cloning and inserting the map-key.
|
|
||||||
std::unordered_map<const Cloneable*, CloneableList> insert_after_;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/// A map of object in #src to functions that create their replacement in
|
/// Edit returns a reference to the ListTransforms for the given vector pointer and
|
||||||
/// #dst
|
/// increments #list_transform_generation_ signalling that the list transforms have been
|
||||||
std::unordered_map<const Cloneable*, std::function<const Cloneable*()>> replacements_;
|
/// modified.
|
||||||
|
inline ListTransforms& Edit(const void* list) {
|
||||||
|
generation_++;
|
||||||
|
return map_.GetOrZero(list);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// @returns an Index to the transforms for the given list.
|
||||||
|
inline Index Find(const void* list) {
|
||||||
|
return Index{list, *this, generation_, map_.Find(list)};
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
/// The map of vector pointer to ListTransforms
|
||||||
|
utils::Hashmap<const void*, ListTransforms, 4> map_;
|
||||||
|
|
||||||
|
/// A counter that's incremented each time list transforms are modified.
|
||||||
|
uint32_t generation_ = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
/// A map of object in #src to functions that create their replacement in #dst
|
||||||
|
utils::Hashmap<const Cloneable*, std::function<const Cloneable*()>, 8> replacements_;
|
||||||
|
|
||||||
/// A map of symbol in #src to their cloned equivalent in #dst
|
/// A map of symbol in #src to their cloned equivalent in #dst
|
||||||
std::unordered_map<Symbol, Symbol> cloned_symbols_;
|
utils::Hashmap<Symbol, Symbol, 32> cloned_symbols_;
|
||||||
|
|
||||||
/// Cloneable transform functions registered with ReplaceAll()
|
/// Cloneable transform functions registered with ReplaceAll()
|
||||||
utils::Vector<CloneableTransform, 8> transforms_;
|
utils::Vector<CloneableTransform, 8> transforms_;
|
||||||
|
|
||||||
/// Map of utils::Vector pointer to transforms for that list
|
/// Transformations to apply to vectors
|
||||||
std::unordered_map<const void*, ListTransforms> list_transforms_;
|
VectorListTransforms list_transforms_;
|
||||||
|
|
||||||
/// Symbol transform registered with ReplaceAll()
|
/// Symbol transform registered with ReplaceAll()
|
||||||
SymbolTransform symbol_transform_;
|
SymbolTransform symbol_transform_;
|
||||||
|
|
Loading…
Reference in New Issue