mirror of
https://github.com/ksherlock/profuse.git
synced 2025-01-10 23:29:42 +00:00
git-svn-id: https://profuse.googlecode.com/svn/branches/v2@209 aa027e90-d47c-11dd-86d7-074df07e0730
This commit is contained in:
parent
035d802d70
commit
ea4f0c4485
@ -6,46 +6,13 @@
|
||||
#include <sys/mman.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#include <Cache/BlockCache.h>
|
||||
#include <Device/BlockDevice.h>
|
||||
#include <Device/BlockCache.h>
|
||||
|
||||
#include <ProFUSE/Exception.h>
|
||||
#include <ProFUSE/auto.h>
|
||||
|
||||
|
||||
/*
|
||||
* Note -- everything is assumed to be single-threaded.
|
||||
*
|
||||
*/
|
||||
|
||||
/*
|
||||
* The primary purpose of the block cache is not as a cache
|
||||
* (the OS probably does a decent job of that) but rather to
|
||||
* simplify read/writes. Blocks will always be accessed by
|
||||
* pointer, so any updates will be shared.
|
||||
* For memory mapped prodos-order files, MappedBlockCache just
|
||||
* returns a pointer to the memory.
|
||||
* For dos-order, nibblized, or raw devices, ConcreteBlockCache
|
||||
* uses an approach similar to minix (although the buffer pool will
|
||||
* expand if needed).
|
||||
*
|
||||
* _buffers is a vector of all buffers and only exists to make
|
||||
* freeing them easier.
|
||||
* _hashTable is a simple hashtable of loaded blocks.
|
||||
* _first and _last are a double-linked list of unused blocks, stored
|
||||
* in lru order.
|
||||
*
|
||||
* The Entry struct contains the buffer, the block, a dirty flag, and an in-use
|
||||
* count as well as pointer for the hashtable and lru list.
|
||||
* When a block is loaded, it is stored in the _hashTable. It remains in the
|
||||
* hash table when the in-use count goes to 0 (it will also be added to the
|
||||
* end of the lru list).
|
||||
*
|
||||
* dirty buffers are only written to disk in 3 scenarios:
|
||||
* a) sync() is called
|
||||
* b) the cache is deleted
|
||||
* c) a buffer is re-used from the lru list.
|
||||
*/
|
||||
|
||||
|
||||
using namespace Device;
|
||||
@ -56,7 +23,7 @@ using ProFUSE::POSIXException;
|
||||
#pragma mark -
|
||||
#pragma mark BlockCache
|
||||
|
||||
BlockCache::~BlockCache(BlockDevice *device)
|
||||
BlockCache::BlockCache(BlockDevice *device)
|
||||
{
|
||||
_device = _device;
|
||||
_blocks = device->blocks();
|
||||
@ -81,334 +48,3 @@ void BlockCache::read(unsigned block, void *bp)
|
||||
std::memcpy(bp, address, 512);
|
||||
release(block, false);
|
||||
}
|
||||
|
||||
#pragma mark -
|
||||
#pragma mark MappedBlockCache
|
||||
|
||||
MappedBlockCache::MappedBlockCache(void *data, unsigned blocks)
|
||||
{
|
||||
_blocks = blocks;
|
||||
_data = (uint8_t *)data;
|
||||
}
|
||||
|
||||
void MappedBlockCache::write()
|
||||
{
|
||||
// TODO...
|
||||
}
|
||||
|
||||
void *MappedBlockCache::load(unsigned block)
|
||||
{
|
||||
#undef __METHOD__
|
||||
#define __METHOD__ "MappedBlockCache::load"
|
||||
|
||||
if (block >= _blocks)
|
||||
throw Exception(__METHOD__ ": Invalid block.");
|
||||
|
||||
|
||||
return _data + block * 512;
|
||||
}
|
||||
void MappedBlockCache::unload(unsigned block, bool dirty)
|
||||
{
|
||||
#undef __METHOD__
|
||||
#define __METHOD__ "MappedBlockCache::unload"
|
||||
|
||||
if (!dirty) return;
|
||||
|
||||
// msync must be page-size aligned.
|
||||
unsigned pagesize = ::getpagesize();
|
||||
unsigned offset = block * 512;
|
||||
void *address = _data + offset / pagesize * pagesize;
|
||||
unsigned length = offset % pagesize + 512;
|
||||
|
||||
if (::msync(address, length, MS_ASYNC) < 0)
|
||||
{
|
||||
throw POSIXException(__METHOD__ ": msync failed.", errno);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
#pragma mark -
|
||||
#pragma mark ConcreteBlockCache
|
||||
|
||||
typedef std::vector<BlockCache::Entry *>::iterator EntryIter;
|
||||
|
||||
|
||||
|
||||
ConcreteBlockCache::ConcreteBlockCache(BlockDevice *device, unsigned size) :
|
||||
BlockCache(device)
|
||||
{
|
||||
if (size < 16) size = 16;
|
||||
|
||||
std::memset(_hashTable, 0, sizeof(Entry *) * HashTableSize);
|
||||
|
||||
for (unsigned i = 0; i < size; ++i)
|
||||
{
|
||||
Entry *e = new Entry;
|
||||
|
||||
std::memset(e, 0, sizeof(Entry));
|
||||
_buffers.push_back(e);
|
||||
}
|
||||
_first = _last = NULL;
|
||||
|
||||
_first = _buffers.front();
|
||||
_last = _buffers.back();
|
||||
// form a chain....
|
||||
for (unsigned i = 0; i < size; ++i)
|
||||
{
|
||||
Entry *e = _buffers[i];
|
||||
|
||||
if (i > 0) e->prev = _buffers[i - 1];
|
||||
|
||||
if (i < size - 1) e->next = _buffers[i + 1];
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
ConcreteBlockCache::~ConcreteBlockCache()
|
||||
{
|
||||
EntryIter iter;
|
||||
for (iter = _buffers.begin(); iter != _buffers.end(); ++iter)
|
||||
{
|
||||
Entry *e = *iter;
|
||||
|
||||
if (e->dirty)
|
||||
{
|
||||
_device->write(e->block, e->buffer);
|
||||
}
|
||||
|
||||
delete e;
|
||||
}
|
||||
_device->sync();
|
||||
}
|
||||
|
||||
|
||||
ConcreteBlockCache::sync()
|
||||
{
|
||||
EntryIter iter;
|
||||
for (iter = _buffers.begin(); iter != _buffers.end(); ++iter)
|
||||
{
|
||||
Entry *e = *iter;
|
||||
|
||||
if (e->dirty)
|
||||
{
|
||||
_device->write(e->block, e->buffer);
|
||||
e->dirty = false;
|
||||
}
|
||||
}
|
||||
_device->sync();
|
||||
}
|
||||
|
||||
|
||||
void ConcreteBlockCache::write(unsigned block, const void *bp)
|
||||
{
|
||||
FileEntry *e = findEntry();
|
||||
|
||||
if (e)
|
||||
{
|
||||
e->dirty = true;
|
||||
std::memcpy(e->buffer, bp, 512);
|
||||
return;
|
||||
}
|
||||
|
||||
// returns a new entry not in the hashtable or the linked list.
|
||||
// we add it to both.
|
||||
e = newEntry(block);
|
||||
|
||||
e->count = 0;
|
||||
e->dirty = true;
|
||||
|
||||
std::memcpy(e->buffer, bp, 512);
|
||||
|
||||
addEntry(e);
|
||||
setLast(e);
|
||||
}
|
||||
|
||||
void ConcreteBlockCache::markDirty(unsigned block)
|
||||
{
|
||||
Entry *e = findEntry(block);
|
||||
|
||||
if (e) e->dirty = true;
|
||||
// error otherwise?
|
||||
}
|
||||
|
||||
|
||||
void ConcreteBlockCache::release(unsigned block, bool dirty)
|
||||
{
|
||||
Entry *e = findEntry(block);
|
||||
|
||||
if (e)
|
||||
{
|
||||
if (dirty) e->dirty = true;
|
||||
|
||||
decrementCount(e);
|
||||
}
|
||||
// error otherwise?
|
||||
}
|
||||
|
||||
void *ConcreteBlockCache::acquire(unsigned block)
|
||||
{
|
||||
Entry *e = findEntry(block);
|
||||
|
||||
if (e)
|
||||
{
|
||||
incrementCount(e);
|
||||
return e->buffer;
|
||||
}
|
||||
|
||||
// returns a new entry, not in hash table, not in free list.
|
||||
e = newEntry(block);
|
||||
|
||||
_device->read(block, e->buffer);
|
||||
|
||||
addEntry(e);
|
||||
|
||||
return e->buffer;
|
||||
}
|
||||
|
||||
unsigned ConcreteBlockCache::hashFunction(unsigned block)
|
||||
{
|
||||
return block % HashTableSize;
|
||||
}
|
||||
|
||||
/*
|
||||
* remove a block from the hashtable
|
||||
* and write to dick if dirty.
|
||||
*/
|
||||
void removeEntry(unsigned block)
|
||||
{
|
||||
Entry *e;
|
||||
Entry *prev;
|
||||
unsigned hash = hashFunction(block);
|
||||
|
||||
e = _hashTable[hash];
|
||||
if (!e) return;
|
||||
|
||||
// head pointer, special case.
|
||||
if (e->block == block)
|
||||
{
|
||||
_hashTable[hash] = e->nextHash;
|
||||
e->nextHash = NULL;
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
for(;;)
|
||||
{
|
||||
prev = e;
|
||||
e = e->next;
|
||||
|
||||
if (!e) break;
|
||||
if (e->block == block)
|
||||
{
|
||||
prev->nextHash = e->nextHash;
|
||||
e->nextHash = NULL;
|
||||
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void ConcreteBlockCache::addEntry(Entry *e)
|
||||
{
|
||||
unsigned hash = hashFunction(e->block);
|
||||
|
||||
e->nextHash = _hashTable[hash];
|
||||
_hashTable[hash] = e;
|
||||
}
|
||||
|
||||
// increment the count and remove from the free list
|
||||
// if necessary.
|
||||
void ConcreteBlockCache::incrementCount(Entry *e)
|
||||
{
|
||||
|
||||
if (e->count == 0)
|
||||
{
|
||||
Entry *prev = e->prev;
|
||||
Entry *next = e->next;
|
||||
|
||||
e->prev = e->next = NULL;
|
||||
|
||||
if (prev) prev->next = next;
|
||||
if (next) next->prev = prev;
|
||||
|
||||
if (_first == e) _first = next;
|
||||
if (_last == e) _last = prev;
|
||||
}
|
||||
|
||||
e->count = e->count + 1;
|
||||
|
||||
}
|
||||
|
||||
// decrement the count. If it goes to 0,
|
||||
// add it as the last block entry.
|
||||
void ConcreteBlockCache::decrementCount(Entry *e)
|
||||
{
|
||||
e->count = e->count - 1;
|
||||
if (e->count == 0)
|
||||
{
|
||||
if (_last == NULL)
|
||||
{
|
||||
_first = _last = e;
|
||||
return;
|
||||
}
|
||||
|
||||
e->prev = _last;
|
||||
_last = e;
|
||||
}
|
||||
}
|
||||
|
||||
Entry *ConcreteBlockCache::newEntry(unsigned block)
|
||||
{
|
||||
Entry *e;
|
||||
|
||||
if (_first)
|
||||
{
|
||||
e = _first;
|
||||
if (_first == _last)
|
||||
{
|
||||
_first = _last = NULL;
|
||||
}
|
||||
else
|
||||
{
|
||||
_first = e->next;
|
||||
_first->prev = NULL;
|
||||
}
|
||||
|
||||
|
||||
if (e->dirty)
|
||||
{
|
||||
_device->write(e->block, e->buffer);
|
||||
e->dirty = false;
|
||||
}
|
||||
removeEntry(e->block);
|
||||
}
|
||||
else
|
||||
{
|
||||
e = new Entry;
|
||||
_buffers.push_back(e);
|
||||
}
|
||||
|
||||
e->next = NULL;
|
||||
e->prev= NULL;
|
||||
e->nextHash = NULL;
|
||||
e->count = 1;
|
||||
e->block = block;
|
||||
e->dirty = false;
|
||||
|
||||
return e;
|
||||
}
|
||||
|
||||
|
||||
void ConcreteBlockCache::setLast(Entry *e)
|
||||
{
|
||||
if (_last == NULL)
|
||||
{
|
||||
_first = _last = e;
|
||||
return;
|
||||
}
|
||||
|
||||
e->prev = _last;
|
||||
_last->next = e;
|
||||
_last = e;
|
||||
}
|
@ -54,7 +54,7 @@ using ProFUSE::Exception;
|
||||
using ProFUSE::POSIXException;
|
||||
|
||||
|
||||
typedef std::vector<BlockCache::Entry *>::iterator EntryIter;
|
||||
typedef std::vector<ConcreteBlockCache::Entry *>::iterator EntryIter;
|
||||
|
||||
|
||||
|
||||
@ -106,7 +106,7 @@ ConcreteBlockCache::~ConcreteBlockCache()
|
||||
}
|
||||
|
||||
|
||||
ConcreteBlockCache::sync()
|
||||
void ConcreteBlockCache::sync()
|
||||
{
|
||||
EntryIter iter;
|
||||
for (iter = _buffers.begin(); iter != _buffers.end(); ++iter)
|
||||
@ -125,7 +125,7 @@ ConcreteBlockCache::sync()
|
||||
|
||||
void ConcreteBlockCache::write(unsigned block, const void *bp)
|
||||
{
|
||||
FileEntry *e = findEntry();
|
||||
Entry *e = findEntry(block);
|
||||
|
||||
if (e)
|
||||
{
|
||||
@ -167,7 +167,7 @@ void ConcreteBlockCache::release(unsigned block, int flags)
|
||||
|
||||
decrementCount(e);
|
||||
|
||||
if (flags & kCommitNow)
|
||||
if (flags & kBlockCommitNow)
|
||||
{
|
||||
_device->write(block, e->buffer);
|
||||
e->dirty = false;
|
||||
@ -206,7 +206,7 @@ unsigned ConcreteBlockCache::hashFunction(unsigned block)
|
||||
* remove a block from the hashtable
|
||||
* and write to dick if dirty.
|
||||
*/
|
||||
void removeEntry(unsigned block)
|
||||
void ConcreteBlockCache::removeEntry(unsigned block)
|
||||
{
|
||||
Entry *e;
|
||||
Entry *prev;
|
||||
@ -283,7 +283,7 @@ void ConcreteBlockCache::decrementCount(Entry *e)
|
||||
}
|
||||
}
|
||||
|
||||
Entry *ConcreteBlockCache::newEntry(unsigned block)
|
||||
ConcreteBlockCache::Entry *ConcreteBlockCache::newEntry(unsigned block)
|
||||
{
|
||||
Entry *e;
|
||||
|
||||
@ -351,3 +351,4 @@ void ConcreteBlockCache::setFirst(Entry *e)
|
||||
_first->prev = e;
|
||||
_first = e;
|
||||
}
|
||||
|
||||
|
@ -58,8 +58,10 @@ private:
|
||||
void setLast(Entry *);
|
||||
void setFirst(Entry *);
|
||||
|
||||
incrementCount(Entry *);
|
||||
decrementCount(Entry *);
|
||||
void incrementCount(Entry *);
|
||||
void decrementCount(Entry *);
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
#endif
|
||||
|
Loading…
x
Reference in New Issue
Block a user