Tracker: Refactor IconCache to use BOpenHashTable.
This removes the last usage of the old OpenTracker OpenHashTable, and so it can now be removed. Change-Id: I7a7bceef1d3fc74c7fdfa7b079e53576452703dc Reviewed-on: https://review.haiku-os.org/c/haiku/+/2339 Reviewed-by: waddlesplash <waddlesplash@gmail.com> Reviewed-by: John Scipione <jscipione@gmail.com> Reviewed-by: Adrien Destugues <pulkomandy@gmail.com>
This commit is contained in:
parent
910651da23
commit
8eaf4427f0
@ -107,14 +107,14 @@ IconCacheEntry::IconCacheEntry()
|
||||
fHighlightedLargeIcon(NULL),
|
||||
fMiniIcon(NULL),
|
||||
fHighlightedMiniIcon(NULL),
|
||||
fAliasForIndex(-1)
|
||||
fAliasTo(NULL)
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
IconCacheEntry::~IconCacheEntry()
|
||||
{
|
||||
if (fAliasForIndex < 0) {
|
||||
if (fAliasTo == NULL) {
|
||||
delete fLargeIcon;
|
||||
delete fHighlightedLargeIcon;
|
||||
delete fMiniIcon;
|
||||
@ -126,7 +126,7 @@ IconCacheEntry::~IconCacheEntry()
|
||||
fMiniIcon = NULL;
|
||||
fHighlightedMiniIcon = NULL;
|
||||
}
|
||||
fAliasForIndex = -1;
|
||||
fAliasTo = NULL;
|
||||
}
|
||||
|
||||
|
||||
@ -427,7 +427,7 @@ IconCache::GetIconFromMetaMime(const char* fileType, IconDrawMode mode,
|
||||
PRINT_ADD_ITEM(
|
||||
("File %s; Line %d # adding entry as alias for type %s\n",
|
||||
__FILE__, __LINE__, fileType));
|
||||
entry = fSharedCache.AddItem(&aliasTo, fileType);
|
||||
entry = fSharedCache.AddItem(fileType);
|
||||
entry->SetAliasFor(&fSharedCache, aliasTo);
|
||||
}
|
||||
ASSERT(aliasTo->HaveIconBitmap(mode, size));
|
||||
@ -531,8 +531,7 @@ IconCache::GetIconFromFileTypes(ModelNodeLazyOpener* modelOpener,
|
||||
"preferredApp %s, type %s\n",
|
||||
__FILE__, __LINE__, nodePreferredApp, fileType));
|
||||
IconCacheEntry* aliasedEntry
|
||||
= fSharedCache.AddItem((SharedCacheEntry**)&entry, fileType,
|
||||
nodePreferredApp);
|
||||
= fSharedCache.AddItem(fileType, nodePreferredApp);
|
||||
aliasedEntry->SetAliasFor(&fSharedCache,
|
||||
(SharedCacheEntry*)entry);
|
||||
// OK to cast here, have a runtime check
|
||||
@ -840,8 +839,7 @@ IconCache::GetGenericIcon(AutoLock<SimpleIconCache>* sharedCacheLocker,
|
||||
__FILE__, __LINE__, model->PreferredAppSignature(),
|
||||
model->MimeType()));
|
||||
IconCacheEntry* aliasedEntry = fSharedCache.AddItem(
|
||||
(SharedCacheEntry**)&entry, model->MimeType(),
|
||||
model->PreferredAppSignature());
|
||||
model->MimeType(), model->PreferredAppSignature());
|
||||
aliasedEntry->SetAliasFor(&fSharedCache, (SharedCacheEntry*)entry);
|
||||
|
||||
source = kMetaMime;
|
||||
@ -1223,10 +1221,9 @@ IconCache::IconChanged(const char* mimeType, const char* appSignature)
|
||||
|
||||
entry = (SharedCacheEntry*)fSharedCache.ResolveIfAlias(entry);
|
||||
ASSERT(entry != NULL);
|
||||
int32 index = fSharedCache.EntryIndex(entry);
|
||||
|
||||
fNodeCache.RemoveAliasesTo(index);
|
||||
fSharedCache.RemoveAliasesTo(index);
|
||||
fNodeCache.RemoveAliasesTo(entry);
|
||||
fSharedCache.RemoveAliasesTo(entry);
|
||||
|
||||
fSharedCache.IconChanged(entry);
|
||||
}
|
||||
@ -1398,24 +1395,13 @@ IconCacheEntry::RetireIcons(BObjectList<BBitmap>* retiredBitmapList)
|
||||
// #pragma mark - SharedIconCache
|
||||
|
||||
|
||||
// In debug mode keep the hash table sizes small so that they grow a lot and
|
||||
// execercise the resizing code a lot. In release mode allocate them large
|
||||
// up-front for better performance
|
||||
SharedIconCache::SharedIconCache()
|
||||
:
|
||||
#if DEBUG
|
||||
SimpleIconCache("Shared icon cache aka \"The Dead-Locker\""),
|
||||
fElementArray(20),
|
||||
fHashTable(20),
|
||||
fRetiredBitmaps(20, true)
|
||||
#else
|
||||
SimpleIconCache("Tracker shared icon cache"),
|
||||
fElementArray(1024),
|
||||
fHashTable(1000),
|
||||
fHashTable(),
|
||||
fRetiredBitmaps(256, true)
|
||||
#endif
|
||||
{
|
||||
fHashTable.SetElementVector(&fElementArray);
|
||||
fHashTable.Init(256);
|
||||
}
|
||||
|
||||
|
||||
@ -1445,27 +1431,8 @@ SharedIconCache::FindItem(const char* fileType,
|
||||
if (!fileType)
|
||||
fileType = B_FILE_MIMETYPE;
|
||||
|
||||
SharedCacheEntry* result
|
||||
= fHashTable.FindFirst(SharedCacheEntry::Hash(fileType,
|
||||
appSignature));
|
||||
|
||||
if (result == NULL)
|
||||
return NULL;
|
||||
|
||||
for(;;) {
|
||||
if (result->fFileType == fileType
|
||||
&& result->fAppSignature == appSignature) {
|
||||
return result;
|
||||
}
|
||||
|
||||
if (result->fNext < 0)
|
||||
break;
|
||||
|
||||
result = const_cast<SharedCacheEntry*>(&fElementArray.At(
|
||||
result->fNext));
|
||||
}
|
||||
|
||||
return NULL;
|
||||
return fHashTable.Lookup(SharedCacheEntry::TypeAndSignature(fileType,
|
||||
appSignature));
|
||||
}
|
||||
|
||||
|
||||
@ -1476,31 +1443,12 @@ SharedIconCache::AddItem(const char* fileType, const char* appSignature)
|
||||
if (fileType == NULL)
|
||||
fileType = B_FILE_MIMETYPE;
|
||||
|
||||
SharedCacheEntry* result = fHashTable.Add(SharedCacheEntry::Hash(fileType,
|
||||
appSignature));
|
||||
result->SetTo(fileType, appSignature);
|
||||
SharedCacheEntry* entry = new SharedCacheEntry(fileType, appSignature);
|
||||
if (fHashTable.Insert(entry) == B_OK)
|
||||
return entry;
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
|
||||
SharedCacheEntry*
|
||||
SharedIconCache::AddItem(SharedCacheEntry** outstandingEntry,
|
||||
const char* fileType, const char* appSignature)
|
||||
{
|
||||
int32 entryToken = fHashTable.ElementIndex(*outstandingEntry);
|
||||
ASSERT(entryToken >= 0);
|
||||
|
||||
ASSERT(fileType != NULL);
|
||||
if (fileType == NULL)
|
||||
fileType = B_FILE_MIMETYPE;
|
||||
|
||||
SharedCacheEntry* result = fHashTable.Add(SharedCacheEntry::Hash(fileType,
|
||||
appSignature));
|
||||
result->SetTo(fileType, appSignature);
|
||||
*outstandingEntry = fHashTable.ElementAt(entryToken);
|
||||
|
||||
return result;
|
||||
delete entry;
|
||||
return NULL;
|
||||
}
|
||||
|
||||
|
||||
@ -1516,13 +1464,13 @@ SharedIconCache::IconChanged(SharedCacheEntry* entry)
|
||||
|
||||
|
||||
void
|
||||
SharedIconCache::RemoveAliasesTo(int32 aliasIndex)
|
||||
SharedIconCache::RemoveAliasesTo(SharedCacheEntry* alias)
|
||||
{
|
||||
int32 count = fHashTable.VectorSize();
|
||||
for (int32 index = 0; index < count; index++) {
|
||||
SharedCacheEntry* entry = fHashTable.ElementAt(index);
|
||||
if (entry->fAliasForIndex == aliasIndex)
|
||||
fHashTable.Remove(entry);
|
||||
EntryHashTable::Iterator it = fHashTable.GetIterator();
|
||||
while (it.HasNext()) {
|
||||
SharedCacheEntry* entry = it.Next();
|
||||
if (entry->fAliasTo == alias)
|
||||
fHashTable.RemoveUnchecked(entry);
|
||||
}
|
||||
}
|
||||
|
||||
@ -1531,13 +1479,13 @@ void
|
||||
SharedIconCache::SetAliasFor(IconCacheEntry* entry,
|
||||
const SharedCacheEntry* original) const
|
||||
{
|
||||
entry->fAliasForIndex = fHashTable.ElementIndex(original);
|
||||
entry->fAliasTo = original;
|
||||
}
|
||||
|
||||
|
||||
SharedCacheEntry::SharedCacheEntry()
|
||||
:
|
||||
fNext(-1)
|
||||
fNext(NULL)
|
||||
{
|
||||
}
|
||||
|
||||
@ -1545,7 +1493,7 @@ SharedCacheEntry::SharedCacheEntry()
|
||||
SharedCacheEntry::SharedCacheEntry(const char* fileType,
|
||||
const char* appSignature)
|
||||
:
|
||||
fNext(-1),
|
||||
fNext(NULL),
|
||||
fFileType(fileType),
|
||||
fAppSignature(appSignature)
|
||||
{
|
||||
@ -1591,55 +1539,30 @@ SharedCacheEntry::Draw(BView* view, BPoint where, IconDrawMode mode,
|
||||
}
|
||||
|
||||
|
||||
uint32
|
||||
SharedCacheEntry::Hash(const char* fileType, const char* appSignature)
|
||||
/* static */ size_t
|
||||
SharedCacheEntry::Hash(const TypeAndSignature& typeAndSignature)
|
||||
{
|
||||
uint32 hash = HashString(fileType, 0);
|
||||
if (appSignature != NULL && *appSignature != '\0')
|
||||
hash = HashString(appSignature, hash);
|
||||
size_t hash = HashString(typeAndSignature.type, 0);
|
||||
if (typeAndSignature.signature != NULL
|
||||
&& *typeAndSignature.signature != '\0')
|
||||
hash = HashString(typeAndSignature.signature, hash);
|
||||
|
||||
return hash;
|
||||
}
|
||||
|
||||
|
||||
uint32
|
||||
size_t
|
||||
SharedCacheEntry::Hash() const
|
||||
{
|
||||
uint32 hash = HashString(fFileType.String(), 0);
|
||||
if (fAppSignature.Length() > 0)
|
||||
hash = HashString(fAppSignature.String(), hash);
|
||||
|
||||
return hash;
|
||||
return Hash(TypeAndSignature(fFileType.String(), fAppSignature.String()));
|
||||
}
|
||||
|
||||
|
||||
bool
|
||||
SharedCacheEntry::operator==(const SharedCacheEntry &entry) const
|
||||
SharedCacheEntry::operator==(const TypeAndSignature& typeAndSignature) const
|
||||
{
|
||||
return fFileType == entry.FileType()
|
||||
&& fAppSignature == entry.AppSignature();
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
SharedCacheEntry::SetTo(const char* fileType, const char* appSignature)
|
||||
{
|
||||
fFileType = fileType;
|
||||
fAppSignature = appSignature;
|
||||
}
|
||||
|
||||
|
||||
SharedCacheEntryArray::SharedCacheEntryArray(int32 initialSize)
|
||||
:
|
||||
OpenHashElementArray<SharedCacheEntry>(initialSize)
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
SharedCacheEntry*
|
||||
SharedCacheEntryArray::Add()
|
||||
{
|
||||
return OpenHashElementArray<SharedCacheEntry>::Add();
|
||||
return fFileType == typeAndSignature.type
|
||||
&& fAppSignature == typeAndSignature.signature;
|
||||
}
|
||||
|
||||
|
||||
@ -1648,7 +1571,7 @@ SharedCacheEntryArray::Add()
|
||||
|
||||
NodeCacheEntry::NodeCacheEntry(bool permanent)
|
||||
:
|
||||
fNext(-1),
|
||||
fNext(NULL),
|
||||
fPermanent(permanent)
|
||||
{
|
||||
}
|
||||
@ -1656,7 +1579,7 @@ NodeCacheEntry::NodeCacheEntry(bool permanent)
|
||||
|
||||
NodeCacheEntry::NodeCacheEntry(const node_ref* node, bool permanent)
|
||||
:
|
||||
fNext(-1),
|
||||
fNext(NULL),
|
||||
fRef(*node),
|
||||
fPermanent(permanent)
|
||||
{
|
||||
@ -1712,14 +1635,14 @@ NodeCacheEntry::Node() const
|
||||
}
|
||||
|
||||
|
||||
uint32
|
||||
size_t
|
||||
NodeCacheEntry::Hash() const
|
||||
{
|
||||
return Hash(&fRef);
|
||||
}
|
||||
|
||||
|
||||
uint32
|
||||
/* static */ size_t
|
||||
NodeCacheEntry::Hash(const node_ref* node)
|
||||
{
|
||||
return node->device ^ ((uint32*)&node->node)[0]
|
||||
@ -1728,16 +1651,9 @@ NodeCacheEntry::Hash(const node_ref* node)
|
||||
|
||||
|
||||
bool
|
||||
NodeCacheEntry::operator==(const NodeCacheEntry &entry) const
|
||||
NodeCacheEntry::operator==(const node_ref* node) const
|
||||
{
|
||||
return fRef == entry.fRef;
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
NodeCacheEntry::SetTo(const node_ref* node)
|
||||
{
|
||||
fRef = *node;
|
||||
return fRef == *node;
|
||||
}
|
||||
|
||||
|
||||
@ -1748,29 +1664,14 @@ NodeCacheEntry::Permanent() const
|
||||
}
|
||||
|
||||
|
||||
void
|
||||
NodeCacheEntry::MakePermanent()
|
||||
{
|
||||
fPermanent = true;
|
||||
}
|
||||
|
||||
|
||||
// #pragma mark - NodeIconCache
|
||||
|
||||
|
||||
NodeIconCache::NodeIconCache()
|
||||
:
|
||||
#if DEBUG
|
||||
SimpleIconCache("Node icon cache aka \"The Dead-Locker\""),
|
||||
fElementArray(20),
|
||||
fHashTable(20)
|
||||
#else
|
||||
SimpleIconCache("Tracker node icon cache"),
|
||||
fElementArray(100),
|
||||
fHashTable(100)
|
||||
#endif
|
||||
SimpleIconCache("Tracker node icon cache")
|
||||
{
|
||||
fHashTable.SetElementVector(&fElementArray);
|
||||
fHashTable.Init(100);
|
||||
}
|
||||
|
||||
|
||||
@ -1795,47 +1696,19 @@ NodeIconCache::Draw(IconCacheEntry* entry, BView* view, BPoint where,
|
||||
NodeCacheEntry*
|
||||
NodeIconCache::FindItem(const node_ref* node) const
|
||||
{
|
||||
NodeCacheEntry* entry = fHashTable.FindFirst(NodeCacheEntry::Hash(node));
|
||||
if (entry == NULL)
|
||||
return NULL;
|
||||
|
||||
for(;;) {
|
||||
if (*entry->Node() == *node)
|
||||
return entry;
|
||||
|
||||
if (entry->fNext < 0)
|
||||
break;
|
||||
|
||||
entry = const_cast<NodeCacheEntry*>(&fElementArray.At(entry->fNext));
|
||||
}
|
||||
|
||||
return NULL;
|
||||
return fHashTable.Lookup(node);
|
||||
}
|
||||
|
||||
|
||||
NodeCacheEntry*
|
||||
NodeIconCache::AddItem(const node_ref* node, bool permanent)
|
||||
{
|
||||
NodeCacheEntry* entry = fHashTable.Add(NodeCacheEntry::Hash(node));
|
||||
entry->SetTo(node);
|
||||
if (permanent)
|
||||
entry->MakePermanent();
|
||||
NodeCacheEntry* entry = new NodeCacheEntry(node, permanent);
|
||||
if (fHashTable.Insert(entry) == B_OK)
|
||||
return entry;
|
||||
|
||||
return entry;
|
||||
}
|
||||
|
||||
|
||||
NodeCacheEntry*
|
||||
NodeIconCache::AddItem(NodeCacheEntry** outstandingEntry,
|
||||
const node_ref* node)
|
||||
{
|
||||
int32 entryToken = fHashTable.ElementIndex(*outstandingEntry);
|
||||
|
||||
NodeCacheEntry* entry = fHashTable.Add(NodeCacheEntry::Hash(node));
|
||||
entry->SetTo(node);
|
||||
*outstandingEntry = fHashTable.ElementAt(entryToken);
|
||||
|
||||
return entry;
|
||||
delete entry;
|
||||
return NULL;
|
||||
}
|
||||
|
||||
|
||||
@ -1880,34 +1753,17 @@ NodeIconCache::IconChanged(const Model* model)
|
||||
|
||||
|
||||
void
|
||||
NodeIconCache::RemoveAliasesTo(int32 aliasIndex)
|
||||
NodeIconCache::RemoveAliasesTo(SharedCacheEntry* alias)
|
||||
{
|
||||
int32 count = fHashTable.VectorSize();
|
||||
for (int32 index = 0; index < count; index++) {
|
||||
NodeCacheEntry* entry = fHashTable.ElementAt(index);
|
||||
if (entry->fAliasForIndex == aliasIndex)
|
||||
fHashTable.Remove(entry);
|
||||
EntryHashTable::Iterator it = fHashTable.GetIterator();
|
||||
while (it.HasNext()) {
|
||||
NodeCacheEntry* entry = it.Next();
|
||||
if (entry->fAliasTo == alias)
|
||||
fHashTable.RemoveUnchecked(entry);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// #pragma mark - NodeCacheEntryArray
|
||||
|
||||
|
||||
NodeCacheEntryArray::NodeCacheEntryArray(int32 initialSize)
|
||||
:
|
||||
OpenHashElementArray<NodeCacheEntry>(initialSize)
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
NodeCacheEntry*
|
||||
NodeCacheEntryArray::Add()
|
||||
{
|
||||
return OpenHashElementArray<NodeCacheEntry>::Add();
|
||||
}
|
||||
|
||||
|
||||
// #pragma mark - SimpleIconCache
|
||||
|
||||
|
||||
|
@ -45,7 +45,7 @@ All rights reserved.
|
||||
#include <String.h>
|
||||
|
||||
#include "AutoLock.h"
|
||||
#include "OpenHashTable.h"
|
||||
#include "HashSet.h"
|
||||
#include "Utilities.h"
|
||||
|
||||
|
||||
@ -115,6 +115,33 @@ enum IconSource {
|
||||
};
|
||||
|
||||
|
||||
template<typename Class>
|
||||
struct SelfHashing {
|
||||
typedef typename Class::HashKeyType KeyType;
|
||||
typedef Class ValueType;
|
||||
|
||||
size_t HashKey(KeyType key) const
|
||||
{
|
||||
return Class::Hash(key);
|
||||
}
|
||||
|
||||
size_t Hash(ValueType* value) const
|
||||
{
|
||||
return value->Hash();
|
||||
}
|
||||
|
||||
bool Compare(KeyType key, ValueType* value) const
|
||||
{
|
||||
return *value == key;
|
||||
}
|
||||
|
||||
ValueType*& GetLink(ValueType* value) const
|
||||
{
|
||||
return value->HashNext();
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
class IconCacheEntry {
|
||||
// aliased entries don't own their icons, just point
|
||||
// to some other entry that does
|
||||
@ -168,7 +195,8 @@ protected:
|
||||
BBitmap* fHighlightedLargeIcon;
|
||||
BBitmap* fMiniIcon;
|
||||
BBitmap* fHighlightedMiniIcon;
|
||||
int32 fAliasForIndex;
|
||||
|
||||
const IconCacheEntry* fAliasTo;
|
||||
|
||||
// list of other icon kinds would be added here
|
||||
|
||||
@ -211,14 +239,23 @@ public:
|
||||
const char* FileType() const;
|
||||
const char* AppSignature() const;
|
||||
|
||||
public:
|
||||
// hash table support
|
||||
uint32 Hash() const;
|
||||
static uint32 Hash(const char* fileType, const char* appSignature = 0);
|
||||
bool operator==(const SharedCacheEntry &) const;
|
||||
void SetTo(const char* fileType, const char* appSignature = 0);
|
||||
struct TypeAndSignature {
|
||||
const char* type, *signature;
|
||||
TypeAndSignature(const char* t, const char* s)
|
||||
: type(t), signature(s) {}
|
||||
};
|
||||
typedef TypeAndSignature HashKeyType;
|
||||
static size_t Hash(const TypeAndSignature& typeAndSignature);
|
||||
|
||||
size_t Hash() const;
|
||||
SharedCacheEntry*& HashNext() { return fNext; }
|
||||
bool operator==(const TypeAndSignature& typeAndSignature) const;
|
||||
|
||||
int32 fNext;
|
||||
private:
|
||||
SharedCacheEntry* fNext;
|
||||
|
||||
BString fFileType;
|
||||
BString fAppSignature;
|
||||
|
||||
@ -226,14 +263,6 @@ private:
|
||||
};
|
||||
|
||||
|
||||
class SharedCacheEntryArray : public OpenHashElementArray<SharedCacheEntry> {
|
||||
// SharedIconCache stores all it's elements in this array
|
||||
public:
|
||||
SharedCacheEntryArray(int32 initialSize);
|
||||
SharedCacheEntry* Add();
|
||||
};
|
||||
|
||||
|
||||
class SharedIconCache : public SimpleIconCache {
|
||||
// SharedIconCache is used for icons that come from the mime database
|
||||
public:
|
||||
@ -248,22 +277,18 @@ public:
|
||||
const char* appSignature = 0) const;
|
||||
SharedCacheEntry* AddItem(const char* fileType,
|
||||
const char* appSignature = 0);
|
||||
SharedCacheEntry* AddItem(SharedCacheEntry** outstandingEntry,
|
||||
const char* fileType, const char* appSignature = 0);
|
||||
// same as previous AddItem, updates the pointer to outstandingEntry,
|
||||
// because adding to the hash table makes any pending pointer invalid
|
||||
void IconChanged(SharedCacheEntry*);
|
||||
|
||||
void SetAliasFor(IconCacheEntry* entry,
|
||||
const SharedCacheEntry* original) const;
|
||||
IconCacheEntry* ResolveIfAlias(IconCacheEntry* entry) const;
|
||||
int32 EntryIndex(const SharedCacheEntry* entry) const;
|
||||
|
||||
void RemoveAliasesTo(int32 index);
|
||||
void RemoveAliasesTo(SharedCacheEntry* alias);
|
||||
|
||||
private:
|
||||
SharedCacheEntryArray fElementArray;
|
||||
OpenHashTable<SharedCacheEntry, SharedCacheEntryArray> fHashTable;
|
||||
typedef BOpenHashTable<SelfHashing<SharedCacheEntry> > EntryHashTable;
|
||||
EntryHashTable fHashTable;
|
||||
|
||||
BObjectList<BBitmap> fRetiredBitmaps;
|
||||
// icons are drawn asynchronously, can't just delete them right away,
|
||||
// instead have to place them onto the retired bitmap list and wait
|
||||
@ -283,15 +308,20 @@ public:
|
||||
|
||||
const node_ref* Node() const;
|
||||
|
||||
uint32 Hash() const;
|
||||
static uint32 Hash(const node_ref*);
|
||||
bool operator==(const NodeCacheEntry&) const;
|
||||
void SetTo(const node_ref*);
|
||||
void MakePermanent();
|
||||
bool Permanent() const;
|
||||
|
||||
int32 fNext;
|
||||
public:
|
||||
// hash table support
|
||||
typedef const node_ref* HashKeyType;
|
||||
static size_t Hash(const node_ref* node);
|
||||
|
||||
size_t Hash() const;
|
||||
NodeCacheEntry*& HashNext() { return fNext; }
|
||||
bool operator==(const node_ref* ref) const;
|
||||
|
||||
private:
|
||||
NodeCacheEntry* fNext;
|
||||
|
||||
node_ref fRef;
|
||||
bool fPermanent;
|
||||
// special cache entry that has to be deleted explicitly
|
||||
@ -300,14 +330,6 @@ private:
|
||||
};
|
||||
|
||||
|
||||
class NodeCacheEntryArray : public OpenHashElementArray<NodeCacheEntry> {
|
||||
// NodeIconCache stores all it's elements in this array
|
||||
public:
|
||||
NodeCacheEntryArray(int32 initialSize);
|
||||
NodeCacheEntry* Add();
|
||||
};
|
||||
|
||||
|
||||
class NodeIconCache : public SimpleIconCache {
|
||||
// NodeIconCache is used for nodes that define their own icons
|
||||
public:
|
||||
@ -321,10 +343,6 @@ public:
|
||||
|
||||
NodeCacheEntry* FindItem(const node_ref*) const;
|
||||
NodeCacheEntry* AddItem(const node_ref*, bool permanent = false);
|
||||
NodeCacheEntry* AddItem(NodeCacheEntry** outstandingEntry,
|
||||
const node_ref*);
|
||||
// same as previous AddItem, updates the pointer to outstandingEntry,
|
||||
// because adding to the hash table makes any pending pointer invalid
|
||||
void Deleting(const node_ref*);
|
||||
// model for this node is getting deleted
|
||||
// (not necessarily the node itself)
|
||||
@ -333,11 +351,11 @@ public:
|
||||
void Deleting(const BView*);
|
||||
void IconChanged(const Model*);
|
||||
|
||||
void RemoveAliasesTo(int32 index);
|
||||
void RemoveAliasesTo(SharedCacheEntry* alias);
|
||||
|
||||
private:
|
||||
NodeCacheEntryArray fElementArray;
|
||||
OpenHashTable<NodeCacheEntry, NodeCacheEntryArray> fHashTable;
|
||||
typedef BOpenHashTable<SelfHashing<NodeCacheEntry> > EntryHashTable;
|
||||
EntryHashTable fHashTable;
|
||||
};
|
||||
|
||||
|
||||
@ -507,19 +525,13 @@ IconCache::NeedsDeletionNotification(IconSource from)
|
||||
inline IconCacheEntry*
|
||||
SharedIconCache::ResolveIfAlias(IconCacheEntry* entry) const
|
||||
{
|
||||
if (entry->fAliasForIndex < 0)
|
||||
if (entry->fAliasTo == NULL)
|
||||
return entry;
|
||||
|
||||
return fHashTable.ElementAt(entry->fAliasForIndex);
|
||||
return const_cast<IconCacheEntry*>(entry->fAliasTo);
|
||||
}
|
||||
|
||||
|
||||
inline int32
|
||||
SharedIconCache::EntryIndex(const SharedCacheEntry* entry) const
|
||||
{
|
||||
return fHashTable.ElementIndex(entry);
|
||||
}
|
||||
|
||||
} // namespace BPrivate
|
||||
|
||||
using namespace BPrivate;
|
||||
|
@ -1,514 +0,0 @@
|
||||
/*
|
||||
Open Tracker License
|
||||
|
||||
Terms and Conditions
|
||||
|
||||
Copyright (c) 1991-2000, Be Incorporated. All rights reserved.
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
||||
this software and associated documentation files (the "Software"), to deal in
|
||||
the Software without restriction, including without limitation the rights to
|
||||
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
|
||||
of the Software, and to permit persons to whom the Software is furnished to do
|
||||
so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice applies to all licensees
|
||||
and shall be included in all copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF TITLE, MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||||
BE INCORPORATED BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
|
||||
AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF, OR IN CONNECTION
|
||||
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||
|
||||
Except as contained in this notice, the name of Be Incorporated shall not be
|
||||
used in advertising or otherwise to promote the sale, use or other dealings in
|
||||
this Software without prior written authorization from Be Incorporated.
|
||||
|
||||
Tracker(TM), Be(R), BeOS(R), and BeIA(TM) are trademarks or registered trademarks
|
||||
of Be Incorporated in the United States and other countries. Other brand product
|
||||
names are registered trademarks or trademarks of their respective holders.
|
||||
All rights reserved.
|
||||
*/
|
||||
|
||||
// bonefish:
|
||||
// * removed need for exceptions
|
||||
// * fixed warnings
|
||||
// * implemented rehashing
|
||||
// * added RemoveAll()
|
||||
// TODO:
|
||||
// * shrinking of element vectors
|
||||
|
||||
// Hash table with open addresssing
|
||||
|
||||
#ifndef __OPEN_HASH_TABLE__
|
||||
#define __OPEN_HASH_TABLE__
|
||||
|
||||
#include <stdlib.h>
|
||||
#include <new>
|
||||
|
||||
// don't include <Debug.h>
|
||||
#ifndef _OPEN_HASH_TABLE_ASSERT
|
||||
# define _OPEN_HASH_TABLE_ASSERT(E) (void)0
|
||||
#endif
|
||||
#ifndef _OPEN_HASH_TABLE_TRESPASS
|
||||
# define _OPEN_HASH_TABLE_TRESPASS() (void)0
|
||||
#endif
|
||||
|
||||
namespace BPrivate {
|
||||
|
||||
template <class Element>
|
||||
class ElementVector {
|
||||
// element vector for OpenHashTable needs to implement this
|
||||
// interface
|
||||
public:
|
||||
Element &At(int32 index);
|
||||
Element *Add();
|
||||
int32 IndexOf(const Element &) const;
|
||||
void Remove(int32 index);
|
||||
};
|
||||
|
||||
class OpenHashElement {
|
||||
public:
|
||||
uint32 Hash() const;
|
||||
bool operator==(const OpenHashElement &) const;
|
||||
void Adopt(OpenHashElement &);
|
||||
// low overhead copy, original element is in undefined state
|
||||
// after call (calls Adopt on BString members, etc.)
|
||||
int32 fNext;
|
||||
};
|
||||
|
||||
const uint32 kPrimes [] = {
|
||||
509, 1021, 2039, 4093, 8191, 16381, 32749, 65521, 131071, 262139,
|
||||
524287, 1048573, 2097143, 4194301, 8388593, 16777213, 33554393, 67108859,
|
||||
134217689, 268435399, 536870909, 1073741789, 2147483647, 0
|
||||
};
|
||||
|
||||
template <class Element, class ElementVec = ElementVector<Element> >
|
||||
class OpenHashTable {
|
||||
public:
|
||||
OpenHashTable(int32 minSize, ElementVec *elementVector = 0,
|
||||
float maxLoadFactor = 0.8);
|
||||
// it is up to the subclass of OpenHashTable to supply
|
||||
// elementVector
|
||||
~OpenHashTable();
|
||||
|
||||
bool InitCheck() const;
|
||||
|
||||
void SetElementVector(ElementVec *elementVector);
|
||||
|
||||
Element *FindFirst(uint32 elementHash) const;
|
||||
Element *Add(uint32 elementHash);
|
||||
|
||||
void Remove(Element *element, bool dontRehash = false);
|
||||
void RemoveAll();
|
||||
|
||||
// when calling Add, any outstanding element pointer may become
|
||||
// invalid; to deal with this, get the element index and restore
|
||||
// it after the add
|
||||
int32 ElementIndex(const Element *) const;
|
||||
Element *ElementAt(int32 index) const;
|
||||
|
||||
int32 ArraySize() const;
|
||||
int32 VectorSize() const;
|
||||
int32 CountElements() const;
|
||||
|
||||
protected:
|
||||
static int32 OptimalSize(int32 minSize);
|
||||
|
||||
private:
|
||||
bool _RehashIfNeeded();
|
||||
bool _Rehash();
|
||||
|
||||
int32 fArraySize;
|
||||
int32 fInitialSize;
|
||||
int32 fElementCount;
|
||||
int32 *fHashArray;
|
||||
ElementVec *fElementVector;
|
||||
float fMaxLoadFactor;
|
||||
};
|
||||
|
||||
template <class Element>
|
||||
class OpenHashElementArray : public ElementVector<Element> {
|
||||
// this is a straightforward implementation of an element vector
|
||||
// deleting is handled by linking deleted elements into a free list
|
||||
// the vector never shrinks
|
||||
public:
|
||||
OpenHashElementArray(int32 initialSize);
|
||||
~OpenHashElementArray();
|
||||
|
||||
bool InitCheck() const;
|
||||
|
||||
Element &At(int32 index);
|
||||
const Element &At(int32 index) const;
|
||||
Element *Add(const Element &);
|
||||
Element *Add();
|
||||
void Remove(int32 index);
|
||||
int32 IndexOf(const Element &) const;
|
||||
int32 Size() const;
|
||||
|
||||
private:
|
||||
Element *fData;
|
||||
int32 fSize;
|
||||
int32 fNextFree;
|
||||
int32 fNextDeleted;
|
||||
};
|
||||
|
||||
|
||||
//-----------------------------------
|
||||
|
||||
template<class Element, class ElementVec>
|
||||
OpenHashTable<Element, ElementVec>::OpenHashTable(int32 minSize,
|
||||
ElementVec *elementVector, float maxLoadFactor)
|
||||
: fArraySize(OptimalSize(minSize)),
|
||||
fInitialSize(fArraySize),
|
||||
fElementCount(0),
|
||||
fElementVector(elementVector),
|
||||
fMaxLoadFactor(maxLoadFactor)
|
||||
{
|
||||
// sanity check the maximal load factor
|
||||
if (fMaxLoadFactor < 0.5)
|
||||
fMaxLoadFactor = 0.5;
|
||||
// allocate and init the array
|
||||
fHashArray = (int32*)calloc(fArraySize, sizeof(int32));
|
||||
if (fHashArray) {
|
||||
for (int32 index = 0; index < fArraySize; index++)
|
||||
fHashArray[index] = -1;
|
||||
}
|
||||
}
|
||||
|
||||
template<class Element, class ElementVec>
|
||||
OpenHashTable<Element, ElementVec>::~OpenHashTable()
|
||||
{
|
||||
RemoveAll();
|
||||
free(fHashArray);
|
||||
}
|
||||
|
||||
template<class Element, class ElementVec>
|
||||
bool
|
||||
OpenHashTable<Element, ElementVec>::InitCheck() const
|
||||
{
|
||||
return (fHashArray && fElementVector);
|
||||
}
|
||||
|
||||
template<class Element, class ElementVec>
|
||||
int32
|
||||
OpenHashTable<Element, ElementVec>::OptimalSize(int32 minSize)
|
||||
{
|
||||
for (int32 index = 0; ; index++)
|
||||
if (!kPrimes[index] || kPrimes[index] >= (uint32)minSize)
|
||||
return (int32)kPrimes[index];
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
template<class Element, class ElementVec>
|
||||
Element *
|
||||
OpenHashTable<Element, ElementVec>::FindFirst(uint32 hash) const
|
||||
{
|
||||
_OPEN_HASH_TABLE_ASSERT(fElementVector);
|
||||
hash %= fArraySize;
|
||||
if (fHashArray[hash] < 0)
|
||||
return 0;
|
||||
|
||||
return &fElementVector->At(fHashArray[hash]);
|
||||
}
|
||||
|
||||
template<class Element, class ElementVec>
|
||||
int32
|
||||
OpenHashTable<Element, ElementVec>::ElementIndex(const Element *element) const
|
||||
{
|
||||
return fElementVector->IndexOf(*element);
|
||||
}
|
||||
|
||||
template<class Element, class ElementVec>
|
||||
Element *
|
||||
OpenHashTable<Element, ElementVec>::ElementAt(int32 index) const
|
||||
{
|
||||
return &fElementVector->At(index);
|
||||
}
|
||||
|
||||
template<class Element, class ElementVec>
|
||||
int32
|
||||
OpenHashTable<Element, ElementVec>::ArraySize() const
|
||||
{
|
||||
return fArraySize;
|
||||
}
|
||||
|
||||
template<class Element, class ElementVec>
|
||||
int32
|
||||
OpenHashTable<Element, ElementVec>::VectorSize() const
|
||||
{
|
||||
return fElementVector->Size();
|
||||
}
|
||||
|
||||
template<class Element, class ElementVec>
|
||||
int32
|
||||
OpenHashTable<Element, ElementVec>::CountElements() const
|
||||
{
|
||||
return fElementCount;
|
||||
}
|
||||
|
||||
|
||||
template<class Element, class ElementVec>
|
||||
Element *
|
||||
OpenHashTable<Element, ElementVec>::Add(uint32 hash)
|
||||
{
|
||||
_OPEN_HASH_TABLE_ASSERT(fElementVector);
|
||||
_RehashIfNeeded();
|
||||
hash %= fArraySize;
|
||||
Element *result = fElementVector->Add();
|
||||
if (result) {
|
||||
result->fNext = fHashArray[hash];
|
||||
fHashArray[hash] = fElementVector->IndexOf(*result);
|
||||
fElementCount++;
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
template<class Element, class ElementVec>
|
||||
void
|
||||
OpenHashTable<Element, ElementVec>::Remove(Element *element, bool dontRehash)
|
||||
{
|
||||
if (!dontRehash)
|
||||
_RehashIfNeeded();
|
||||
uint32 hash = element->Hash() % fArraySize;
|
||||
int32 next = fHashArray[hash];
|
||||
_OPEN_HASH_TABLE_ASSERT(next >= 0);
|
||||
|
||||
if (&fElementVector->At(next) == element) {
|
||||
fHashArray[hash] = element->fNext;
|
||||
fElementVector->Remove(next);
|
||||
fElementCount--;
|
||||
return;
|
||||
}
|
||||
|
||||
for (int32 index = next; index >= 0; ) {
|
||||
// look for an existing match in table
|
||||
next = fElementVector->At(index).fNext;
|
||||
if (next < 0) {
|
||||
_OPEN_HASH_TABLE_TRESPASS();
|
||||
return;
|
||||
}
|
||||
|
||||
if (&fElementVector->At(next) == element) {
|
||||
fElementVector->At(index).fNext = element->fNext;
|
||||
fElementVector->Remove(next);
|
||||
fElementCount--;
|
||||
return;
|
||||
}
|
||||
index = next;
|
||||
}
|
||||
}
|
||||
|
||||
template<class Element, class ElementVec>
|
||||
void
|
||||
OpenHashTable<Element, ElementVec>::RemoveAll()
|
||||
{
|
||||
for (int32 i = 0; fElementCount > 0 && i < fArraySize; i++) {
|
||||
int32 index = fHashArray[i];
|
||||
while (index >= 0) {
|
||||
Element* element = &fElementVector->At(index);
|
||||
int32 next = element->fNext;
|
||||
fElementVector->Remove(index);
|
||||
fElementCount--;
|
||||
index = next;
|
||||
}
|
||||
fHashArray[i] = -1;
|
||||
}
|
||||
_RehashIfNeeded();
|
||||
}
|
||||
|
||||
template<class Element, class ElementVec>
|
||||
void
|
||||
OpenHashTable<Element, ElementVec>::SetElementVector(ElementVec *elementVector)
|
||||
{
|
||||
fElementVector = elementVector;
|
||||
}
|
||||
|
||||
// _RehashIfNeeded
|
||||
template<class Element, class ElementVec>
|
||||
bool
|
||||
OpenHashTable<Element, ElementVec>::_RehashIfNeeded()
|
||||
{
|
||||
// The load factor range [fMaxLoadFactor / 3, fMaxLoadFactor] is fine,
|
||||
// I think. After rehashing the load factor will be about
|
||||
// fMaxLoadFactor * 2 / 3, respectively fMaxLoadFactor / 2.
|
||||
float loadFactor = (float)fElementCount / (float)fArraySize;
|
||||
if (loadFactor > fMaxLoadFactor
|
||||
|| (fArraySize > fInitialSize && loadFactor < fMaxLoadFactor / 3)) {
|
||||
return _Rehash();
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
// _Rehash
|
||||
template<class Element, class ElementVec>
|
||||
bool
|
||||
OpenHashTable<Element, ElementVec>::_Rehash()
|
||||
{
|
||||
bool result = true;
|
||||
int32 newSize = int32(fElementCount * 1.73 * fMaxLoadFactor);
|
||||
newSize = (fInitialSize > newSize ? fInitialSize : newSize);
|
||||
if (newSize != fArraySize) {
|
||||
// allocate a new array
|
||||
int32 *newHashArray = (int32*)calloc(newSize, sizeof(int32));
|
||||
if (newHashArray) {
|
||||
// init the new hash array
|
||||
for (int32 index = 0; index < newSize; index++)
|
||||
newHashArray[index] = -1;
|
||||
// iterate through all elements and put them into the new
|
||||
// hash array
|
||||
for (int i = 0; i < fArraySize; i++) {
|
||||
int32 index = fHashArray[i];
|
||||
while (index >= 0) {
|
||||
// insert the element in the new array
|
||||
Element &element = fElementVector->At(index);
|
||||
int32 next = element.fNext;
|
||||
uint32 hash = (element.Hash() % newSize);
|
||||
element.fNext = newHashArray[hash];
|
||||
newHashArray[hash] = index;
|
||||
// next element in old list
|
||||
index = next;
|
||||
}
|
||||
}
|
||||
// delete the old array and set the new one
|
||||
free(fHashArray);
|
||||
fHashArray = newHashArray;
|
||||
fArraySize = newSize;
|
||||
} else
|
||||
result = false;
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
|
||||
template<class Element>
|
||||
OpenHashElementArray<Element>::OpenHashElementArray(int32 initialSize)
|
||||
: fSize(initialSize),
|
||||
fNextFree(0),
|
||||
fNextDeleted(-1)
|
||||
{
|
||||
fData = (Element*)calloc((size_t)initialSize, sizeof(Element));
|
||||
}
|
||||
|
||||
template<class Element>
|
||||
OpenHashElementArray<Element>::~OpenHashElementArray()
|
||||
{
|
||||
free(fData);
|
||||
}
|
||||
|
||||
template<class Element>
|
||||
bool
|
||||
OpenHashElementArray<Element>::InitCheck() const
|
||||
{
|
||||
return fData;
|
||||
}
|
||||
|
||||
template<class Element>
|
||||
Element &
|
||||
OpenHashElementArray<Element>::At(int32 index)
|
||||
{
|
||||
_OPEN_HASH_TABLE_ASSERT(index < fSize);
|
||||
return fData[index];
|
||||
}
|
||||
|
||||
template<class Element>
|
||||
const Element &
|
||||
OpenHashElementArray<Element>::At(int32 index) const
|
||||
{
|
||||
_OPEN_HASH_TABLE_ASSERT(index < fSize);
|
||||
return fData[index];
|
||||
}
|
||||
|
||||
template<class Element>
|
||||
int32
|
||||
OpenHashElementArray<Element>::IndexOf(const Element &element) const
|
||||
{
|
||||
int32 result = &element - fData;
|
||||
if (result < 0 || result > fSize)
|
||||
return -1;
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
template<class Element>
|
||||
int32
|
||||
OpenHashElementArray<Element>::Size() const
|
||||
{
|
||||
return fSize;
|
||||
}
|
||||
|
||||
|
||||
template<class Element>
|
||||
Element *
|
||||
OpenHashElementArray<Element>::Add(const Element &newElement)
|
||||
{
|
||||
Element *element = Add();
|
||||
if (element)
|
||||
element->Adopt(newElement);
|
||||
return element;
|
||||
}
|
||||
|
||||
#if DEBUG
|
||||
const int32 kGrowChunk = 10;
|
||||
#else
|
||||
const int32 kGrowChunk = 1024;
|
||||
#endif
|
||||
|
||||
template<class Element>
|
||||
Element *
|
||||
OpenHashElementArray<Element>::Add()
|
||||
{
|
||||
int32 index = fNextFree;
|
||||
if (fNextDeleted >= 0) {
|
||||
index = fNextDeleted;
|
||||
fNextDeleted = At(index).fNext;
|
||||
} else if (fNextFree >= fSize - 1) {
|
||||
int32 newSize = fSize + kGrowChunk;
|
||||
/*
|
||||
Element *newData = (Element *)calloc((size_t)newSize , sizeof(Element));
|
||||
if (!newData)
|
||||
return NULL;
|
||||
memcpy(newData, fData, fSize * sizeof(Element));
|
||||
free(fData);
|
||||
*/
|
||||
Element *newData = (Element*)realloc((void*)fData,
|
||||
(size_t)newSize * sizeof(Element));
|
||||
if (!newData)
|
||||
return NULL;
|
||||
|
||||
fData = newData;
|
||||
fSize = newSize;
|
||||
index = fNextFree;
|
||||
fNextFree++;
|
||||
} else
|
||||
fNextFree++;
|
||||
|
||||
new (&At(index)) Element;
|
||||
// call placement new to initialize the element properly
|
||||
_OPEN_HASH_TABLE_ASSERT(At(index).fNext == -1);
|
||||
|
||||
return &At(index);
|
||||
}
|
||||
|
||||
template<class Element>
|
||||
void
|
||||
OpenHashElementArray<Element>::Remove(int32 index)
|
||||
{
|
||||
// delete by chaining empty elements in a single linked
|
||||
// list, reusing the next field
|
||||
_OPEN_HASH_TABLE_ASSERT(index < fSize);
|
||||
At(index).~Element();
|
||||
// call the destructor explicitly to destroy the element
|
||||
// properly
|
||||
At(index).fNext = fNextDeleted;
|
||||
fNextDeleted = index;
|
||||
}
|
||||
|
||||
} // namespace BPrivate
|
||||
|
||||
using BPrivate::OpenHashTable;
|
||||
|
||||
#endif // __OPEN_HASH_TABLE__
|
Loading…
Reference in New Issue
Block a user