vcglib/wrap/gcache/provider.h

88 lines
2.4 KiB
C++

#ifndef GCACHE_PROVIDER_H
#define GCACHE_PROVIDER_H
#include <QMutex>
#include "dheap.h"
#include "door.h"
#include "token.h"
/* this cache system enforce the rule that the items in a cache are always in all the cache below */
/* two mechanism to remove tokens from the cache:
1) set token count to something low
2) set maximum number of tokens in the provider
*/
/** Base class for Cache and last cache in the GCache system.
You should never interact with this class.
*/
template <typename Token>
class Provider: public QThread {
public:
///holds the resources in this cache but not in the cache above
PtrDHeap<Token> heap;
///tokens above this number will be scheduled for deletion
int max_tokens;
///signals we need to rebuild heap.
bool heap_dirty;
///lock this before manipulating heap.
QMutex heap_lock;
///used to sincronize priorities update
QMutex priority_lock;
///signals (to next cache!) priorities have changed or something is available
QDoor check_queue;
Provider(): max_tokens(-1), heap_dirty(false) {}
virtual ~Provider() {}
/// [should be protected, do not use]
void pushPriorities() {
QMutexLocker locker(&priority_lock);
for(int i = 0; i < heap.size(); i++)
heap[i].pushPriority();
heap_dirty = true;
check_queue.open();
}
/// assumes heap lock is locked, runs in cache thread [should be protected, do not use]
void rebuild() {
if(!this->heap_dirty) return;
{
QMutexLocker locker(&priority_lock);
for(int i = 0; i < this->heap.size(); i++)
this->heap[i].pullPriority();
this->heap_dirty = false;
}
this->heap.rebuild();
//remove OUTSIDE tokens from bottom of heap
if(max_tokens != -1) {
while(this->heap.size() > max_tokens) {
Token &t = this->heap.min();
t.count = Token::OUTSIDE;
this->heap.popMin();
}
}
}
///ensure no locked item are to be removed [should be protected, do not use]
template <class FUNCTOR> void flush(FUNCTOR functor) {
int count = 0;
QMutexLocker locker(&(this->heap_lock));
for(int k = 0; k < this->heap.size(); k++) {
Token *token = &this->heap[k];
if(functor(token)) { //drop it
token->count = Token::OUTSIDE;
} else
this->heap.at(count++) = token;
}
this->heap.resize(count);
this->heap_dirty = true;
}
};
#endif