// Copyright (C) 2011 Red Hat, Inc. All rights reserved. // // This file is part of the thin-provisioning-tools source. // // thin-provisioning-tools is free software: you can redistribute it // and/or modify it under the terms of the GNU General Public License // as published by the Free Software Foundation, either version 3 of // the License, or (at your option) any later version. // // thin-provisioning-tools is distributed in the hope that it will be // useful, but WITHOUT ANY WARRANTY; without even the implied warranty // of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License along // with thin-provisioning-tools. If not, see // . #include "thin-provisioning/override_emitter.h" #include "thin-provisioning/restore_emitter.h" #include "thin-provisioning/superblock.h" using namespace std; using namespace thin_provisioning; //---------------------------------------------------------------- namespace { using namespace superblock_detail; class restorer : public emitter { public: restorer(metadata::ptr md) : md_(md), in_superblock_(false), nr_data_blocks_(), empty_mapping_(new_mapping_tree()) { } virtual ~restorer() { // FIXME: replace with a call to empty_mapping_->destroy() md_->metadata_sm_->dec(empty_mapping_->get_root()); } virtual void begin_superblock(std::string const &uuid, uint64_t time, uint64_t trans_id, boost::optional flags, boost::optional version, uint32_t data_block_size, uint64_t nr_data_blocks, boost::optional metadata_snap) { in_superblock_ = true; nr_data_blocks_ = nr_data_blocks; superblock &sb = md_->sb_; memset(&sb.uuid_, 0, sizeof(sb.uuid_)); memcpy(&sb.uuid_, uuid.c_str(), std::min(sizeof(sb.uuid_), uuid.length())); sb.time_ = time; sb.trans_id_ = trans_id; sb.flags_ = flags ? *flags : 0; sb.version_ = version ? *version : 1; sb.data_block_size_ = data_block_size; sb.metadata_snap_ = metadata_snap ? *metadata_snap : 0; md_->data_sm_->extend(nr_data_blocks); } virtual void end_superblock() { if (!in_superblock_) throw runtime_error("missing superblock"); md_->commit(); in_superblock_ = false; } virtual void begin_device(uint32_t dev, uint64_t mapped_blocks, uint64_t trans_id, uint64_t creation_time, uint64_t snap_time) { if (!in_superblock_) throw runtime_error("missing superblock"); if (device_exists(dev)) throw std::runtime_error("Device already exists"); // Store the entry of the details tree current_device_details_.mapped_blocks_ = 0; current_device_details_.transaction_id_ = trans_id; current_device_details_.creation_time_ = (uint32_t)creation_time; current_device_details_.snapshotted_time_ = (uint32_t)snap_time; current_mapping_ = empty_mapping_->clone(); current_device_ = boost::optional(dev); } virtual void end_device() { uint64_t key[1] = {*current_device_}; // Add entry to the details tree md_->details_->insert(key, current_device_details_); md_->mappings_top_level_->insert(key, current_mapping_->get_root()); md_->mappings_->set_root(md_->mappings_top_level_->get_root()); // FIXME: ugly current_device_ = boost::optional(); } virtual void begin_named_mapping(std::string const &name) { throw runtime_error("not implemented"); } virtual void end_named_mapping() { throw runtime_error("not implemented"); } virtual void identifier(std::string const &name) { throw runtime_error("not implemented"); } virtual void range_map(uint64_t origin_begin, uint64_t data_begin, uint32_t time, uint64_t len) { for (uint64_t i = 0; i < len; i++) single_map(origin_begin++, data_begin++, time); } virtual void single_map(uint64_t origin_block, uint64_t data_block, uint32_t time) { if (!current_device_) throw runtime_error("not in device"); if (data_block >= nr_data_blocks_) { std::ostringstream out; out << "mapping beyond end of data device (" << data_block << " >= " << nr_data_blocks_ << ")"; throw std::runtime_error(out.str()); } uint64_t key[1] = {origin_block}; mapping_tree_detail::block_time bt; bt.block_ = data_block; bt.time_ = time; current_device_details_.mapped_blocks_ += static_cast(current_mapping_->insert(key, bt)); md_->data_sm_->inc(data_block); } private: single_mapping_tree::ptr new_mapping_tree() { return single_mapping_tree::ptr( new single_mapping_tree(*md_->tm_, mapping_tree_detail::block_time_ref_counter(md_->data_sm_))); } bool device_exists(thin_dev_t dev) const { uint64_t key[1] = {dev}; device_tree::maybe_value v = md_->details_->lookup(key); return !!v; } metadata::ptr md_; override_options opts_; bool in_superblock_; block_address nr_data_blocks_; boost::optional current_device_; device_tree_detail::device_details current_device_details_; single_mapping_tree::ptr current_mapping_; single_mapping_tree::ptr empty_mapping_; }; } //---------------------------------------------------------------- emitter::ptr thin_provisioning::create_restore_emitter(metadata::ptr md) { return emitter::ptr(new restorer(md)); } //----------------------------------------------------------------