thin-provisioning-tools/metadata.h

177 lines
4.0 KiB
C
Raw Normal View History

#ifndef MULTISNAP_METADATA_H
#define MULTISNAP_METADATA_H
#include "block.h"
#include "transaction_manager.h"
#include "btree.h"
2011-08-22 15:12:13 +05:30
#include "endian_utils.h"
2011-07-22 20:39:56 +05:30
#include "metadata_disk_structures.h"
#include <string>
#include <boost/shared_ptr.hpp>
//----------------------------------------------------------------
2011-07-22 20:39:56 +05:30
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;
2011-07-22 20:39:56 +05:30
typedef uint32_t thin_dev_t;
2011-07-22 20:39:56 +05:30
//------------------------------------------------
2011-06-27 15:15:30 +05:30
2011-07-22 20:39:56 +05:30
class space_map_ref_counter {
public:
space_map_ref_counter(space_map::ptr sm)
: sm_(sm) {
}
2011-06-27 15:15:30 +05:30
2011-07-22 20:39:56 +05:30
void inc(block_address b) {
sm_->inc(b);
}
2011-06-27 15:15:30 +05:30
2011-07-22 20:39:56 +05:30
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;
2011-06-27 15:15:30 +05:30
2011-07-22 20:39:56 +05:30
static void unpack(disk_type const &disk, value_type &value) {
value = base::to_cpu<uint64_t>(disk);
}
static void pack(value_type const &value, disk_type &disk) {
disk = base::to_disk<base::__le64>(value);
2011-06-27 15:15:30 +05:30
}
};
2011-07-22 20:39:56 +05:30
//------------------------------------------------
template <uint32_t BlockSize>
class mtree_ref_counter {
2011-06-27 15:15:30 +05:30
public:
2011-07-22 20:39:56 +05:30
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 15:15:30 +05:30
typedef base::__le64 disk_type;
2011-07-22 20:39:56 +05:30
typedef uint64_t value_type;
typedef mtree_ref_counter<BlockSize> ref_counter;
2011-06-27 15:15:30 +05:30
2011-07-22 20:39:56 +05:30
static void unpack(disk_type const &disk, value_type &value) {
value = base::to_cpu<uint64_t>(disk);
}
2011-06-27 15:15:30 +05:30
2011-07-22 20:39:56 +05:30
static void pack(value_type const &value, disk_type &disk) {
disk = base::to_disk<base::__le64>(value);
2011-06-27 15:15:30 +05:30
}
};
2011-07-22 20:39:56 +05:30
class metadata;
class thin {
public:
typedef boost::shared_ptr<thin> ptr;
typedef boost::optional<block_address> 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_;
};
2011-06-27 15:15:30 +05:30
class metadata {
public:
typedef boost::shared_ptr<metadata> ptr;
2011-08-22 15:12:13 +05:30
metadata(std::string const &dev_path);
~metadata();
void commit();
2011-07-22 20:39:56 +05:30
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;
2011-07-22 20:39:56 +05:30
thin::ptr open_thin(thin_dev_t);
2011-08-23 16:25:37 +05:30
// Validation and repair
void check();
private:
friend class thin;
2011-07-22 20:39:56 +05:30
bool device_exists(thin_dev_t dev) const;
2011-07-22 20:39:56 +05:30
typedef persistent_data::transaction_manager<MD_BLOCK_SIZE>::ptr tm_ptr;
2011-07-22 20:39:56 +05:30
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-08-23 16:25:37 +05:30
// Declaration order is important here
2011-07-22 20:39:56 +05:30
tm_ptr tm_;
2011-08-23 16:25:37 +05:30
superblock sb_;
2011-08-22 15:12:13 +05:30
// Ignoring the metadata sm for now, since we don't need it for the basic 'dump' tool
// space_map::ptr metadata_sm_;
2011-07-22 20:39:56 +05:30
space_map::ptr data_sm_;
detail_tree details_;
dev_tree mappings_top_level_;
mapping_tree mappings_;
};
};
//----------------------------------------------------------------
#endif