#ifndef MULTISNAP_METADATA_H #define MULTISNAP_METADATA_H #include "block.h" #include "transaction_manager.h" #include "btree.h" #include "endian.h" #include "metadata_disk_structures.h" #include #include //---------------------------------------------------------------- namespace thin_provisioning { unsigned const MD_BLOCK_SIZE = 4096; // FIXME: don't use namespaces in a header using namespace base; using namespace persistent_data; typedef uint64_t sector_t; typedef uint32_t thin_dev_t; //------------------------------------------------ class space_map_ref_counter { public: space_map_ref_counter(space_map::ptr sm) : sm_(sm) { } void inc(block_address b) { sm_->inc(b); } void dec(block_address b) { sm_->dec(b); } private: space_map::ptr sm_; }; struct block_traits { typedef base::__le64 disk_type; typedef uint64_t value_type; typedef space_map_ref_counter ref_counter; static void unpack(disk_type const &disk, value_type &value) { value = base::to_cpu(disk); } static void pack(value_type const &value, disk_type &disk) { disk = base::to_disk(value); } }; //------------------------------------------------ template class mtree_ref_counter { public: mtree_ref_counter(typename transaction_manager::ptr tm) : tm_(tm) { } void inc(block_address b) { } void dec(block_address b) { } private: typename transaction_manager::ptr tm_; }; template struct mtree_traits { typedef base::__le64 disk_type; typedef uint64_t value_type; typedef mtree_ref_counter ref_counter; static void unpack(disk_type const &disk, value_type &value) { value = base::to_cpu(disk); } static void pack(value_type const &value, disk_type &disk) { disk = base::to_disk(value); } }; class metadata; class thin { public: typedef boost::shared_ptr ptr; typedef boost::optional maybe_address; thin_dev_t get_dev_t() const; maybe_address lookup(block_address thin_block); void insert(block_address thin_block, block_address data_block); void remove(block_address thin_block); void set_snapshot_time(uint32_t time); block_address get_mapped_blocks() const; void set_mapped_blocks(block_address count); private: friend class metadata; thin(thin_dev_t dev, metadata *metadata); thin_dev_t dev_; metadata *metadata_; }; class metadata { public: typedef boost::shared_ptr ptr; metadata(transaction_manager::ptr tm, block_address superblock, sector_t data_block_size, block_address nr_data_blocks, bool create); ~metadata(); void commit(); void create_thin(thin_dev_t dev); void create_snap(thin_dev_t dev, thin_dev_t origin); void del(thin_dev_t); void set_transaction_id(uint64_t id); uint64_t get_transaction_id() const; block_address get_held_root() const; block_address alloc_data_block(); void free_data_block(block_address b); // accessors block_address get_nr_free_data_blocks() const; sector_t get_data_block_size() const; block_address get_data_dev_size() const; thin::ptr open_thin(thin_dev_t); private: friend class thin; bool device_exists(thin_dev_t dev) const; block_address superblock_; typedef persistent_data::transaction_manager::ptr tm_ptr; typedef persistent_data::btree<1, device_details_traits, MD_BLOCK_SIZE> detail_tree; typedef persistent_data::btree<1, mtree_traits, MD_BLOCK_SIZE> dev_tree; typedef persistent_data::btree<2, block_traits, MD_BLOCK_SIZE> mapping_tree; typedef persistent_data::btree<1, block_traits, MD_BLOCK_SIZE> single_mapping_tree; tm_ptr tm_; space_map::ptr metadata_sm_; space_map::ptr data_sm_; detail_tree details_; dev_tree mappings_top_level_; mapping_tree mappings_; superblock sb_; }; }; //---------------------------------------------------------------- #endif