124 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			124 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #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)
 | |
| 			: in_superblock_(false),
 | |
| 			  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));
 | |
| 			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<unsigned char> 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<unsigned char> 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:
 | |
| 		bool in_superblock_;
 | |
| 		metadata::ptr md_;
 | |
| 		bool clean_shutdown_;
 | |
| 		unsigned metadata_version_;
 | |
| 	};
 | |
| }
 | |
| 
 | |
| //----------------------------------------------------------------
 | |
| 
 | |
| emitter::ptr
 | |
| caching::create_restore_emitter(metadata::ptr md, unsigned metadata_version,
 | |
| 				bool clean_shutdown)
 | |
| {
 | |
| 	return emitter::ptr(new restorer(md, clean_shutdown, metadata_version));
 | |
| }
 | |
| 
 | |
| //----------------------------------------------------------------
 |