//%LICENSE//////////////////////////////////////////////////////////////// // // Licensed to The Open Group (TOG) under one or more contributor license // agreements. Refer to the OpenPegasusNOTICE.txt file distributed with // this work for additional information regarding copyright ownership. // Each contributor licenses this file to you under the OpenPegasus Open // Source License; you may not use this file except in compliance with the // License. // // 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 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 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. // IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS 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. // ////////////////////////////////////////////////////////////////////////// // //%///////////////////////////////////////////////////////////////////////////// #ifndef PegasusRepository_ObjectCache_h #define PegasusRepository_ObjectCache_h #include #include #include #include #include #include #include PEGASUS_NAMESPACE_BEGIN extern PEGASUS_REPOSITORY_LINKAGE Uint32 ObjectCacheHash(const String& str); template class ObjectCache { public: ObjectCache(size_t maxEntries); ~ObjectCache() { clear(); } void put(const String& path, OBJECT& object, bool clone = true); bool get(const String& path, OBJECT& object, bool clone = true); bool evict(const String& path); // Removes all the entries from the cache. void clear(); #ifdef PEGASUS_DEBUG void DisplayCacheStatistics() { PEGASUS_STD(cout) << " Size (current/max): " << _numEntries << "/" << _maxEntries << PEGASUS_STD(endl); PEGASUS_STD(cout) << " Requests satisfied from cache: " << _cacheReadHit << PEGASUS_STD(endl); PEGASUS_STD(cout) << " Requests *not* satisfied from cache: " << _cacheReadMiss << " (implies write to cache)" << PEGASUS_STD(endl); PEGASUS_STD(cout) << " Cache entries \"aged out\" due to cache size constraints: " << _cacheRemoveLRU << PEGASUS_STD(endl); } #endif private: static Uint32 _hash(const String& s) { return ObjectCacheHash(s); } static bool _equal(const String& s1, const String& s2) { return String::equalNoCase(s1, s2); } struct Entry { Uint32 code; String path; OBJECT object; Entry* hashNext; Entry* queueNext; Entry* queuePrev; Entry(Uint32 code_, const String& path_, OBJECT& object_) : code(code_), path(path_), object(object_.clone()) { } }; // If NUM_CHAINS size need to be changed,ensure size is power // of two to simplify moudulus calculation enum { NUM_CHAINS = 128 }; Entry* _chains[NUM_CHAINS]; Entry* _front; Entry* _back; size_t _numEntries; size_t _maxEntries; Mutex _mutex; #ifdef PEGASUS_DEBUG Uint32 _cacheReadHit; Uint32 _cacheReadMiss; Uint32 _cacheRemoveLRU; #endif }; template ObjectCache::ObjectCache(size_t maxEntries) : _front(0), _back(0), _numEntries(0), _maxEntries(maxEntries) #ifdef PEGASUS_DEBUG , _cacheReadHit(0), _cacheReadMiss(0), _cacheRemoveLRU(0) #endif { memset(_chains, 0, sizeof(_chains)); } template void ObjectCache::put( const String& path, OBJECT& object, bool clone) { if (_maxEntries == 0) return; AutoMutex lock(_mutex); //// Update object if it is already in cache: Uint32 code = _hash(path); Uint32 index = code & (NUM_CHAINS -1); for (Entry* p = _chains[index]; p; p = p->hashNext) { if (code == p->code && _equal(p->path, path)) { // Update the repository. if (clone) p->object = object.clone(); else p->object = object; return; } } //// Add to hash table: Entry* newEntry = new Entry(code, path, object); newEntry->hashNext = _chains[index]; _chains[index] = newEntry; //// Add to back of LRU queue: newEntry->queueNext = 0; if (_back) { _back->queueNext = newEntry; newEntry->queuePrev = _back; _back = newEntry; } else { _front = newEntry; _back = newEntry; newEntry->queuePrev = 0; } _numEntries++; //// Evict LRU entry if necessary (from front). if (_numEntries > _maxEntries) { Entry* entry = _front; //// Remove from hash table first. Uint32 frontIndex = entry->code & (NUM_CHAINS -1); Entry* hashPrev = 0; for (Entry* p = _chains[frontIndex]; p; p = p->hashNext) { if (p->code == entry->code && _equal(p->path, entry->path)) { if (hashPrev) hashPrev->hashNext = p->hashNext; else _chains[frontIndex] = p->hashNext; break; } hashPrev = p; } //// Now remove from queue: _front = entry->queueNext; if (_front) _front->queuePrev = 0; delete entry; _numEntries--; #ifdef PEGASUS_DEBUG _cacheRemoveLRU++; #endif } } template bool ObjectCache::get(const String& path, OBJECT& object, bool clone) { if (_maxEntries == 0) return false; AutoMutex lock(_mutex); //// Search cache for object. Uint32 code = _hash(path); Uint32 index = code & (NUM_CHAINS -1); for (Entry* p = _chains[index]; p; p = p->hashNext) { if (code == p->code && _equal(p->path, path)) { // If this entry is not already at the end of the LRU queue, move // it there. if (p->queueNext) { // Remove from queue: if (p->queuePrev) p->queuePrev->queueNext = p->queueNext; else _front = p->queueNext; p->queueNext->queuePrev = p->queuePrev; // Add to back of queue: PEGASUS_ASSERT(_back); p->queueNext = 0; _back->queueNext = p; p->queuePrev = _back; _back = p; } if (clone) object = p->object.clone(); else object = p->object; #ifdef PEGASUS_DEBUG _cacheReadHit++; #endif return true; } } /// Not found! #ifdef PEGASUS_DEBUG _cacheReadMiss++; #endif return false; } template bool ObjectCache::evict(const String& path) { if (_maxEntries == 0) return false; AutoMutex lock(_mutex); //// Find and remove the given element. Uint32 code = _hash(path); Uint32 index = code & (NUM_CHAINS -1); Entry* hashPrev = 0; for (Entry* p = _chains[index]; p; p = p->hashNext) { if (code == p->code && _equal(p->path, path)) { // Remove from hash chain: if (hashPrev) hashPrev->hashNext = p->hashNext; else _chains[index] = p->hashNext; // Remove from queue: if (p->queuePrev) p->queuePrev->queueNext = p->queueNext; else _front = p->queueNext; if (p->queueNext) p->queueNext->queuePrev = p->queuePrev; else _back = p->queuePrev; // Delete the entry and update the number of entries. delete p; _numEntries--; return true; } hashPrev = p; } //// Not found! return false; } template void ObjectCache::clear() { AutoMutex lock(_mutex); Entry* p = _front; while (p) { Entry* next = p->queueNext; delete p; p = next; } _front = 0; _back = 0; _numEntries = 0; memset(_chains, 0, sizeof(_chains)); } PEGASUS_NAMESPACE_END #endif /* PegasusRepository_ObjectCache_h */