[thin_pool] Complete device creation

- Set up the device details
- Cache device details of opened devices
- Update changed devices at once while committing the pool
This commit is contained in:
Ming-Hung Tsai 2020-05-31 20:14:42 +08:00
parent 77adb08c3f
commit 7f7ba950ef
2 changed files with 114 additions and 31 deletions

View File

@ -33,7 +33,20 @@ using namespace thin_provisioning;
thin::thin(thin_dev_t dev, thin_pool &pool) thin::thin(thin_dev_t dev, thin_pool &pool)
: dev_(dev), : dev_(dev),
pool_(pool) pool_(pool),
details_(pool.get_transaction_id(), pool.get_time()),
open_count_(1),
changed_(true)
{
}
thin::thin(thin_dev_t dev, thin_pool &pool,
device_tree_detail::device_details const &details)
: dev_(dev),
pool_(pool),
details_(details),
open_count_(1),
changed_(false)
{ {
} }
@ -54,6 +67,10 @@ bool
thin::insert(block_address thin_block, block_address data_block) thin::insert(block_address thin_block, block_address data_block)
{ {
uint64_t key[2] = {dev_, thin_block}; uint64_t key[2] = {dev_, thin_block};
++details_.mapped_blocks_;
changed_ = true;
mapping_tree_detail::block_time bt; mapping_tree_detail::block_time bt;
bt.block_ = data_block; bt.block_ = data_block;
bt.time_ = 0; // FIXME: use current time. bt.time_ = 0; // FIXME: use current time.
@ -65,41 +82,29 @@ thin::remove(block_address thin_block)
{ {
uint64_t key[2] = {dev_, thin_block}; uint64_t key[2] = {dev_, thin_block};
pool_.md_->mappings_->remove(key); pool_.md_->mappings_->remove(key);
--details_.mapped_blocks_;
changed_ = true;
} }
void void
thin::set_snapshot_time(uint32_t time) thin::set_snapshot_time(uint32_t time)
{ {
uint64_t key[1] = { dev_ }; details_.snapshotted_time_ = time;
boost::optional<device_tree_detail::device_details> mdetail = pool_.md_->details_->lookup(key); changed_ = true;
if (!mdetail)
throw runtime_error("no such device");
mdetail->snapshotted_time_ = time;
pool_.md_->details_->insert(key, *mdetail);
} }
block_address block_address
thin::get_mapped_blocks() const thin::get_mapped_blocks() const
{ {
uint64_t key[1] = { dev_ }; return details_.mapped_blocks_;
boost::optional<device_tree_detail::device_details> mdetail = pool_.md_->details_->lookup(key);
if (!mdetail)
throw runtime_error("no such device");
return mdetail->mapped_blocks_;
} }
void void
thin::set_mapped_blocks(block_address count) thin::set_mapped_blocks(block_address count)
{ {
uint64_t key[1] = { dev_ }; details_.mapped_blocks_ = count;
boost::optional<device_tree_detail::device_details> mdetail = pool_.md_->details_->lookup(key); changed_ = true;
if (!mdetail)
throw runtime_error("no such device");
mdetail->mapped_blocks_ = count;
pool_.md_->details_->insert(key, *mdetail);
} }
//-------------------------------- //--------------------------------
@ -131,14 +136,15 @@ thin_pool::create_thin(thin_dev_t dev)
uint64_t key[1] = {dev}; uint64_t key[1] = {dev};
if (device_exists(dev)) if (device_exists(dev))
throw std::runtime_error("Device already exists"); throw std::runtime_error("device already exists");
single_mapping_tree::ptr new_tree(new single_mapping_tree(*md_->tm_, single_mapping_tree::ptr new_tree(new single_mapping_tree(*md_->tm_,
mapping_tree_detail::block_time_ref_counter(md_->data_sm_))); mapping_tree_detail::block_time_ref_counter(md_->data_sm_)));
md_->mappings_top_level_->insert(key, new_tree->get_root()); md_->mappings_top_level_->insert(key, new_tree->get_root());
md_->mappings_->set_root(md_->mappings_top_level_->get_root()); // FIXME: ugly md_->mappings_->set_root(md_->mappings_top_level_->get_root()); // FIXME: ugly
// FIXME: doesn't set up the device details thin::ptr r = create_device(dev);
close_device(r);
} }
void void
@ -174,6 +180,13 @@ thin_pool::del(thin_dev_t dev)
md_->mappings_top_level_->remove(key); md_->mappings_top_level_->remove(key);
} }
void
thin_pool::commit()
{
write_changed_details();
md_->commit();
}
void void
thin_pool::set_transaction_id(uint64_t id) thin_pool::set_transaction_id(uint64_t id)
{ {
@ -226,17 +239,16 @@ thin_pool::get_data_dev_size() const
return md_->data_sm_->get_nr_blocks(); return md_->data_sm_->get_nr_blocks();
} }
uint32_t
thin_pool::get_time() const
{
return md_->sb_.time_;
}
thin::ptr thin::ptr
thin_pool::open_thin(thin_dev_t dev) thin_pool::open_thin(thin_dev_t dev)
{ {
uint64_t key[1] = {dev}; return open_device(dev);
boost::optional<device_tree_detail::device_details> mdetails = md_->details_->lookup(key);
if (!mdetails)
throw runtime_error("no such device");
thin *ptr = new thin(dev, *this);
thin::ptr r(ptr);
return r;
} }
bool bool
@ -246,4 +258,61 @@ thin_pool::device_exists(thin_dev_t dev) const
return !!md_->details_->lookup(key); return !!md_->details_->lookup(key);
} }
thin::ptr
thin_pool::create_device(thin_dev_t dev)
{
device_map::iterator it = thin_devices_.find(dev);
if (it != thin_devices_.end())
throw std::runtime_error("device already exists");
thin::ptr td(new thin(dev, *this));
thin_devices_[dev] = td;
return td;
}
thin::ptr
thin_pool::open_device(thin_dev_t dev)
{
device_map::iterator it = thin_devices_.find(dev);
if (it != thin_devices_.end()) {
thin::ptr td = it->second;
td->open_count_++;
return td;
}
uint64_t key[1] = {dev};
device_tree::maybe_value details = md_->details_->lookup(key);
if (!details)
throw std::runtime_error("no such device");
thin::ptr td(new thin(dev, *this, *details));
thin_devices_[dev] = td;
return td;
}
void
thin_pool::close_device(thin::ptr td)
{
td->open_count_--;
}
void
thin_pool::write_changed_details()
{
for (auto it = thin_devices_.cbegin(); it != thin_devices_.cend(); ) {
uint64_t key[1] = {it->first};
thin::ptr td = it->second;
if (td->changed_) {
md_->details_->insert(key, td->details_);
td->changed_ = false;
}
if (!td->open_count_)
it = thin_devices_.erase(it);
else
++it;
}
}
//---------------------------------------------------------------- //----------------------------------------------------------------

View File

@ -49,9 +49,14 @@ namespace thin_provisioning {
private: private:
friend class thin_pool; friend class thin_pool;
thin(thin_dev_t dev, thin_pool &pool); thin(thin_dev_t dev, thin_pool &pool);
thin(thin_dev_t dev, thin_pool &pool,
device_tree_detail::device_details const &details);
thin_dev_t dev_; thin_dev_t dev_;
thin_pool &pool_; thin_pool &pool_;
device_tree_detail::device_details details_;
uint32_t open_count_;
bool changed_;
}; };
class thin_pool { class thin_pool {
@ -67,6 +72,7 @@ namespace thin_provisioning {
void create_thin(thin_dev_t dev); void create_thin(thin_dev_t dev);
void create_snap(thin_dev_t dev, thin_dev_t origin); void create_snap(thin_dev_t dev, thin_dev_t origin);
void del(thin_dev_t); void del(thin_dev_t);
void commit();
void set_transaction_id(uint64_t id); void set_transaction_id(uint64_t id);
uint64_t get_transaction_id() const; uint64_t get_transaction_id() const;
@ -80,14 +86,22 @@ namespace thin_provisioning {
block_address get_nr_free_data_blocks() const; block_address get_nr_free_data_blocks() const;
sector_t get_data_block_size() const; sector_t get_data_block_size() const;
block_address get_data_dev_size() const; block_address get_data_dev_size() const;
uint32_t get_time() const;
thin::ptr open_thin(thin_dev_t); thin::ptr open_thin(thin_dev_t);
private: private:
friend class thin; friend class thin;
typedef std::map<thin_dev_t, thin::ptr> device_map;
bool device_exists(thin_dev_t dev) const; bool device_exists(thin_dev_t dev) const;
thin::ptr create_device(thin_dev_t dev);
thin::ptr open_device(thin_dev_t dev);
void close_device(thin::ptr device);
void write_changed_details();
metadata::ptr md_; metadata::ptr md_;
device_map thin_devices_;
}; };
}; };