[block] always use a validator, but default to a noop one

This commit is contained in:
Joe Thornber 2011-08-31 13:18:28 +01:00
parent 9cfdbfb8cc
commit 6d37d86d55
2 changed files with 23 additions and 113 deletions

47
block.h
View File

@ -57,16 +57,21 @@ namespace persistent_data {
virtual void prepare(block &b) const = 0; virtual void prepare(block &b) const = 0;
}; };
class noop_validator : public validator {
public:
void check(block const &b) const {}
void prepare(block &b) const {}
};
struct block { struct block {
typedef boost::shared_ptr<block> ptr; typedef boost::shared_ptr<block> ptr;
typedef boost::optional<typename validator::ptr> maybe_validator;
block(block_address location, block(block_address location,
const_buffer &data, const_buffer &data,
unsigned &count, unsigned &count,
unsigned &type_count, unsigned &type_count,
bool is_superblock = false, bool is_superblock, // FIXME: make this an enum
maybe_validator v = maybe_validator()) typename validator::ptr v)
: location_(location), : location_(location),
adjuster_(count), adjuster_(count),
type_adjuster_(type_count), type_adjuster_(type_count),
@ -79,7 +84,7 @@ namespace persistent_data {
count_adjuster adjuster_; count_adjuster adjuster_;
count_adjuster type_adjuster_; count_adjuster type_adjuster_;
buffer data_; buffer data_;
maybe_validator validator_; typename validator::ptr validator_;
bool is_superblock_; bool is_superblock_;
}; };
@ -108,33 +113,24 @@ namespace persistent_data {
// Locking methods // Locking methods
read_ref read_ref
read_lock(block_address location) const;
boost::optional<read_ref>
read_try_lock(block_address location) const;
write_ref
write_lock(block_address location);
write_ref
write_lock_zero(block_address location);
// Validator variants
read_ref
read_lock(block_address location, read_lock(block_address location,
typename validator::ptr v) const; typename validator::ptr v =
typename validator::ptr(new noop_validator())) const;
boost::optional<read_ref> boost::optional<read_ref>
read_try_lock(block_address location, read_try_lock(block_address location,
typename validator::ptr v) const; typename validator::ptr v =
typename validator::ptr(new noop_validator())) const;
write_ref write_ref
write_lock(block_address location, write_lock(block_address location,
typename validator::ptr v); typename validator::ptr v =
typename validator::ptr(new noop_validator()));
write_ref write_ref
write_lock_zero(block_address location, write_lock_zero(block_address location,
typename validator::ptr v); typename validator::ptr v =
typename validator::ptr(new noop_validator()));
// The super block is the one that should be written last. // The super block is the one that should be written last.
// Unlocking this block triggers the following events: // Unlocking this block triggers the following events:
@ -146,16 +142,17 @@ namespace persistent_data {
// //
// If any locks are held at the time of the superblock // If any locks are held at the time of the superblock
// being unlocked then an exception will be thrown. // being unlocked then an exception will be thrown.
write_ref superblock(block_address b);
write_ref superblock_zero(block_address b);
write_ref superblock(block_address b, write_ref superblock(block_address b,
typename validator::ptr v); typename validator::ptr v =
typename validator::ptr(new noop_validator()));
write_ref superblock_zero(block_address b, write_ref superblock_zero(block_address b,
typename validator::ptr v); typename validator::ptr v =
typename validator::ptr(new noop_validator()));
// If you aren't using a superblock, then this flush method // If you aren't using a superblock, then this flush method
// will write all dirty data. Throws if any locks are // will write all dirty data. Throws if any locks are
// held. // held.
// FIXME: do we need this?
void flush(); void flush();
block_address get_nr_blocks() const; block_address get_nr_blocks() const;

View File

@ -68,58 +68,6 @@ block_manager<BlockSize>::~block_manager()
::close(fd_); ::close(fd_);
} }
template <uint32_t BlockSize>
typename block_manager<BlockSize>::read_ref
block_manager<BlockSize>::read_lock(block_address location) const
{
check(location);
buffer buf;
read_buffer(location, buf);
register_lock(location, READ_LOCK);
return read_ref(
typename block::ptr(
new block(location, buf, lock_count_, ordinary_count_),
bind(&block_manager::read_release, this, _1)));
}
template <uint32_t BlockSize>
optional<typename block_manager<BlockSize>::read_ref>
block_manager<BlockSize>::read_try_lock(block_address location) const
{
return read_lock(location);
}
template <uint32_t BlockSize>
typename block_manager<BlockSize>::write_ref
block_manager<BlockSize>::write_lock(block_address location)
{
check(location);
buffer buf;
read_buffer(location, buf);
register_lock(location, WRITE_LOCK);
return write_ref(
typename block::ptr(
new block(location, buf, lock_count_, ordinary_count_),
bind(&block_manager::write_release, this, _1)));
}
template <uint32_t BlockSize>
typename block_manager<BlockSize>::write_ref
block_manager<BlockSize>::write_lock_zero(block_address location)
{
check(location);
buffer buf;
zero_buffer(buf);
register_lock(location, WRITE_LOCK);
typename block::ptr b(new block(location, buf, lock_count_, ordinary_count_),
bind(&block_manager<BlockSize>::write_release, this, _1));
return write_ref(b);
}
template <uint32_t BlockSize> template <uint32_t BlockSize>
typename block_manager<BlockSize>::read_ref typename block_manager<BlockSize>::read_ref
block_manager<BlockSize>::read_lock(block_address location, block_manager<BlockSize>::read_lock(block_address location,
@ -173,40 +121,6 @@ block_manager<BlockSize>::write_lock_zero(block_address location,
return write_ref(b); return write_ref(b);
} }
template <uint32_t BlockSize>
typename block_manager<BlockSize>::write_ref
block_manager<BlockSize>::superblock(block_address location)
{
check(location);
if (superblock_count_ > 0)
throw runtime_error("already have superblock");
buffer buf;
read_buffer(location, buf);
typename block::ptr b(new block(location, buf, lock_count_, superblock_count_, true),
bind(&block_manager::write_release, this, _1));
register_lock(location, WRITE_LOCK);
return write_ref(b);
}
template <uint32_t BlockSize>
typename block_manager<BlockSize>::write_ref
block_manager<BlockSize>::superblock_zero(block_address location)
{
check(location);
if (superblock_count_ > 0)
throw runtime_error("already have superblock");
buffer buf;
zero_buffer(buf);
typename block::ptr b(new block(location, buf, lock_count_, superblock_count_, true),
bind(&block_manager::write_release, this, _1));
register_lock(location, WRITE_LOCK);
return write_ref(b);
}
template <uint32_t BlockSize> template <uint32_t BlockSize>
typename block_manager<BlockSize>::write_ref typename block_manager<BlockSize>::write_ref
block_manager<BlockSize>::superblock(block_address location, block_manager<BlockSize>::superblock(block_address location,
@ -325,8 +239,7 @@ block_manager<BlockSize>::write_release(block *b)
throw runtime_error("superblock isn't the last block"); throw runtime_error("superblock isn't the last block");
} }
if (b->validator_) b->validator_->prepare(*b);
(*b->validator_)->prepare(*b);
write_buffer(b->location_, b->data_); write_buffer(b->location_, b->data_);
unregister_lock(b->location_, WRITE_LOCK); unregister_lock(b->location_, WRITE_LOCK);