#include "caching/restore_emitter.h" #include "caching/superblock.h" #include "caching/mapping_array.h" using namespace caching; using namespace std; using namespace superblock_damage; //---------------------------------------------------------------- namespace { class restorer : public emitter { public: restorer(metadata::ptr md, bool clean_shutdown, unsigned metadata_version) : md_(md), clean_shutdown_(clean_shutdown), metadata_version_(metadata_version) { } virtual void begin_superblock(std::string const &uuid, pd::block_address block_size, pd::block_address nr_cache_blocks, std::string const &policy, size_t hint_width) { superblock &sb = md_->sb_; sb.version = metadata_version_; strncpy((char *) sb.policy_name, policy.c_str(), sizeof(sb.policy_name) - 1); memset(sb.policy_version, 0, sizeof(sb.policy_version)); // FIXME: should come from xml sb.policy_hint_size = hint_width; md_->setup_hint_array(hint_width); sb.data_block_size = block_size; sb.cache_blocks = nr_cache_blocks; struct mapping unmapped_value; unmapped_value.oblock_ = 0; unmapped_value.flags_ = 0; md_->mappings_->grow(nr_cache_blocks, unmapped_value); if (metadata_version_ > 1) // make everything dirty by default md_->dirty_bits_->grow(nr_cache_blocks, true); vector hint_value(hint_width, '\0'); md_->hints_->grow(nr_cache_blocks, hint_value); } virtual void end_superblock() { md_->commit(clean_shutdown_); } virtual void begin_mappings() { // noop } virtual void end_mappings() { // noop } virtual void mapping(pd::block_address cblock, pd::block_address oblock, bool dirty) { caching::mapping m; m.oblock_ = oblock; m.flags_ = M_VALID; if (metadata_version_ == 1) { if (dirty) m.flags_ = m.flags_ | M_DIRTY; } else md_->dirty_bits_->set(cblock, dirty); md_->mappings_->set(cblock, m); } virtual void begin_hints() { // noop } virtual void end_hints() { // noop } virtual void hint(pd::block_address cblock, vector const &data) { md_->hints_->set_hint(cblock, data); } virtual void begin_discards() { // noop } virtual void end_discards() { // noop } virtual void discard(block_address dblock, block_address dblock_e) { while (dblock != dblock_e) { md_->discard_bits_->set(dblock, true); dblock++; } } private: metadata::ptr md_; bool clean_shutdown_; unsigned metadata_version_; }; } //---------------------------------------------------------------- emitter::ptr caching::create_restore_emitter(metadata::ptr md, unsigned metadata_version, shutdown_type st) { return emitter::ptr(new restorer(md, st == CLEAN_SHUTDOWN, metadata_version)); } //----------------------------------------------------------------