begin refactoring SkTDynamicHash and SkTMultiMap
The biggest mismatch between these and SkTHash{Map,Set,Table} is the old ones provide Iter and ConstIter, the new ones foreach(). This CL, - adds foreach() methods to the old types, - replaces all uses of ConstIter with foreach(), - replaces most uses of Iter with foreach(), I'm leaving one spot using Iter to walk the table and remove its elements for its own CL... it'll be a little more complicated to get that right. From there it should be straightforward to turn SkTDynamicHash into a thin wrapper over an SkTHashTable. Bugs: skia:9703 Change-Id: Ia6ba87c35b89585c42b5b9f118f4cbf3abd04f0d Reviewed-on: https://skia-review.googlesource.com/c/skia/+/277098 Reviewed-by: Herb Derby <herb@google.com> Commit-Queue: Mike Klein <mtklein@google.com>
This commit is contained in:
parent
16701eecd5
commit
cff6396875
@ -151,6 +151,7 @@ public:
|
|||||||
// around to check slot i again.
|
// around to check slot i again.
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// TODO: shrink table capacity like in remove()?
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -32,13 +32,7 @@ public:
|
|||||||
typedef SkImageFilterCacheKey Key;
|
typedef SkImageFilterCacheKey Key;
|
||||||
CacheImpl(size_t maxBytes) : fMaxBytes(maxBytes), fCurrentBytes(0) { }
|
CacheImpl(size_t maxBytes) : fMaxBytes(maxBytes), fCurrentBytes(0) { }
|
||||||
~CacheImpl() override {
|
~CacheImpl() override {
|
||||||
SkTDynamicHash<Value, Key>::Iter iter(&fLookup);
|
fLookup.foreach([&](Value* v) { delete v; });
|
||||||
|
|
||||||
while (!iter.done()) {
|
|
||||||
Value* v = &*iter;
|
|
||||||
++iter;
|
|
||||||
delete v;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
struct Value {
|
struct Value {
|
||||||
Value(const Key& key, const skif::FilterResult<For::kOutput>& image,
|
Value(const Key& key, const skif::FilterResult<For::kOutput>& image,
|
||||||
|
@ -30,59 +30,21 @@ public:
|
|||||||
sk_free(fArray);
|
sk_free(fArray);
|
||||||
}
|
}
|
||||||
|
|
||||||
class Iter {
|
// Call fn on every entry in the table. You may mutate the entries, but be very careful.
|
||||||
public:
|
template <typename Fn> // f(T*)
|
||||||
explicit Iter(SkTDynamicHash* hash) : fHash(hash), fCurrentIndex(-1) {
|
void foreach(Fn&& fn) {
|
||||||
SkASSERT(hash);
|
for (Iter it(this); !it.done(); ++it) {
|
||||||
++(*this);
|
fn(&*it);
|
||||||
}
|
}
|
||||||
bool done() const {
|
|
||||||
SkASSERT(fCurrentIndex <= fHash->fCapacity);
|
|
||||||
return fCurrentIndex == fHash->fCapacity;
|
|
||||||
}
|
|
||||||
T& operator*() const {
|
|
||||||
SkASSERT(!this->done());
|
|
||||||
return *this->current();
|
|
||||||
}
|
|
||||||
void operator++() {
|
|
||||||
do {
|
|
||||||
fCurrentIndex++;
|
|
||||||
} while (!this->done() && (this->current() == Empty() || this->current() == Deleted()));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
// Call fn on every entry in the table. You may not mutate anything.
|
||||||
T* current() const { return fHash->fArray[fCurrentIndex]; }
|
template <typename Fn> // f(T) or f(const T&)
|
||||||
|
void foreach(Fn&& fn) const {
|
||||||
SkTDynamicHash* fHash;
|
for (ConstIter it(this); !it.done(); ++it) {
|
||||||
int fCurrentIndex;
|
fn(*it);
|
||||||
};
|
|
||||||
|
|
||||||
class ConstIter {
|
|
||||||
public:
|
|
||||||
explicit ConstIter(const SkTDynamicHash* hash) : fHash(hash), fCurrentIndex(-1) {
|
|
||||||
SkASSERT(hash);
|
|
||||||
++(*this);
|
|
||||||
}
|
}
|
||||||
bool done() const {
|
|
||||||
SkASSERT(fCurrentIndex <= fHash->fCapacity);
|
|
||||||
return fCurrentIndex == fHash->fCapacity;
|
|
||||||
}
|
}
|
||||||
const T& operator*() const {
|
|
||||||
SkASSERT(!this->done());
|
|
||||||
return *this->current();
|
|
||||||
}
|
|
||||||
void operator++() {
|
|
||||||
do {
|
|
||||||
fCurrentIndex++;
|
|
||||||
} while (!this->done() && (this->current() == Empty() || this->current() == Deleted()));
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
const T* current() const { return fHash->fArray[fCurrentIndex]; }
|
|
||||||
|
|
||||||
const SkTDynamicHash* fHash;
|
|
||||||
int fCurrentIndex;
|
|
||||||
};
|
|
||||||
|
|
||||||
int count() const { return fCount; }
|
int count() const { return fCount; }
|
||||||
|
|
||||||
@ -160,6 +122,66 @@ private:
|
|||||||
static T* Empty() { return reinterpret_cast<T*>(0); } // i.e. nullptr
|
static T* Empty() { return reinterpret_cast<T*>(0); } // i.e. nullptr
|
||||||
static T* Deleted() { return reinterpret_cast<T*>(1); } // Also an invalid pointer.
|
static T* Deleted() { return reinterpret_cast<T*>(1); } // Also an invalid pointer.
|
||||||
|
|
||||||
|
// GrProxyProvider::removeAllUniqueKeys() uses Iter but removes elements from the table
|
||||||
|
// as it iterates, so it can't safely use foreach().
|
||||||
|
// TODO(mtklein): add mutate(), replace that use of Iter with that.
|
||||||
|
friend class GrProxyProvider;
|
||||||
|
|
||||||
|
class Iter {
|
||||||
|
public:
|
||||||
|
explicit Iter(SkTDynamicHash* hash) : fHash(hash), fCurrentIndex(-1) {
|
||||||
|
SkASSERT(hash);
|
||||||
|
++(*this);
|
||||||
|
}
|
||||||
|
bool done() const {
|
||||||
|
SkASSERT(fCurrentIndex <= fHash->fCapacity);
|
||||||
|
return fCurrentIndex == fHash->fCapacity;
|
||||||
|
}
|
||||||
|
T& operator*() const {
|
||||||
|
SkASSERT(!this->done());
|
||||||
|
return *this->current();
|
||||||
|
}
|
||||||
|
void operator++() {
|
||||||
|
do {
|
||||||
|
fCurrentIndex++;
|
||||||
|
} while (!this->done() && (this->current() == Empty() || this->current() == Deleted()));
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
T* current() const { return fHash->fArray[fCurrentIndex]; }
|
||||||
|
|
||||||
|
SkTDynamicHash* fHash;
|
||||||
|
int fCurrentIndex;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
class ConstIter {
|
||||||
|
public:
|
||||||
|
explicit ConstIter(const SkTDynamicHash* hash) : fHash(hash), fCurrentIndex(-1) {
|
||||||
|
SkASSERT(hash);
|
||||||
|
++(*this);
|
||||||
|
}
|
||||||
|
bool done() const {
|
||||||
|
SkASSERT(fCurrentIndex <= fHash->fCapacity);
|
||||||
|
return fCurrentIndex == fHash->fCapacity;
|
||||||
|
}
|
||||||
|
const T& operator*() const {
|
||||||
|
SkASSERT(!this->done());
|
||||||
|
return *this->current();
|
||||||
|
}
|
||||||
|
void operator++() {
|
||||||
|
do {
|
||||||
|
fCurrentIndex++;
|
||||||
|
} while (!this->done() && (this->current() == Empty() || this->current() == Deleted()));
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
const T* current() const { return fHash->fArray[fCurrentIndex]; }
|
||||||
|
|
||||||
|
const SkTDynamicHash* fHash;
|
||||||
|
int fCurrentIndex;
|
||||||
|
};
|
||||||
|
|
||||||
bool validate() const {
|
bool validate() const {
|
||||||
#define SKTDYNAMICHASH_CHECK(x) SkASSERT(x); if (!(x)) return false
|
#define SKTDYNAMICHASH_CHECK(x) SkASSERT(x); if (!(x)) return false
|
||||||
static const int kLarge = 50; // Arbitrary, tweak to suit your patience.
|
static const int kLarge = 50; // Arbitrary, tweak to suit your patience.
|
||||||
|
@ -30,15 +30,14 @@ public:
|
|||||||
SkTMultiMap() : fCount(0) {}
|
SkTMultiMap() : fCount(0) {}
|
||||||
|
|
||||||
~SkTMultiMap() {
|
~SkTMultiMap() {
|
||||||
typename SkTDynamicHash<ValueList, Key>::Iter iter(&fHash);
|
fHash.foreach([&](ValueList* vl) {
|
||||||
for ( ; !iter.done(); ++iter) {
|
|
||||||
ValueList* next;
|
ValueList* next;
|
||||||
for (ValueList* cur = &(*iter); cur; cur = next) {
|
for (ValueList* it = vl; it; it = next) {
|
||||||
HashTraits::OnFree(cur->fValue);
|
HashTraits::OnFree(it->fValue);
|
||||||
next = cur->fNext;
|
next = it->fNext;
|
||||||
delete cur;
|
delete it;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
void insert(const Key& key, T* value) {
|
void insert(const Key& key, T* value) {
|
||||||
@ -127,42 +126,15 @@ public:
|
|||||||
int count() const { return fCount; }
|
int count() const { return fCount; }
|
||||||
|
|
||||||
#ifdef SK_DEBUG
|
#ifdef SK_DEBUG
|
||||||
class ConstIter {
|
template <typename Fn> // f(T) or f(const T&)
|
||||||
public:
|
void foreach(Fn&& fn) const {
|
||||||
explicit ConstIter(const SkTMultiMap* mmap)
|
fHash.foreach([&](const ValueList& vl) {
|
||||||
: fIter(&(mmap->fHash))
|
for (const ValueList* it = &vl; it; it = it->fNext) {
|
||||||
, fList(nullptr) {
|
fn(*it->fValue);
|
||||||
if (!fIter.done()) {
|
|
||||||
fList = &(*fIter);
|
|
||||||
}
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
bool done() const {
|
|
||||||
return fIter.done();
|
|
||||||
}
|
|
||||||
|
|
||||||
const T* operator*() {
|
|
||||||
SkASSERT(fList);
|
|
||||||
return fList->fValue;
|
|
||||||
}
|
|
||||||
|
|
||||||
void operator++() {
|
|
||||||
if (fList) {
|
|
||||||
fList = fList->fNext;
|
|
||||||
}
|
|
||||||
if (!fList) {
|
|
||||||
++fIter;
|
|
||||||
if (!fIter.done()) {
|
|
||||||
fList = &(*fIter);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
typename SkTDynamicHash<ValueList, Key>::ConstIter fIter;
|
|
||||||
const ValueList* fList;
|
|
||||||
};
|
|
||||||
|
|
||||||
bool has(const T* value, const Key& key) const {
|
bool has(const T* value, const Key& key) const {
|
||||||
for (ValueList* list = fHash.find(key); list; list = list->fNext) {
|
for (ValueList* list = fHash.find(key); list; list = list->fNext) {
|
||||||
if (list->fValue == value) {
|
if (list->fValue == value) {
|
||||||
|
@ -891,12 +891,9 @@ bool GrProxyProvider::isAbandoned() const {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void GrProxyProvider::orphanAllUniqueKeys() {
|
void GrProxyProvider::orphanAllUniqueKeys() {
|
||||||
UniquelyKeyedProxyHash::Iter iter(&fUniquelyKeyedProxies);
|
fUniquelyKeyedProxies.foreach([&](GrTextureProxy* proxy){
|
||||||
for (UniquelyKeyedProxyHash::Iter iter(&fUniquelyKeyedProxies); !iter.done(); ++iter) {
|
proxy->fProxyProvider = nullptr;
|
||||||
GrTextureProxy& tmp = *iter;
|
});
|
||||||
|
|
||||||
tmp.fProxyProvider = nullptr;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void GrProxyProvider::removeAllUniqueKeys() {
|
void GrProxyProvider::removeAllUniqueKeys() {
|
||||||
|
@ -863,16 +863,13 @@ void GrResourceCache::validate() const {
|
|||||||
};
|
};
|
||||||
|
|
||||||
{
|
{
|
||||||
ScratchMap::ConstIter iter(&fScratchMap);
|
|
||||||
|
|
||||||
int count = 0;
|
int count = 0;
|
||||||
for ( ; !iter.done(); ++iter) {
|
fScratchMap.foreach([&](const GrGpuResource& resource) {
|
||||||
const GrGpuResource* resource = *iter;
|
SkASSERT(resource.resourcePriv().getScratchKey().isValid());
|
||||||
SkASSERT(resource->resourcePriv().getScratchKey().isValid());
|
SkASSERT(!resource.getUniqueKey().isValid());
|
||||||
SkASSERT(!resource->getUniqueKey().isValid());
|
|
||||||
count++;
|
count++;
|
||||||
}
|
});
|
||||||
SkASSERT(count == fScratchMap.count()); // ensure the iterator is working correctly
|
SkASSERT(count == fScratchMap.count());
|
||||||
}
|
}
|
||||||
|
|
||||||
Stats stats(this);
|
Stats stats(this);
|
||||||
|
@ -70,18 +70,10 @@ GrMtlSampler* GrMtlResourceProvider::findOrCreateCompatibleSampler(GrSamplerStat
|
|||||||
}
|
}
|
||||||
|
|
||||||
void GrMtlResourceProvider::destroyResources() {
|
void GrMtlResourceProvider::destroyResources() {
|
||||||
// Iterate through all stored GrMtlSamplers and unref them before resetting the hash.
|
fSamplers.foreach([&](GrMtlSampler* sampler) { sampler->unref(); });
|
||||||
SkTDynamicHash<GrMtlSampler, GrMtlSampler::Key>::Iter samplerIter(&fSamplers);
|
|
||||||
for (; !samplerIter.done(); ++samplerIter) {
|
|
||||||
(*samplerIter).unref();
|
|
||||||
}
|
|
||||||
fSamplers.reset();
|
fSamplers.reset();
|
||||||
|
|
||||||
// Iterate through all stored GrMtlDepthStencils and unref them before resetting the hash.
|
fDepthStencilStates.foreach([&](GrMtlDepthStencil* stencil) { stencil->unref(); });
|
||||||
SkTDynamicHash<GrMtlDepthStencil, GrMtlDepthStencil::Key>::Iter dsIter(&fDepthStencilStates);
|
|
||||||
for (; !dsIter.done(); ++dsIter) {
|
|
||||||
(*dsIter).unref();
|
|
||||||
}
|
|
||||||
fDepthStencilStates.reset();
|
fDepthStencilStates.reset();
|
||||||
|
|
||||||
fPipelineStateCache->release();
|
fPipelineStateCache->release();
|
||||||
|
@ -392,14 +392,10 @@ void GrVkResourceProvider::destroyResources(bool deviceLost) {
|
|||||||
fExternalRenderPasses.reset();
|
fExternalRenderPasses.reset();
|
||||||
|
|
||||||
// Iterate through all store GrVkSamplers and unref them before resetting the hash.
|
// Iterate through all store GrVkSamplers and unref them before resetting the hash.
|
||||||
for (decltype(fSamplers)::Iter iter(&fSamplers); !iter.done(); ++iter) {
|
fSamplers.foreach([&](auto* elt) { elt->unref(); });
|
||||||
(*iter).unref();
|
|
||||||
}
|
|
||||||
fSamplers.reset();
|
fSamplers.reset();
|
||||||
|
|
||||||
for (decltype(fYcbcrConversions)::Iter iter(&fYcbcrConversions); !iter.done(); ++iter) {
|
fYcbcrConversions.foreach([&](auto* elt) { elt->unref(); });
|
||||||
(*iter).unref();
|
|
||||||
}
|
|
||||||
fYcbcrConversions.reset();
|
fYcbcrConversions.reset();
|
||||||
|
|
||||||
fPipelineStateCache->release();
|
fPipelineStateCache->release();
|
||||||
|
@ -171,59 +171,6 @@ DEF_TEST(DynamicHash_remove, reporter) {
|
|||||||
ASSERT(hash.find(5)->value == 3.0);
|
ASSERT(hash.find(5)->value == 3.0);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename T> static void TestIter(skiatest::Reporter* reporter) {
|
|
||||||
Hash hash;
|
|
||||||
|
|
||||||
int count = 0;
|
|
||||||
// this should fall out of loop immediately
|
|
||||||
for (T iter(&hash); !iter.done(); ++iter) {
|
|
||||||
++count;
|
|
||||||
}
|
|
||||||
ASSERT(0 == count);
|
|
||||||
|
|
||||||
// These collide.
|
|
||||||
Entry a = { 1, 2.0 };
|
|
||||||
Entry b = { 5, 3.0 };
|
|
||||||
Entry c = { 9, 4.0 };
|
|
||||||
|
|
||||||
hash.add(&a);
|
|
||||||
hash.add(&b);
|
|
||||||
hash.add(&c);
|
|
||||||
|
|
||||||
// should see all 3 unique keys when iterating over hash
|
|
||||||
count = 0;
|
|
||||||
int keys[3] = {0, 0, 0};
|
|
||||||
for (T iter(&hash); !iter.done(); ++iter) {
|
|
||||||
int key = (*iter).key;
|
|
||||||
keys[count] = key;
|
|
||||||
ASSERT(hash.find(key) != nullptr);
|
|
||||||
++count;
|
|
||||||
}
|
|
||||||
ASSERT(3 == count);
|
|
||||||
ASSERT(keys[0] != keys[1]);
|
|
||||||
ASSERT(keys[0] != keys[2]);
|
|
||||||
ASSERT(keys[1] != keys[2]);
|
|
||||||
|
|
||||||
// should see 2 unique keys when iterating over hash that aren't 1
|
|
||||||
hash.remove(1);
|
|
||||||
count = 0;
|
|
||||||
memset(keys, 0, sizeof(keys));
|
|
||||||
for (T iter(&hash); !iter.done(); ++iter) {
|
|
||||||
int key = (*iter).key;
|
|
||||||
keys[count] = key;
|
|
||||||
ASSERT(key != 1);
|
|
||||||
ASSERT(hash.find(key) != nullptr);
|
|
||||||
++count;
|
|
||||||
}
|
|
||||||
ASSERT(2 == count);
|
|
||||||
ASSERT(keys[0] != keys[1]);
|
|
||||||
}
|
|
||||||
|
|
||||||
DEF_TEST(DynamicHash_iterator, reporter) {
|
|
||||||
TestIter<Hash::Iter>(reporter);
|
|
||||||
TestIter<Hash::ConstIter>(reporter);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void TestResetOrRewind(skiatest::Reporter* reporter, bool testReset) {
|
static void TestResetOrRewind(skiatest::Reporter* reporter, bool testReset) {
|
||||||
Hash hash;
|
Hash hash;
|
||||||
Entry a = { 1, 2.0 };
|
Entry a = { 1, 2.0 };
|
||||||
|
@ -35,13 +35,9 @@ SkString UrlDataManager::addData(SkData* data, const char* contentType) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void UrlDataManager::reset() {
|
void UrlDataManager::reset() {
|
||||||
SkTDynamicHash<UrlData, SkData, LookupTrait>::Iter iter(&fCache);
|
fCache.foreach([&](UrlData* urlData) {
|
||||||
while (!iter.done()) {
|
|
||||||
UrlData* urlData = &(*iter);
|
|
||||||
urlData->unref();
|
urlData->unref();
|
||||||
++iter;
|
});
|
||||||
}
|
|
||||||
|
|
||||||
fCache.rewind();
|
fCache.rewind();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -43,13 +43,11 @@ void GrResourceCache::changeTimestamp(uint32_t newTimestamp) { fTimestamp = newT
|
|||||||
#ifdef SK_DEBUG
|
#ifdef SK_DEBUG
|
||||||
int GrResourceCache::countUniqueKeysWithTag(const char* tag) const {
|
int GrResourceCache::countUniqueKeysWithTag(const char* tag) const {
|
||||||
int count = 0;
|
int count = 0;
|
||||||
UniqueHash::ConstIter iter(&fUniqueHash);
|
fUniqueHash.foreach([&](const GrGpuResource& resource){
|
||||||
while (!iter.done()) {
|
if (0 == strcmp(tag, resource.getUniqueKey().tag())) {
|
||||||
if (0 == strcmp(tag, (*iter).getUniqueKey().tag())) {
|
|
||||||
++count;
|
++count;
|
||||||
}
|
}
|
||||||
++iter;
|
});
|
||||||
}
|
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -148,13 +148,11 @@ bool Window_mac::attach(BackendType attachType) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void Window_mac::PaintWindows() {
|
void Window_mac::PaintWindows() {
|
||||||
SkTDynamicHash<Window_mac, NSInteger>::Iter iter(&gWindowMap);
|
gWindowMap.foreach([&](Window_mac* window) {
|
||||||
while (!iter.done()) {
|
if (window->fIsContentInvalidated) {
|
||||||
if ((*iter).fIsContentInvalidated) {
|
window->onPaint();
|
||||||
(*iter).onPaint();
|
|
||||||
}
|
|
||||||
++iter;
|
|
||||||
}
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace sk_app
|
} // namespace sk_app
|
||||||
|
Loading…
Reference in New Issue
Block a user