mirror of
https://github.com/verilator/verilator.git
synced 2025-04-04 19:52:39 +00:00
Rework SortByValueMap for better performance
Keep a single std::set of key/value pairs, and a single unordered_map from key to iterators into the set. Also improve some of the accessing mechanisms using modern C++. This speeds up multi-threaded ordering by about 10%.
This commit is contained in:
parent
b864f5f5ba
commit
fac8e76923
@ -195,12 +195,9 @@ public:
|
|||||||
// longest !wayward edge. Schedule that to be resolved.
|
// longest !wayward edge. Schedule that to be resolved.
|
||||||
const uint32_t newPendingVal
|
const uint32_t newPendingVal
|
||||||
= newInclusiveCp - m_accessp->critPathCost(relativep, m_way);
|
= newInclusiveCp - m_accessp->critPathCost(relativep, m_way);
|
||||||
if (m_pending.has(relativep)) {
|
const auto pair = m_pending.emplace(relativep, newPendingVal);
|
||||||
if (newPendingVal > m_pending.at(relativep)) {
|
if (!pair.second && (newPendingVal > pair.first->second)) {
|
||||||
m_pending.set(relativep, newPendingVal);
|
m_pending.update(pair.first, newPendingVal);
|
||||||
}
|
|
||||||
} else {
|
|
||||||
m_pending.set(relativep, newPendingVal);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -225,8 +222,8 @@ public:
|
|||||||
// This generalizes to multiple seed nodes also.
|
// This generalizes to multiple seed nodes also.
|
||||||
while (!m_pending.empty()) {
|
while (!m_pending.empty()) {
|
||||||
const auto it = m_pending.rbegin();
|
const auto it = m_pending.rbegin();
|
||||||
V3GraphVertex* const updateMep = (*it).key();
|
V3GraphVertex* const updateMep = it->first;
|
||||||
const uint32_t cpGrowBy = (*it).value();
|
const uint32_t cpGrowBy = it->second;
|
||||||
m_pending.erase(it);
|
m_pending.erase(it);
|
||||||
|
|
||||||
// For *updateMep, whose critPathCost was out-of-date with respect
|
// For *updateMep, whose critPathCost was out-of-date with respect
|
||||||
@ -396,8 +393,8 @@ public:
|
|||||||
LogicMTask* const updateVxp = static_cast<LogicMTask*>(vxp);
|
LogicMTask* const updateVxp = static_cast<LogicMTask*>(vxp);
|
||||||
LogicMTask* const lthrouvhVxp = static_cast<LogicMTask*>(throuvhVxp);
|
LogicMTask* const lthrouvhVxp = static_cast<LogicMTask*>(throuvhVxp);
|
||||||
EdgeSet& edges = updateVxp->m_edges[way.invert()];
|
EdgeSet& edges = updateVxp->m_edges[way.invert()];
|
||||||
const uint32_t edgeCp = edges.at(lthrouvhVxp);
|
const auto it = edges.find(lthrouvhVxp);
|
||||||
if (cp > edgeCp) edges.set(lthrouvhVxp, cp);
|
if (cp > it->second) edges.update(it, cp);
|
||||||
}
|
}
|
||||||
// Check that CP matches that of the longest edge wayward of vxp.
|
// Check that CP matches that of the longest edge wayward of vxp.
|
||||||
void checkNewCpVersusEdges(V3GraphVertex* vxp, GraphWay way, uint32_t cp) const {
|
void checkNewCpVersusEdges(V3GraphVertex* vxp, GraphWay way, uint32_t cp) const {
|
||||||
@ -405,7 +402,7 @@ public:
|
|||||||
const EdgeSet& edges = mtaskp->m_edges[way.invert()];
|
const EdgeSet& edges = mtaskp->m_edges[way.invert()];
|
||||||
// This is mtaskp's relative with longest !wayward inclusive CP:
|
// This is mtaskp's relative with longest !wayward inclusive CP:
|
||||||
const auto edgeIt = edges.rbegin();
|
const auto edgeIt = edges.rbegin();
|
||||||
const uint32_t edgeCp = (*edgeIt).value();
|
const uint32_t edgeCp = edgeIt->second;
|
||||||
UASSERT_OBJ(edgeCp == cp, vxp, "CP doesn't match longest wayward edge");
|
UASSERT_OBJ(edgeCp == cp, vxp, "CP doesn't match longest wayward edge");
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -512,26 +509,21 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
void addRelative(GraphWay way, LogicMTask* relativep) {
|
void addRelative(GraphWay way, LogicMTask* relativep) {
|
||||||
EdgeSet& edges = m_edges[way];
|
|
||||||
UASSERT(!edges.has(relativep), "Adding existing edge");
|
|
||||||
// value is !way cp to this edge
|
// value is !way cp to this edge
|
||||||
edges.set(relativep, relativep->stepCost() + relativep->critPathCost(way.invert()));
|
const uint32_t cp = relativep->stepCost() + relativep->critPathCost(way.invert());
|
||||||
}
|
VL_ATTR_UNUSED const bool exits = !m_edges[way].emplace(relativep, cp).second;
|
||||||
void removeRelative(GraphWay way, LogicMTask* relativep) {
|
#if VL_DEBUG
|
||||||
EdgeSet& edges = m_edges[way];
|
UASSERT(!exits, "Adding existing edge");
|
||||||
edges.erase(relativep);
|
#endif
|
||||||
}
|
|
||||||
bool hasRelative(GraphWay way, LogicMTask* relativep) {
|
|
||||||
const EdgeSet& edges = m_edges[way];
|
|
||||||
return edges.has(relativep);
|
|
||||||
}
|
}
|
||||||
|
void removeRelative(GraphWay way, LogicMTask* relativep) { m_edges[way].erase(relativep); }
|
||||||
|
bool hasRelative(GraphWay way, LogicMTask* relativep) { return m_edges[way].has(relativep); }
|
||||||
void checkRelativesCp(GraphWay way) const {
|
void checkRelativesCp(GraphWay way) const {
|
||||||
const EdgeSet& edges = m_edges[way];
|
for (const auto& edge : vlstd::reverse_view(m_edges[way])) {
|
||||||
for (const auto& edge : vlstd::reverse_view(edges)) {
|
const LogicMTask* const relativep = edge.first;
|
||||||
const LogicMTask* const relativep = edge.key();
|
const uint32_t cachedCp = edge.second;
|
||||||
const uint32_t cachedCp = edge.value();
|
const uint32_t cp = relativep->critPathCost(way.invert()) + relativep->stepCost();
|
||||||
partCheckCachedScoreVsActual(cachedCp, relativep->critPathCost(way.invert())
|
partCheckCachedScoreVsActual(cachedCp, cp);
|
||||||
+ relativep->stepCost());
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -557,11 +549,11 @@ public:
|
|||||||
const EdgeSet& edges = m_edges[way.invert()];
|
const EdgeSet& edges = m_edges[way.invert()];
|
||||||
uint32_t result = 0;
|
uint32_t result = 0;
|
||||||
for (const auto& edge : vlstd::reverse_view(edges)) {
|
for (const auto& edge : vlstd::reverse_view(edges)) {
|
||||||
if (edge.key() != withoutp->furtherp(way.invert())) {
|
if (edge.first != withoutp->furtherp(way.invert())) {
|
||||||
// Use the cached cost. It could be a small overestimate
|
// Use the cached cost. It could be a small overestimate
|
||||||
// due to stepping. This is consistent with critPathCost()
|
// due to stepping. This is consistent with critPathCost()
|
||||||
// which also returns the cached cost.
|
// which also returns the cached cost.
|
||||||
result = edge.value();
|
result = edge.second;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -657,7 +649,7 @@ public:
|
|||||||
if (it == children.rend()) {
|
if (it == children.rend()) {
|
||||||
nextp = nullptr;
|
nextp = nullptr;
|
||||||
} else {
|
} else {
|
||||||
nextp = (*it).key();
|
nextp = it->first;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1477,6 +1469,7 @@ private:
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
VL_ATTR_NOINLINE
|
||||||
static uint32_t siblingScore(const SiblingMC* sibsp) {
|
static uint32_t siblingScore(const SiblingMC* sibsp) {
|
||||||
const LogicMTask* const ap = sibsp->ap();
|
const LogicMTask* const ap = sibsp->ap();
|
||||||
const LogicMTask* const bp = sibsp->bp();
|
const LogicMTask* const bp = sibsp->bp();
|
||||||
@ -1487,6 +1480,7 @@ private:
|
|||||||
return mergedCpCostRev + mergedCpCostFwd + LogicMTask::stepCost(ap->cost() + bp->cost());
|
return mergedCpCostRev + mergedCpCostFwd + LogicMTask::stepCost(ap->cost() + bp->cost());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
VL_ATTR_NOINLINE
|
||||||
static uint32_t edgeScore(const V3GraphEdge* edgep) {
|
static uint32_t edgeScore(const V3GraphEdge* edgep) {
|
||||||
// Score this edge. Lower is better. The score is the new local CP
|
// Score this edge. Lower is better. The score is the new local CP
|
||||||
// length if we merge these mtasks. ("Local" means the longest
|
// length if we merge these mtasks. ("Local" means the longest
|
||||||
|
@ -34,298 +34,103 @@
|
|||||||
#include <set>
|
#include <set>
|
||||||
#include <unordered_map>
|
#include <unordered_map>
|
||||||
|
|
||||||
//######################################################################
|
// ######################################################################
|
||||||
// SortByValueMap
|
// SortByValueMap
|
||||||
|
|
||||||
/// A generic key-value map, except it also supports iterating in
|
// A generic key-value map, except iteration is in *value* sorted order. Values need not be unique.
|
||||||
/// value-sorted order. Values need not be unique. Uses T_KeyCompare to
|
// Uses T_KeyCompare to break ties in the sort when values collide. Note: Only const iteration is
|
||||||
/// break ties in the sort when values collide.
|
// possible, as updating mapped values via iterators is not safe.
|
||||||
|
|
||||||
template <typename T_Key, typename T_Value, class T_KeyCompare = std::less<T_Key>>
|
template <typename T_Key, typename T_Value, class T_KeyCompare = std::less<T_Key>>
|
||||||
class SortByValueMap final {
|
class SortByValueMap final {
|
||||||
// TYPES
|
// Current implementation is a std::set of key/value pairs, plus a std_unordered_map from keys
|
||||||
private:
|
// to iterators into the set. This keeps most operations fairly cheap and also has the benefit
|
||||||
using KeySet = std::set<T_Key, T_KeyCompare>;
|
// of being able to re-use the std::set iterators.
|
||||||
using Val2Keys = std::map<T_Value, KeySet>;
|
|
||||||
|
|
||||||
|
// TYPES
|
||||||
|
|
||||||
|
using Pair = std::pair<T_Key, T_Value>;
|
||||||
|
|
||||||
|
struct PairCmp final {
|
||||||
|
bool operator()(const Pair& a, const Pair& b) const {
|
||||||
|
// First compare values
|
||||||
|
if (a.second != b.second) return a.second < b.second;
|
||||||
|
// Then compare keys
|
||||||
|
return T_KeyCompare{}(a.first, b.first);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
using PairSet = std::set<Pair, PairCmp>;
|
||||||
|
|
||||||
|
public:
|
||||||
|
using const_iterator = typename PairSet::const_iterator;
|
||||||
|
using const_reverse_iterator = typename PairSet::const_reverse_iterator;
|
||||||
|
|
||||||
|
private:
|
||||||
// MEMBERS
|
// MEMBERS
|
||||||
std::unordered_map<T_Key, T_Value> m_keys; // Map each key to its value. Not sorted.
|
PairSet m_pairs; // The contents of the map, stored directly as key-value pairs
|
||||||
Val2Keys m_vals; // Map each value to its keys. Sorted.
|
std::unordered_map<T_Key, const_iterator> m_kiMap; // Key to iterator map
|
||||||
|
|
||||||
|
VL_UNCOPYABLE(SortByValueMap);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
// CONSTRUCTORS
|
// CONSTRUCTORS
|
||||||
SortByValueMap() = default;
|
SortByValueMap() = default;
|
||||||
|
|
||||||
class const_iterator VL_NOT_FINAL {
|
// Only const iteration is possible
|
||||||
// TYPES
|
const_iterator begin() const { return m_pairs.begin(); }
|
||||||
public:
|
const_iterator end() const { return m_pairs.end(); }
|
||||||
using value_type = const_iterator;
|
const_iterator cbegin() const { m_pairs.cbegin(); }
|
||||||
using reference = const_iterator; // See comment on operator*()
|
const_iterator cend() const { return m_pairs.cend(); }
|
||||||
using pointer = void;
|
const_reverse_iterator rbegin() const { return m_pairs.rbegin(); }
|
||||||
using difference_type = std::ptrdiff_t;
|
const_reverse_iterator rend() const { return m_pairs.rend(); }
|
||||||
using iterator_category = std::bidirectional_iterator_tag;
|
const_reverse_iterator crbegin() const { return m_pairs.crbegin(); }
|
||||||
|
const_reverse_iterator crend() const { return m_pairs.crend(); }
|
||||||
|
|
||||||
protected:
|
const_iterator find(const T_Key& key) const {
|
||||||
friend class SortByValueMap;
|
const auto kiIt = m_kiMap.find(key);
|
||||||
|
if (kiIt == m_kiMap.end()) return cend();
|
||||||
// MEMBERS
|
return kiIt->second;
|
||||||
typename KeySet::iterator m_keyIt;
|
|
||||||
typename Val2Keys::iterator m_valIt;
|
|
||||||
SortByValueMap* const m_sbvmp;
|
|
||||||
bool m_end = true; // At the end()
|
|
||||||
|
|
||||||
// CONSTRUCTORS
|
|
||||||
explicit const_iterator(SortByValueMap* sbmvp) // for end()
|
|
||||||
: m_sbvmp{sbmvp} {}
|
|
||||||
const_iterator(typename Val2Keys::iterator valIt, typename KeySet::iterator keyIt,
|
|
||||||
SortByValueMap* sbvmp)
|
|
||||||
: m_keyIt{keyIt}
|
|
||||||
, m_valIt{valIt}
|
|
||||||
, m_sbvmp{sbvmp}
|
|
||||||
, m_end{false} {}
|
|
||||||
|
|
||||||
// METHODS
|
|
||||||
void advanceUntilValid() {
|
|
||||||
++m_keyIt;
|
|
||||||
if (m_keyIt != m_valIt->second.end()) { // Valid iterator, done.
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// Try the next value?
|
|
||||||
++m_valIt;
|
|
||||||
if (m_valIt == m_sbvmp->m_vals.end()) { // No more values
|
|
||||||
m_end = true;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// Should find a value here, as every value bucket is supposed
|
|
||||||
// to have at least one key, even after keys get removed.
|
|
||||||
m_keyIt = m_valIt->second.begin();
|
|
||||||
UASSERT(m_keyIt != m_valIt->second.end(), "Algorithm should have key left");
|
|
||||||
}
|
|
||||||
void reverseUntilValid() {
|
|
||||||
if (m_end) {
|
|
||||||
UASSERT(!m_sbvmp->m_vals.empty(), "Reverse iterator causes underflow");
|
|
||||||
m_valIt = m_sbvmp->m_vals.end();
|
|
||||||
--m_valIt;
|
|
||||||
|
|
||||||
UASSERT(!m_valIt->second.empty(), "Reverse iterator causes underflow");
|
|
||||||
m_keyIt = m_valIt->second.end();
|
|
||||||
--m_keyIt;
|
|
||||||
|
|
||||||
m_end = false;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (m_keyIt != m_valIt->second.begin()) {
|
|
||||||
// Valid iterator, we're done.
|
|
||||||
--m_keyIt;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// Try the previous value?
|
|
||||||
if (VL_UNCOVERABLE(m_valIt == m_sbvmp->m_vals.begin())) {
|
|
||||||
// No more values but it's not defined to decrement an
|
|
||||||
// iterator past the beginning.
|
|
||||||
v3fatalSrc("Decremented iterator past beginning");
|
|
||||||
return; // LCOV_EXCL_LINE
|
|
||||||
}
|
|
||||||
--m_valIt;
|
|
||||||
// Should find a value here, as Every value bucket is supposed
|
|
||||||
// to have at least one key, even after keys get removed.
|
|
||||||
UASSERT(!m_valIt->second.empty(), "Value bucket should have key");
|
|
||||||
m_keyIt = m_valIt->second.end();
|
|
||||||
--m_keyIt;
|
|
||||||
UASSERT(m_keyIt != m_valIt->second.end(), "Value bucket should have key");
|
|
||||||
}
|
|
||||||
|
|
||||||
public:
|
|
||||||
const T_Key& key() const { return *m_keyIt; }
|
|
||||||
const T_Value& value() const { return m_valIt->first; }
|
|
||||||
const_iterator& operator++() {
|
|
||||||
advanceUntilValid();
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
const_iterator& operator--() {
|
|
||||||
reverseUntilValid();
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
bool operator==(const const_iterator& other) const {
|
|
||||||
// It's not legal to compare iterators from different
|
|
||||||
// sequences. So check m_end before comparing m_valIt, and
|
|
||||||
// compare m_valIt's before comparing m_keyIt to ensure nothing
|
|
||||||
// here is undefined.
|
|
||||||
if (m_end || other.m_end) return m_end && other.m_end;
|
|
||||||
return ((m_valIt == other.m_valIt) && (m_keyIt == other.m_keyIt));
|
|
||||||
}
|
|
||||||
bool operator!=(const const_iterator& other) const { return (!this->operator==(other)); }
|
|
||||||
|
|
||||||
// WARNING: Cleverness.
|
|
||||||
//
|
|
||||||
// The "reference" returned by *it must remain valid after 'it'
|
|
||||||
// gets destroyed. The reverse_iterator relies on this for its
|
|
||||||
// operator*(), so it's not just a theoretical requirement, it's a
|
|
||||||
// real requirement.
|
|
||||||
//
|
|
||||||
// To make that work, define the "reference" type to be the
|
|
||||||
// iterator itself. So clients can do (*it).key() and
|
|
||||||
// (*it).value(). This is the clever part.
|
|
||||||
//
|
|
||||||
// That's mostly useful for a reverse iterator, where *rit returns
|
|
||||||
// the forward iterator pointing the to same element, so
|
|
||||||
// (*rit).key() and (*rit).value() work where rit.key() and
|
|
||||||
// rit.value() cannot.
|
|
||||||
//
|
|
||||||
// It would be nice to support it->key() and it->value(), however
|
|
||||||
// uncertain what would be an appropriate 'pointer' type define
|
|
||||||
// that makes this work safely through a reverse iterator. So this
|
|
||||||
// class does not provide an operator->().
|
|
||||||
//
|
|
||||||
// Q) Why not make our value_type be a pair<T_Key, T_Value> like a
|
|
||||||
// normal map, and return a reference to that? This could
|
|
||||||
// return a reference to one of the pairs inside m_keys, that
|
|
||||||
// would satisfy the constraint above.
|
|
||||||
//
|
|
||||||
// A) It would take a lookup to find that pair within m_keys. This
|
|
||||||
// iterator is designed to minimize the number of hashtable and
|
|
||||||
// tree lookups. Increment, decrement, key(), value(), erase()
|
|
||||||
// by iterator, begin(), end() -- none of these require a
|
|
||||||
// container lookup. That's true for reverse_iterators too.
|
|
||||||
reference operator*() const {
|
|
||||||
UASSERT(!m_end, "Dereferencing iterator that is at end()");
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
class iterator final : public const_iterator {
|
|
||||||
public:
|
|
||||||
// TYPES
|
|
||||||
using value_type = iterator;
|
|
||||||
using reference = iterator;
|
|
||||||
// pointer, difference_type, and iterator_category inherit from
|
|
||||||
// const_iterator
|
|
||||||
|
|
||||||
// CONSTRUCTORS
|
|
||||||
explicit iterator(SortByValueMap* sbvmp)
|
|
||||||
: const_iterator{sbvmp} {}
|
|
||||||
iterator(typename Val2Keys::iterator valIt, typename KeySet::iterator keyIt,
|
|
||||||
SortByValueMap* sbvmp)
|
|
||||||
: const_iterator{valIt, keyIt, sbvmp} {}
|
|
||||||
|
|
||||||
// METHODS
|
|
||||||
iterator& operator++() {
|
|
||||||
this->advanceUntilValid();
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
iterator& operator--() {
|
|
||||||
this->reverseUntilValid();
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
reference operator*() const {
|
|
||||||
UASSERT(!this->m_end, "Dereferencing iterator that is at end()");
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
using reverse_iterator = std::reverse_iterator<iterator>;
|
|
||||||
using const_reverse_iterator = std::reverse_iterator<const_iterator>;
|
|
||||||
|
|
||||||
// METHODS
|
|
||||||
private:
|
|
||||||
void removeKeyFromOldVal(const T_Key& k, const T_Value& oldVal) {
|
|
||||||
// The value of 'k' is about to change, or, 'k' is about to be
|
|
||||||
// removed from the map.
|
|
||||||
// Clear the m_vals mapping for k.
|
|
||||||
KeySet& keysAtOldVal = m_vals[oldVal];
|
|
||||||
const size_t erased = keysAtOldVal.erase(k);
|
|
||||||
UASSERT(erased == 1, "removeKeyFromOldVal() removal key not found");
|
|
||||||
if (keysAtOldVal.empty()) {
|
|
||||||
// Don't keep empty sets in the value map.
|
|
||||||
m_vals.erase(oldVal);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
void removeKeyFromOldVal(iterator it) {
|
size_t erase(const T_Key& key) {
|
||||||
it.m_valIt->second.erase(it.m_keyIt);
|
const auto kiIt = m_kiMap.find(key);
|
||||||
if (it.m_valIt->second.empty()) m_vals.erase(it.m_valIt);
|
if (kiIt == m_kiMap.end()) return 0;
|
||||||
}
|
m_pairs.erase(kiIt->second);
|
||||||
|
m_kiMap.erase(kiIt);
|
||||||
public:
|
|
||||||
iterator begin() {
|
|
||||||
const auto valIt = m_vals.begin();
|
|
||||||
if (valIt == m_vals.end()) return end();
|
|
||||||
const auto keyIt = valIt->second.begin();
|
|
||||||
return iterator(valIt, keyIt, this);
|
|
||||||
}
|
|
||||||
const_iterator begin() const {
|
|
||||||
SortByValueMap* const mutp = const_cast<SortByValueMap*>(this);
|
|
||||||
const auto valIt = mutp->m_vals.begin();
|
|
||||||
if (valIt == mutp->m_vals.end()) return end();
|
|
||||||
const auto keyIt = valIt->second.begin();
|
|
||||||
return const_iterator(valIt, keyIt, mutp);
|
|
||||||
}
|
|
||||||
iterator end() { return iterator(this); }
|
|
||||||
const_iterator end() const {
|
|
||||||
// Safe to cast away const; the const_iterator will still enforce
|
|
||||||
// it. Same for the const begin() below.
|
|
||||||
return const_iterator(const_cast<SortByValueMap*>(this));
|
|
||||||
}
|
|
||||||
reverse_iterator rbegin() { return reverse_iterator(end()); }
|
|
||||||
reverse_iterator rend() { return reverse_iterator(begin()); }
|
|
||||||
const_reverse_iterator rbegin() const { return const_reverse_iterator(end()); }
|
|
||||||
const_reverse_iterator rend() const { return const_reverse_iterator(begin()); }
|
|
||||||
|
|
||||||
iterator find(const T_Key& k) {
|
|
||||||
const auto kvit = m_keys.find(k);
|
|
||||||
if (kvit == m_keys.end()) return end();
|
|
||||||
|
|
||||||
const auto valIt = m_vals.find(kvit->second);
|
|
||||||
const auto keyIt = valIt->second.find(k);
|
|
||||||
return iterator(valIt, keyIt, this);
|
|
||||||
}
|
|
||||||
const_iterator find(const T_Key& k) const {
|
|
||||||
SortByValueMap* const mutp = const_cast<SortByValueMap*>(this);
|
|
||||||
const auto kvit = mutp->m_keys.find(k);
|
|
||||||
if (kvit == mutp->m_keys.end()) return end();
|
|
||||||
|
|
||||||
const auto valIt = mutp->m_vals.find(kvit->second);
|
|
||||||
const auto keyIt = valIt->second.find(k);
|
|
||||||
return const_iterator(valIt, keyIt, mutp);
|
|
||||||
}
|
|
||||||
void set(const T_Key& k, const T_Value& v) {
|
|
||||||
const auto kvit = m_keys.find(k);
|
|
||||||
if (kvit != m_keys.end()) {
|
|
||||||
if (kvit->second == v) {
|
|
||||||
return; // LCOV_EXCL_LINE // Same value already present; stop.
|
|
||||||
}
|
|
||||||
// Must remove element from m_vals[oldValue]
|
|
||||||
removeKeyFromOldVal(k, kvit->second);
|
|
||||||
}
|
|
||||||
m_keys[k] = v;
|
|
||||||
m_vals[v].insert(k);
|
|
||||||
}
|
|
||||||
size_t erase(const T_Key& k) {
|
|
||||||
const auto kvit = m_keys.find(k);
|
|
||||||
if (kvit == m_keys.end()) return 0;
|
|
||||||
removeKeyFromOldVal(k, kvit->second);
|
|
||||||
m_keys.erase(kvit);
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
void erase(const iterator& it) {
|
void erase(const_iterator it) {
|
||||||
m_keys.erase(it.key());
|
m_kiMap.erase(it->first);
|
||||||
removeKeyFromOldVal(it);
|
m_pairs.erase(it);
|
||||||
}
|
}
|
||||||
void erase(const reverse_iterator& it) {
|
void erase(const_reverse_iterator rit) {
|
||||||
erase(*it); // Dereferencing returns a copy of the forward iterator
|
m_kiMap.erase(rit->first);
|
||||||
|
m_pairs.erase(std::next(rit).base());
|
||||||
}
|
}
|
||||||
bool has(const T_Key& k) const { return (m_keys.find(k) != m_keys.end()); }
|
bool has(const T_Key& key) const { return m_kiMap.count(key); }
|
||||||
bool empty() const { return m_keys.empty(); }
|
bool empty() const { return m_pairs.empty(); }
|
||||||
// Look up a value. Returns a reference for efficiency. Note this must
|
// Returns const reference.
|
||||||
// be a const reference, otherwise the client could corrupt the sorted
|
const T_Value& at(const T_Key& key) const { return m_kiMap.at(key)->second; }
|
||||||
// order of m_byValue by reaching through and changing the value.
|
// Note this returns const_iterator
|
||||||
const T_Value& at(const T_Key& k) const {
|
template <typename... Args> //
|
||||||
const auto kvit = m_keys.find(k);
|
std::pair<const_iterator, bool> emplace(const T_Key& key, Args&&... args) {
|
||||||
UASSERT(kvit != m_keys.end(), "at() lookup key not found");
|
const auto kiEmp = m_kiMap.emplace(key, end());
|
||||||
return kvit->second;
|
if (kiEmp.second) {
|
||||||
|
const auto result = m_pairs.emplace(key, std::forward<Args>(args)...);
|
||||||
|
#if VL_DEBUG
|
||||||
|
UASSERT(result.second, "Should not be in set yet");
|
||||||
|
#endif
|
||||||
|
kiEmp.first->second = result.first;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
return {kiEmp.first->second, false};
|
||||||
|
}
|
||||||
|
// Invalidates iterators
|
||||||
|
void update(const_iterator it, T_Value value) {
|
||||||
|
const auto kiIt = m_kiMap.find(it->first);
|
||||||
|
m_pairs.erase(it);
|
||||||
|
kiIt->second = m_pairs.emplace(kiIt->first, value).first;
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
|
||||||
VL_UNCOPYABLE(SortByValueMap);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
//######################################################################
|
//######################################################################
|
||||||
@ -333,7 +138,7 @@ private:
|
|||||||
/// V3Scoreboard takes a set of Elem*'s, each having some score.
|
/// V3Scoreboard takes a set of Elem*'s, each having some score.
|
||||||
/// Scores are assigned by a user-supplied scoring function.
|
/// Scores are assigned by a user-supplied scoring function.
|
||||||
///
|
///
|
||||||
/// At any time, the V3Scoreboard can return the elem with the "best" score
|
/// At any time, the V3Scoreboard can return th515e elem with the "best" score
|
||||||
/// among those elements whose scores are known.
|
/// among those elements whose scores are known.
|
||||||
///
|
///
|
||||||
/// The best score is the _lowest_ score. This makes sense in contexts
|
/// The best score is the _lowest_ score. This makes sense in contexts
|
||||||
@ -418,9 +223,9 @@ public:
|
|||||||
// reflected in the result of bestp(). Otherwise, bestp() only
|
// reflected in the result of bestp(). Otherwise, bestp() only
|
||||||
// considers elements that aren't pending rescore.
|
// considers elements that aren't pending rescore.
|
||||||
const T_Elem* bestp() {
|
const T_Elem* bestp() {
|
||||||
const auto result = m_sorted.begin();
|
const auto it = m_sorted.begin();
|
||||||
if (VL_UNLIKELY(result == m_sorted.end())) return nullptr;
|
if (VL_UNLIKELY(it == m_sorted.end())) return nullptr;
|
||||||
return (*result).key();
|
return it->first;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Tell the scoreboard that this element's score may have changed.
|
// Tell the scoreboard that this element's score may have changed.
|
||||||
@ -444,20 +249,18 @@ public:
|
|||||||
bool needsRescore() { return !m_unknown.empty(); }
|
bool needsRescore() { return !m_unknown.empty(); }
|
||||||
// False if elp's score is known to V3Scoreboard,
|
// False if elp's score is known to V3Scoreboard,
|
||||||
// else true if elp's score is unknown until the next rescore().
|
// else true if elp's score is unknown until the next rescore().
|
||||||
bool needsRescore(const T_Elem* elp) { return (m_unknown.find(elp) != m_unknown.end()); }
|
bool needsRescore(const T_Elem* elp) { return m_unknown.count(elp); }
|
||||||
// Retrieve the last known score for an element.
|
// Retrieve the last known score for an element.
|
||||||
T_Score cachedScore(const T_Elem* elp) {
|
T_Score cachedScore(const T_Elem* elp) { return m_sorted.at(elp); }
|
||||||
const auto result = m_sorted.find(elp);
|
|
||||||
UASSERT(result != m_sorted.end(), "V3Scoreboard::cachedScore() failed to find element");
|
|
||||||
return (*result).value();
|
|
||||||
}
|
|
||||||
// For each element whose score is unknown to V3Scoreboard,
|
// For each element whose score is unknown to V3Scoreboard,
|
||||||
// call the client's scoring function to get a new score,
|
// call the client's scoring function to get a new score,
|
||||||
// and sort all elements by their current score.
|
// and sort all elements by their current score.
|
||||||
void rescore() {
|
void rescore() {
|
||||||
for (const T_Elem* elp : m_unknown) {
|
for (const T_Elem* elp : m_unknown) {
|
||||||
const T_Score sortScore = m_scoreFnp(elp);
|
VL_ATTR_UNUSED const bool exists = !m_sorted.emplace(elp, m_scoreFnp(elp)).second;
|
||||||
m_sorted.set(elp, sortScore);
|
#if VL_DEBUG
|
||||||
|
UASSERT(!exists, "Should not be in both m_unknown and m_sorted");
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
m_unknown.clear();
|
m_unknown.clear();
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user