2011-06-23 14:47:08 +01:00
|
|
|
#ifndef MULTISNAP_METADATA_H
|
|
|
|
#define MULTISNAP_METADATA_H
|
|
|
|
|
|
|
|
#include "block.h"
|
|
|
|
#include "btree.h"
|
2011-08-22 10:42:13 +01:00
|
|
|
#include "endian_utils.h"
|
2011-08-24 10:45:39 +01:00
|
|
|
#include "error_set.h"
|
2011-07-22 16:09:56 +01:00
|
|
|
#include "metadata_disk_structures.h"
|
2011-08-24 10:45:39 +01:00
|
|
|
#include "space_map_disk.h"
|
|
|
|
#include "transaction_manager.h"
|
2011-06-23 14:47:08 +01:00
|
|
|
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
#include <boost/shared_ptr.hpp>
|
|
|
|
|
|
|
|
//----------------------------------------------------------------
|
|
|
|
|
2011-07-22 16:09:56 +01:00
|
|
|
namespace thin_provisioning {
|
|
|
|
unsigned const MD_BLOCK_SIZE = 4096;
|
|
|
|
|
|
|
|
// FIXME: don't use namespaces in a header
|
|
|
|
using namespace base;
|
|
|
|
using namespace persistent_data;
|
2011-06-23 14:47:08 +01:00
|
|
|
|
|
|
|
typedef uint64_t sector_t;
|
2011-07-22 16:09:56 +01:00
|
|
|
typedef uint32_t thin_dev_t;
|
2011-06-23 14:47:08 +01:00
|
|
|
|
2011-07-22 16:09:56 +01:00
|
|
|
//------------------------------------------------
|
2011-06-27 10:45:30 +01:00
|
|
|
|
2011-07-22 16:09:56 +01:00
|
|
|
class space_map_ref_counter {
|
|
|
|
public:
|
|
|
|
space_map_ref_counter(space_map::ptr sm)
|
|
|
|
: sm_(sm) {
|
|
|
|
}
|
2011-06-27 10:45:30 +01:00
|
|
|
|
2011-07-22 16:09:56 +01:00
|
|
|
void inc(block_address b) {
|
|
|
|
sm_->inc(b);
|
|
|
|
}
|
2011-06-27 10:45:30 +01:00
|
|
|
|
2011-07-22 16:09:56 +01:00
|
|
|
void dec(block_address b) {
|
|
|
|
sm_->dec(b);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
space_map::ptr sm_;
|
|
|
|
};
|
|
|
|
|
2011-08-24 10:45:39 +01:00
|
|
|
struct block_time {
|
|
|
|
uint64_t block_;
|
|
|
|
uint32_t time_;
|
|
|
|
};
|
|
|
|
|
|
|
|
class block_time_ref_counter {
|
|
|
|
public:
|
|
|
|
block_time_ref_counter(space_map::ptr sm)
|
|
|
|
: sm_(sm) {
|
|
|
|
}
|
|
|
|
|
|
|
|
void inc(block_time bt) {
|
|
|
|
sm_->inc(bt.block_);
|
|
|
|
}
|
|
|
|
|
|
|
|
void dec(block_time bt) {
|
|
|
|
sm_->dec(bt.block_);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
space_map::ptr sm_;
|
|
|
|
};
|
|
|
|
|
2011-07-22 16:09:56 +01:00
|
|
|
struct block_traits {
|
|
|
|
typedef base::__le64 disk_type;
|
2011-08-24 10:45:39 +01:00
|
|
|
typedef block_time value_type;
|
|
|
|
typedef block_time_ref_counter ref_counter;
|
2011-06-27 10:45:30 +01:00
|
|
|
|
2011-07-22 16:09:56 +01:00
|
|
|
static void unpack(disk_type const &disk, value_type &value) {
|
2011-08-24 10:45:39 +01:00
|
|
|
uint64_t v = to_cpu<uint64_t>(disk);
|
|
|
|
value.block_ = v >> 24;
|
|
|
|
value.time_ = v & ((1 << 24) - 1);
|
2011-07-22 16:09:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void pack(value_type const &value, disk_type &disk) {
|
2011-08-24 10:45:39 +01:00
|
|
|
uint64_t v = (value.block_ << 24) | value.time_;
|
|
|
|
disk = base::to_disk<base::__le64>(v);
|
2011-06-27 10:45:30 +01:00
|
|
|
}
|
|
|
|
};
|
2011-07-22 16:09:56 +01:00
|
|
|
|
|
|
|
//------------------------------------------------
|
|
|
|
|
|
|
|
template <uint32_t BlockSize>
|
|
|
|
class mtree_ref_counter {
|
2011-06-27 10:45:30 +01:00
|
|
|
public:
|
2011-07-22 16:09:56 +01:00
|
|
|
mtree_ref_counter(typename transaction_manager<BlockSize>::ptr tm)
|
|
|
|
: tm_(tm) {
|
|
|
|
}
|
|
|
|
|
|
|
|
void inc(block_address b) {
|
|
|
|
}
|
|
|
|
|
|
|
|
void dec(block_address b) {
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
typename transaction_manager<BlockSize>::ptr tm_;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <uint32_t BlockSize>
|
|
|
|
struct mtree_traits {
|
2011-06-27 10:45:30 +01:00
|
|
|
typedef base::__le64 disk_type;
|
2011-07-22 16:09:56 +01:00
|
|
|
typedef uint64_t value_type;
|
|
|
|
typedef mtree_ref_counter<BlockSize> ref_counter;
|
2011-06-27 10:45:30 +01:00
|
|
|
|
2011-07-22 16:09:56 +01:00
|
|
|
static void unpack(disk_type const &disk, value_type &value) {
|
|
|
|
value = base::to_cpu<uint64_t>(disk);
|
|
|
|
}
|
2011-06-27 10:45:30 +01:00
|
|
|
|
2011-07-22 16:09:56 +01:00
|
|
|
static void pack(value_type const &value, disk_type &disk) {
|
|
|
|
disk = base::to_disk<base::__le64>(value);
|
2011-06-27 10:45:30 +01:00
|
|
|
}
|
|
|
|
};
|
2011-07-22 16:09:56 +01:00
|
|
|
|
|
|
|
class metadata;
|
|
|
|
class thin {
|
|
|
|
public:
|
|
|
|
typedef boost::shared_ptr<thin> ptr;
|
2011-08-24 10:45:39 +01:00
|
|
|
typedef boost::optional<block_time> maybe_address;
|
2011-07-22 16:09:56 +01:00
|
|
|
|
|
|
|
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_;
|
|
|
|
};
|
2011-06-27 10:45:30 +01:00
|
|
|
|
2011-06-23 14:47:08 +01:00
|
|
|
class metadata {
|
|
|
|
public:
|
|
|
|
typedef boost::shared_ptr<metadata> ptr;
|
2011-08-25 10:54:43 +01:00
|
|
|
typedef block_manager<MD_BLOCK_SIZE>::read_ref read_ref;
|
|
|
|
typedef block_manager<MD_BLOCK_SIZE>::write_ref write_ref;
|
2011-06-23 14:47:08 +01:00
|
|
|
|
2011-08-22 10:42:13 +01:00
|
|
|
metadata(std::string const &dev_path);
|
2011-06-23 14:47:08 +01:00
|
|
|
~metadata();
|
|
|
|
|
|
|
|
void commit();
|
|
|
|
|
2011-07-22 16:09:56 +01:00
|
|
|
void create_thin(thin_dev_t dev);
|
|
|
|
void create_snap(thin_dev_t dev, thin_dev_t origin);
|
|
|
|
void del(thin_dev_t);
|
2011-06-23 14:47:08 +01:00
|
|
|
|
|
|
|
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;
|
|
|
|
|
2011-07-22 16:09:56 +01:00
|
|
|
thin::ptr open_thin(thin_dev_t);
|
2011-06-23 14:47:08 +01:00
|
|
|
|
2011-08-23 11:55:37 +01:00
|
|
|
// Validation and repair
|
2011-08-24 10:45:39 +01:00
|
|
|
boost::optional<persistent_data::error_set::ptr> check();
|
2011-08-23 11:55:37 +01:00
|
|
|
|
2011-06-23 14:47:08 +01:00
|
|
|
private:
|
|
|
|
friend class thin;
|
|
|
|
|
2011-07-22 16:09:56 +01:00
|
|
|
bool device_exists(thin_dev_t dev) const;
|
2011-06-23 14:47:08 +01:00
|
|
|
|
2011-07-22 16:09:56 +01:00
|
|
|
typedef persistent_data::transaction_manager<MD_BLOCK_SIZE>::ptr tm_ptr;
|
2011-06-23 14:47:08 +01:00
|
|
|
|
2011-07-22 16:09:56 +01:00
|
|
|
typedef persistent_data::btree<1, device_details_traits, MD_BLOCK_SIZE> detail_tree;
|
|
|
|
typedef persistent_data::btree<1, mtree_traits<MD_BLOCK_SIZE>, 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;
|
2011-06-23 14:47:08 +01:00
|
|
|
|
2011-08-23 11:55:37 +01:00
|
|
|
// Declaration order is important here
|
2011-07-22 16:09:56 +01:00
|
|
|
tm_ptr tm_;
|
2011-08-23 11:55:37 +01:00
|
|
|
superblock sb_;
|
2011-08-22 10:42:13 +01:00
|
|
|
|
2011-08-25 16:10:03 +01:00
|
|
|
sm_disk_detail::sm_metadata<MD_BLOCK_SIZE>::ptr metadata_sm_;
|
2011-08-24 10:45:39 +01:00
|
|
|
sm_disk_detail::sm_disk<MD_BLOCK_SIZE>::ptr data_sm_;
|
2011-06-23 14:47:08 +01:00
|
|
|
detail_tree details_;
|
|
|
|
dev_tree mappings_top_level_;
|
|
|
|
mapping_tree mappings_;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
//----------------------------------------------------------------
|
|
|
|
|
|
|
|
#endif
|