2014-06-09 13:26:55 +01:00
|
|
|
#include <boost/lexical_cast.hpp>
|
|
|
|
#include <boost/optional.hpp>
|
2014-06-09 10:37:46 +01:00
|
|
|
#include <getopt.h>
|
|
|
|
#include <iostream>
|
2014-06-09 10:48:29 +01:00
|
|
|
#include <libgen.h>
|
2014-06-09 10:37:46 +01:00
|
|
|
|
|
|
|
#include "version.h"
|
|
|
|
|
2015-03-25 10:09:39 +00:00
|
|
|
#include "base/indented_stream.h"
|
2020-07-27 10:48:54 +08:00
|
|
|
#include "base/run.h"
|
2014-06-10 16:38:20 +01:00
|
|
|
#include "persistent-data/data-structures/btree_damage_visitor.h"
|
|
|
|
#include "persistent-data/space-maps/core.h"
|
2015-03-25 10:09:39 +00:00
|
|
|
#include "persistent-data/space-maps/disk.h"
|
2014-06-10 16:38:20 +01:00
|
|
|
#include "persistent-data/file_utils.h"
|
|
|
|
#include "thin-provisioning/superblock.h"
|
|
|
|
#include "thin-provisioning/mapping_tree.h"
|
2015-12-15 10:08:07 +00:00
|
|
|
#include "thin-provisioning/metadata.h"
|
2014-08-27 14:01:31 +01:00
|
|
|
#include "thin-provisioning/commands.h"
|
2014-06-10 16:38:20 +01:00
|
|
|
|
2014-06-09 10:37:46 +01:00
|
|
|
using namespace std;
|
2014-06-10 16:38:20 +01:00
|
|
|
using namespace thin_provisioning;
|
2014-06-09 10:37:46 +01:00
|
|
|
|
|
|
|
//----------------------------------------------------------------
|
|
|
|
|
2014-08-27 14:01:31 +01:00
|
|
|
namespace local {
|
2014-06-09 13:26:55 +01:00
|
|
|
class application {
|
|
|
|
public:
|
|
|
|
application(string const &cmd)
|
|
|
|
: cmd_(cmd) {
|
|
|
|
}
|
|
|
|
|
|
|
|
void usage(ostream &out) {
|
2015-03-25 10:09:39 +00:00
|
|
|
out << "Usage: " << cmd_ << " [options] <device or file>\n"
|
2014-06-17 15:00:38 +01:00
|
|
|
<< "Options:\n"
|
2015-03-25 10:09:39 +00:00
|
|
|
<< " {--thin1, --snap1}\n"
|
|
|
|
<< " {--thin2, --snap2}\n"
|
2015-03-25 11:10:18 +00:00
|
|
|
<< " {-m, --metadata-snap} [block#]\n"
|
2014-06-17 15:00:38 +01:00
|
|
|
<< " {--verbose}\n"
|
|
|
|
<< " {-h|--help}\n"
|
2014-06-09 13:26:55 +01:00
|
|
|
<< " {-V|--version}" << endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
void die(string const &msg) {
|
|
|
|
cerr << msg << endl;
|
|
|
|
usage(cerr);
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2015-03-25 11:10:18 +00:00
|
|
|
uint64_t parse_int(string const &str, string const &desc) {
|
2014-06-09 13:26:55 +01:00
|
|
|
try {
|
2014-06-10 16:38:20 +01:00
|
|
|
return boost::lexical_cast<uint64_t>(str);
|
2014-06-09 13:26:55 +01:00
|
|
|
|
|
|
|
} catch (...) {
|
|
|
|
ostringstream out;
|
2015-03-25 11:10:18 +00:00
|
|
|
out << "Couldn't parse " << desc << ": '" << str << "'";
|
2014-06-09 13:26:55 +01:00
|
|
|
die(out.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0; // never get here
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
string cmd_;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct flags {
|
2014-06-17 14:20:33 +01:00
|
|
|
flags()
|
2015-07-28 11:26:58 +01:00
|
|
|
: verbose(false),
|
2015-12-15 10:08:07 +00:00
|
|
|
use_metadata_snap(false) {
|
2014-06-17 14:20:33 +01:00
|
|
|
}
|
|
|
|
|
2015-12-15 10:08:07 +00:00
|
|
|
bool verbose;
|
|
|
|
bool use_metadata_snap;
|
|
|
|
|
2014-06-09 13:26:55 +01:00
|
|
|
boost::optional<string> dev;
|
2015-03-25 11:10:18 +00:00
|
|
|
boost::optional<uint64_t> metadata_snap;
|
2014-06-10 16:38:20 +01:00
|
|
|
boost::optional<uint64_t> snap1;
|
|
|
|
boost::optional<uint64_t> snap2;
|
2014-06-09 13:26:55 +01:00
|
|
|
};
|
2014-06-10 16:38:20 +01:00
|
|
|
|
|
|
|
//--------------------------------
|
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
struct mapping {
|
|
|
|
mapping()
|
|
|
|
: vbegin_(0),
|
|
|
|
dbegin_(0),
|
|
|
|
len_(0) {
|
|
|
|
}
|
|
|
|
|
|
|
|
mapping(uint64_t vbegin, uint64_t dbegin, uint64_t len)
|
|
|
|
: vbegin_(vbegin),
|
|
|
|
dbegin_(dbegin),
|
|
|
|
len_(len) {
|
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t vbegin_, dbegin_, len_;
|
|
|
|
};
|
|
|
|
|
2014-06-17 16:20:38 +01:00
|
|
|
ostream &operator <<(ostream &out, mapping const &m) {
|
|
|
|
out << "mapping[vbegin = " << m.vbegin_
|
|
|
|
<< ", dbegin = " << m.dbegin_
|
|
|
|
<< ", len = " << m.len_ << "]";
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
2016-02-17 15:17:02 +00:00
|
|
|
//--------------------------------
|
|
|
|
|
|
|
|
template <typename Container>
|
|
|
|
class mapping_stream {
|
|
|
|
public:
|
|
|
|
mapping_stream(Container const &c)
|
|
|
|
: it_(c.begin()),
|
|
|
|
end_(c.end()) {
|
|
|
|
m_ = *it_;
|
|
|
|
}
|
|
|
|
|
|
|
|
mapping const &get_mapping() const {
|
|
|
|
return m_;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool more_mappings() const {
|
|
|
|
return it_ != end_;
|
|
|
|
}
|
|
|
|
|
|
|
|
void consume(uint64_t delta) {
|
|
|
|
if (it_ == end_)
|
|
|
|
throw runtime_error("end of stream already reached");
|
|
|
|
|
|
|
|
if (delta > m_.len_)
|
|
|
|
throw runtime_error("delta too long");
|
|
|
|
|
|
|
|
if (delta == m_.len_) {
|
|
|
|
++it_;
|
|
|
|
m_ = *it_;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
m_.vbegin_ += delta;
|
|
|
|
m_.dbegin_ += delta;
|
|
|
|
m_.len_ -= delta;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
typename Container::const_iterator it_;
|
|
|
|
typename Container::const_iterator end_;
|
|
|
|
mapping m_;
|
|
|
|
};
|
|
|
|
|
|
|
|
//--------------------------------
|
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
typedef std::deque<mapping> mapping_deque;
|
|
|
|
|
|
|
|
// Builds up an in core rep of the mappings for a device.
|
|
|
|
class mapping_recorder {
|
2014-06-10 16:38:20 +01:00
|
|
|
public:
|
2014-06-17 12:39:13 +01:00
|
|
|
mapping_recorder() {
|
2015-03-24 13:36:45 +00:00
|
|
|
no_range();
|
2014-06-10 16:38:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void visit(btree_path const &path, mapping_tree_detail::block_time const &bt) {
|
2014-06-17 12:39:13 +01:00
|
|
|
record(path[0], bt.block_);
|
|
|
|
}
|
|
|
|
|
2015-03-24 13:36:45 +00:00
|
|
|
void complete() {
|
|
|
|
if (range_in_progress()) {
|
|
|
|
push_range();
|
|
|
|
no_range();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
mapping_deque const &get_mappings() const {
|
|
|
|
return mappings_;
|
2014-06-10 16:38:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2015-03-24 13:36:45 +00:00
|
|
|
void no_range() {
|
2014-06-10 16:38:20 +01:00
|
|
|
obegin_ = oend_ = 0;
|
|
|
|
dbegin_ = dend_ = 0;
|
|
|
|
}
|
|
|
|
|
2015-03-24 13:36:45 +00:00
|
|
|
void inc_range() {
|
|
|
|
oend_++;
|
|
|
|
dend_++;
|
|
|
|
}
|
2014-06-10 16:38:20 +01:00
|
|
|
|
2015-03-24 13:36:45 +00:00
|
|
|
void begin_range(uint64_t oblock, uint64_t dblock) {
|
|
|
|
obegin_ = oend_ = oblock;
|
|
|
|
dbegin_ = dend_ = dblock;
|
|
|
|
inc_range();
|
|
|
|
}
|
2014-06-10 16:38:20 +01:00
|
|
|
|
2015-03-24 13:36:45 +00:00
|
|
|
bool range_in_progress() {
|
|
|
|
return oend_ != obegin_;
|
|
|
|
}
|
2014-06-10 16:38:20 +01:00
|
|
|
|
2015-03-24 13:36:45 +00:00
|
|
|
bool continues_range(uint64_t oblock, uint64_t dblock) {
|
|
|
|
return (oblock == oend_) && (dblock == dend_);
|
|
|
|
}
|
2014-06-10 16:38:20 +01:00
|
|
|
|
2015-03-24 13:36:45 +00:00
|
|
|
void push_range() {
|
|
|
|
mapping m(obegin_, dbegin_, oend_ - obegin_);
|
|
|
|
mappings_.push_back(m);
|
2014-06-10 16:38:20 +01:00
|
|
|
}
|
|
|
|
|
2015-03-24 13:36:45 +00:00
|
|
|
void record(uint64_t oblock, uint64_t dblock) {
|
|
|
|
if (!range_in_progress())
|
|
|
|
begin_range(oblock, dblock);
|
|
|
|
|
|
|
|
else if (!continues_range(oblock, dblock)) {
|
|
|
|
push_range();
|
|
|
|
begin_range(oblock, dblock);
|
|
|
|
} else
|
|
|
|
inc_range();
|
2014-06-17 12:39:13 +01:00
|
|
|
}
|
|
|
|
|
2014-06-10 16:38:20 +01:00
|
|
|
uint64_t obegin_, oend_;
|
|
|
|
uint64_t dbegin_, dend_;
|
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
mapping_deque mappings_;
|
2014-06-10 16:38:20 +01:00
|
|
|
};
|
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
//--------------------------------
|
|
|
|
|
2014-06-10 16:38:20 +01:00
|
|
|
class damage_visitor {
|
|
|
|
public:
|
|
|
|
virtual void visit(btree_path const &path, btree_detail::damage const &d) {
|
|
|
|
throw std::runtime_error("damage in mapping tree, please run thin_check");
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-06-17 14:20:33 +01:00
|
|
|
//--------------------------------
|
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
class diff_emitter {
|
|
|
|
public:
|
2015-03-25 10:09:39 +00:00
|
|
|
diff_emitter(indented_stream &out)
|
2014-06-17 12:39:13 +01:00
|
|
|
: out_(out) {
|
|
|
|
}
|
2014-06-10 16:38:20 +01:00
|
|
|
|
2014-06-17 14:20:33 +01:00
|
|
|
virtual void left_only(uint64_t vbegin, uint64_t dbegin, uint64_t len) = 0;
|
|
|
|
virtual void right_only(uint64_t vbegin, uint64_t dbegin, uint64_t len) = 0;
|
|
|
|
virtual void blocks_differ(uint64_t vbegin, uint64_t left_dbegin, uint64_t right_dbegin, uint64_t len) = 0;
|
|
|
|
virtual void blocks_same(uint64_t vbegin, uint64_t dbegin, uint64_t len) = 0;
|
|
|
|
virtual void complete() = 0;
|
|
|
|
|
|
|
|
protected:
|
2015-03-25 10:09:39 +00:00
|
|
|
void indent() {
|
|
|
|
out_.indent();
|
|
|
|
}
|
|
|
|
|
|
|
|
indented_stream &out() {
|
2014-06-17 14:20:33 +01:00
|
|
|
return out_;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2015-03-25 10:09:39 +00:00
|
|
|
indented_stream &out_;
|
2014-06-17 14:20:33 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class simple_emitter : public diff_emitter {
|
|
|
|
public:
|
2015-03-25 10:09:39 +00:00
|
|
|
simple_emitter(indented_stream &out)
|
2014-06-17 14:20:33 +01:00
|
|
|
: diff_emitter(out) {
|
|
|
|
}
|
|
|
|
|
|
|
|
void left_only(uint64_t vbegin, uint64_t dbegin, uint64_t len) {
|
|
|
|
add_range(LEFT_ONLY, vbegin, len);
|
|
|
|
}
|
|
|
|
|
|
|
|
void right_only(uint64_t vbegin, uint64_t dbegin, uint64_t len) {
|
|
|
|
add_range(RIGHT_ONLY, vbegin, len);
|
|
|
|
}
|
|
|
|
|
|
|
|
void blocks_differ(uint64_t vbegin, uint64_t left_dbegin, uint64_t right_dbegin, uint64_t len) {
|
|
|
|
add_range(DIFFER, vbegin, len);
|
|
|
|
}
|
|
|
|
|
|
|
|
void blocks_same(uint64_t vbegin, uint64_t dbegin, uint64_t len) {
|
|
|
|
add_range(SAME, vbegin, len);
|
|
|
|
}
|
|
|
|
|
|
|
|
void complete() {
|
|
|
|
if (current_type_)
|
|
|
|
emit_range();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
enum block_type {
|
|
|
|
LEFT_ONLY,
|
|
|
|
RIGHT_ONLY,
|
|
|
|
DIFFER,
|
|
|
|
SAME
|
|
|
|
};
|
|
|
|
|
|
|
|
void add_range(block_type t, uint64_t vbegin, uint64_t len) {
|
|
|
|
if (current_type_ && *current_type_ == t && vbegin == vend_) {
|
|
|
|
vend_ += len;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
emit_range();
|
|
|
|
current_type_ = t;
|
|
|
|
vbegin_ = vbegin;
|
|
|
|
vend_ = vbegin_ + len;
|
|
|
|
}
|
|
|
|
|
|
|
|
void emit_range() {
|
|
|
|
if (!current_type_)
|
|
|
|
return;
|
|
|
|
|
2015-03-25 10:09:39 +00:00
|
|
|
indent();
|
2014-06-17 14:20:33 +01:00
|
|
|
switch (*current_type_) {
|
|
|
|
case LEFT_ONLY:
|
|
|
|
out() << "<left_only";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RIGHT_ONLY:
|
|
|
|
out() << "<right_only";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DIFFER:
|
|
|
|
out() << "<different";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SAME:
|
|
|
|
out() << "<same";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
out() << " begin=\"" << vbegin_ << "\""
|
|
|
|
<< " length=\"" << vend_ - vbegin_ << "\"/>\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
boost::optional<block_type> current_type_;
|
|
|
|
uint64_t vbegin_, vend_;
|
|
|
|
};
|
|
|
|
|
|
|
|
class verbose_emitter : public diff_emitter {
|
|
|
|
public:
|
2015-03-25 10:09:39 +00:00
|
|
|
verbose_emitter(indented_stream &out)
|
2014-06-17 14:20:33 +01:00
|
|
|
: diff_emitter(out) {
|
|
|
|
}
|
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
void left_only(uint64_t vbegin, uint64_t dbegin, uint64_t len) {
|
|
|
|
begin_block(LEFT_ONLY);
|
2015-03-25 10:09:39 +00:00
|
|
|
indent();
|
|
|
|
out() << "<range begin=\"" << vbegin << "\""
|
2014-06-17 14:20:33 +01:00
|
|
|
<< " data_begin=\"" << dbegin << "\""
|
|
|
|
<< " length=\"" << len << "\"/>\n";
|
2014-06-17 12:39:13 +01:00
|
|
|
}
|
2014-06-10 16:38:20 +01:00
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
void right_only(uint64_t vbegin, uint64_t dbegin, uint64_t len) {
|
|
|
|
begin_block(RIGHT_ONLY);
|
2015-03-25 10:09:39 +00:00
|
|
|
indent();
|
|
|
|
out() << "<range begin=\"" << vbegin << "\""
|
2014-06-17 14:20:33 +01:00
|
|
|
<< " data_begin=\"" << dbegin << "\""
|
|
|
|
<< " length=\"" << len << "\"/>\n";
|
2014-06-17 12:39:13 +01:00
|
|
|
}
|
2014-06-10 16:38:20 +01:00
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
void blocks_differ(uint64_t vbegin, uint64_t left_dbegin, uint64_t right_dbegin, uint64_t len) {
|
|
|
|
begin_block(DIFFER);
|
2015-03-25 10:09:39 +00:00
|
|
|
indent();
|
|
|
|
out() << "<range begin=\"" << vbegin << "\""
|
2014-06-17 14:20:33 +01:00
|
|
|
<< " left_data_begin=\"" << left_dbegin << "\""
|
|
|
|
<< " right_data_begin=\"" << right_dbegin << "\""
|
|
|
|
<< " length=\"" << len << "\"/>\n";
|
2014-06-17 12:39:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void blocks_same(uint64_t vbegin, uint64_t dbegin, uint64_t len) {
|
|
|
|
begin_block(SAME);
|
2015-03-25 10:09:39 +00:00
|
|
|
indent();
|
|
|
|
out() << "<range begin=\"" << vbegin << "\""
|
2014-06-17 14:20:33 +01:00
|
|
|
<< " data_begin=\"" << dbegin << "\""
|
|
|
|
<< " length=\"" << len << "\"/>\n";
|
2014-06-17 12:39:13 +01:00
|
|
|
}
|
2014-06-10 16:38:20 +01:00
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
void complete() {
|
|
|
|
if (current_type_)
|
|
|
|
close(*current_type_);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
enum block_type {
|
|
|
|
LEFT_ONLY,
|
|
|
|
RIGHT_ONLY,
|
|
|
|
DIFFER,
|
|
|
|
SAME
|
|
|
|
};
|
|
|
|
|
|
|
|
void begin_block(block_type t) {
|
|
|
|
if (!current_type_) {
|
|
|
|
current_type_ = t;
|
|
|
|
open(t);
|
|
|
|
|
|
|
|
} else if (*current_type_ != t) {
|
|
|
|
close(*current_type_);
|
|
|
|
current_type_ = t;
|
|
|
|
open(t);
|
|
|
|
}
|
2014-06-10 16:38:20 +01:00
|
|
|
}
|
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
void open(block_type t) {
|
2015-03-25 10:09:39 +00:00
|
|
|
indent();
|
2014-06-17 12:39:13 +01:00
|
|
|
switch (t) {
|
|
|
|
case LEFT_ONLY:
|
2014-06-17 14:20:33 +01:00
|
|
|
out() << "<left_only>\n";
|
2014-06-17 12:39:13 +01:00
|
|
|
break;
|
2014-06-10 16:38:20 +01:00
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
case RIGHT_ONLY:
|
2014-06-17 14:20:33 +01:00
|
|
|
out() << "<right_only>\n";
|
2014-06-17 12:39:13 +01:00
|
|
|
break;
|
2014-06-10 16:38:20 +01:00
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
case DIFFER:
|
2014-06-17 14:20:33 +01:00
|
|
|
out() << "<different>\n";
|
2014-06-17 12:39:13 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SAME:
|
2014-06-17 14:20:33 +01:00
|
|
|
out() << "<same>\n";
|
2014-06-17 12:39:13 +01:00
|
|
|
break;
|
|
|
|
}
|
2015-03-25 10:09:39 +00:00
|
|
|
out().inc();
|
2014-06-10 16:38:20 +01:00
|
|
|
}
|
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
void close(block_type t) {
|
2015-03-25 10:09:39 +00:00
|
|
|
out().dec();
|
|
|
|
indent();
|
2014-06-17 12:39:13 +01:00
|
|
|
switch (t) {
|
|
|
|
case LEFT_ONLY:
|
2015-03-25 10:09:39 +00:00
|
|
|
out() << "</left_only>\n";
|
2014-06-17 12:39:13 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case RIGHT_ONLY:
|
2015-03-25 10:09:39 +00:00
|
|
|
out() << "</right_only>\n";
|
2014-06-17 12:39:13 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case DIFFER:
|
2015-03-25 10:09:39 +00:00
|
|
|
out() << "</different>\n";
|
2014-06-17 12:39:13 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case SAME:
|
2015-03-25 10:09:39 +00:00
|
|
|
out() << "</same>\n";
|
2014-06-17 12:39:13 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2014-06-10 16:38:20 +01:00
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
boost::optional<block_type> current_type_;
|
|
|
|
};
|
|
|
|
|
|
|
|
//----------------------------------------------------------------
|
|
|
|
|
|
|
|
void dump_diff(mapping_deque const &left,
|
2014-06-17 14:20:33 +01:00
|
|
|
mapping_deque const &right,
|
|
|
|
diff_emitter &e) {
|
2014-06-17 12:39:13 +01:00
|
|
|
|
|
|
|
// We iterate through both sets of mappings in parallel
|
|
|
|
// noting any differences.
|
2016-02-17 15:17:02 +00:00
|
|
|
mapping_stream<mapping_deque> ls{left};
|
|
|
|
mapping_stream<mapping_deque> rs{right};
|
|
|
|
|
|
|
|
while (ls.more_mappings() && rs.more_mappings()) {
|
|
|
|
auto &lm = ls.get_mapping();
|
|
|
|
auto &rm = rs.get_mapping();
|
|
|
|
|
|
|
|
if (lm.vbegin_ < rm.vbegin_) {
|
|
|
|
auto delta = min<uint64_t>(lm.len_, rm.vbegin_ - lm.vbegin_);
|
|
|
|
e.left_only(lm.vbegin_, lm.dbegin_, delta);
|
|
|
|
ls.consume(delta);
|
|
|
|
|
|
|
|
} else if (lm.vbegin_ > rm.vbegin_) {
|
|
|
|
auto delta = min<uint64_t>(rm.len_, lm.vbegin_ - rm.vbegin_);
|
|
|
|
e.right_only(rm.vbegin_, rm.dbegin_, delta);
|
|
|
|
rs.consume(delta);
|
|
|
|
|
|
|
|
} else if (lm.dbegin_ != rm.dbegin_) {
|
|
|
|
auto delta = min<uint64_t>(lm.len_, rm.len_);
|
|
|
|
e.blocks_differ(lm.vbegin_, lm.dbegin_, rm.dbegin_, delta);
|
|
|
|
ls.consume(delta);
|
|
|
|
rs.consume(delta);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
auto delta = min<uint64_t>(lm.len_, rm.len_);
|
|
|
|
e.blocks_same(lm.vbegin_, lm.dbegin_, delta);
|
|
|
|
ls.consume(delta);
|
|
|
|
rs.consume(delta);
|
2014-06-17 12:39:13 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-17 15:17:02 +00:00
|
|
|
while (ls.more_mappings()) {
|
|
|
|
auto &lm = ls.get_mapping();
|
|
|
|
e.left_only(lm.vbegin_, lm.dbegin_, lm.len_);
|
2016-02-18 11:31:43 +00:00
|
|
|
ls.consume(lm.len_);
|
2015-03-24 13:36:45 +00:00
|
|
|
}
|
|
|
|
|
2016-02-17 15:17:02 +00:00
|
|
|
while (rs.more_mappings()) {
|
|
|
|
auto &rm = rs.get_mapping();
|
|
|
|
e.right_only(rm.vbegin_, rm.dbegin_, rm.len_);
|
2016-02-18 11:31:43 +00:00
|
|
|
rs.consume(rm.len_);
|
2015-03-24 13:36:45 +00:00
|
|
|
}
|
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
e.complete();
|
|
|
|
}
|
|
|
|
|
2015-03-25 10:09:39 +00:00
|
|
|
// FIXME: duplication with xml_format
|
|
|
|
void begin_superblock(indented_stream &out,
|
|
|
|
string const &uuid,
|
|
|
|
uint64_t time,
|
|
|
|
uint64_t trans_id,
|
|
|
|
uint32_t data_block_size,
|
|
|
|
uint64_t nr_data_blocks,
|
|
|
|
boost::optional<uint64_t> metadata_snap) {
|
|
|
|
out.indent();
|
|
|
|
out << "<superblock uuid=\"" << uuid << "\""
|
|
|
|
<< " time=\"" << time << "\""
|
|
|
|
<< " transaction=\"" << trans_id << "\""
|
|
|
|
<< " data_block_size=\"" << data_block_size << "\""
|
|
|
|
<< " nr_data_blocks=\"" << nr_data_blocks;
|
|
|
|
|
|
|
|
if (metadata_snap)
|
|
|
|
out << "\" metadata_snap=\"" << *metadata_snap;
|
|
|
|
|
|
|
|
out << "\">\n";
|
|
|
|
out.inc();
|
|
|
|
}
|
|
|
|
|
|
|
|
void end_superblock(indented_stream &out) {
|
|
|
|
out.dec();
|
|
|
|
out.indent();
|
|
|
|
out << "</superblock>\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
void begin_diff(indented_stream &out, uint64_t snap1, uint64_t snap2) {
|
|
|
|
out.indent();
|
|
|
|
out << "<diff left=\"" << snap1 << "\" right=\"" << snap2 << "\">\n";
|
|
|
|
out.inc();
|
|
|
|
}
|
|
|
|
|
|
|
|
void end_diff(indented_stream &out) {
|
|
|
|
out.dec();
|
|
|
|
out.indent();
|
|
|
|
out << "</diff>\n";
|
|
|
|
}
|
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
void delta_(application &app, flags const &fs) {
|
|
|
|
mapping_recorder mr1;
|
|
|
|
mapping_recorder mr2;
|
2014-06-10 16:38:20 +01:00
|
|
|
damage_visitor damage_v;
|
2015-03-25 10:09:39 +00:00
|
|
|
superblock_detail::superblock sb;
|
2016-02-17 10:42:42 +00:00
|
|
|
block_address nr_data_blocks = 0ull;
|
2014-06-17 12:39:13 +01:00
|
|
|
|
|
|
|
{
|
2020-04-30 14:30:01 +01:00
|
|
|
block_manager::ptr bm = open_bm(*fs.dev, block_manager::READ_ONLY, !fs.use_metadata_snap);
|
2015-12-15 10:08:07 +00:00
|
|
|
metadata::ptr md(fs.use_metadata_snap ? new metadata(bm, fs.metadata_snap) : new metadata(bm));
|
|
|
|
sb = md->sb_;
|
2014-06-17 12:39:13 +01:00
|
|
|
|
|
|
|
dev_tree::key k = {*fs.snap1};
|
2015-12-15 10:08:07 +00:00
|
|
|
boost::optional<uint64_t> snap1_root = md->mappings_top_level_->lookup(k);
|
2014-06-17 12:39:13 +01:00
|
|
|
|
|
|
|
if (!snap1_root) {
|
|
|
|
ostringstream out;
|
|
|
|
out << "Unable to find mapping tree for snap1 (" << *fs.snap1 << ")";
|
|
|
|
app.die(out.str());
|
|
|
|
}
|
|
|
|
|
2015-12-15 10:08:07 +00:00
|
|
|
single_mapping_tree snap1(*md->tm_, *snap1_root,
|
|
|
|
mapping_tree_detail::block_traits::ref_counter(md->tm_->get_sm()));
|
2014-06-17 12:39:13 +01:00
|
|
|
|
|
|
|
k[0] = *fs.snap2;
|
2015-12-15 10:08:07 +00:00
|
|
|
boost::optional<uint64_t> snap2_root = md->mappings_top_level_->lookup(k);
|
2014-06-17 12:39:13 +01:00
|
|
|
|
|
|
|
if (!snap2_root) {
|
|
|
|
ostringstream out;
|
|
|
|
out << "Unable to find mapping tree for snap2 (" << *fs.snap2 << ")";
|
|
|
|
app.die(out.str());
|
|
|
|
}
|
|
|
|
|
2015-12-15 10:08:07 +00:00
|
|
|
single_mapping_tree snap2(*md->tm_, *snap2_root,
|
|
|
|
mapping_tree_detail::block_traits::ref_counter(md->tm_->get_sm()));
|
2014-06-17 12:39:13 +01:00
|
|
|
btree_visit_values(snap1, mr1, damage_v);
|
2015-03-24 13:36:45 +00:00
|
|
|
mr1.complete();
|
|
|
|
|
2014-06-17 12:39:13 +01:00
|
|
|
btree_visit_values(snap2, mr2, damage_v);
|
2015-03-24 13:36:45 +00:00
|
|
|
mr2.complete();
|
2016-02-17 10:42:42 +00:00
|
|
|
|
|
|
|
if (md->data_sm_)
|
|
|
|
nr_data_blocks = md->data_sm_->get_nr_blocks();
|
2014-06-17 12:39:13 +01:00
|
|
|
}
|
|
|
|
|
2015-03-25 10:09:39 +00:00
|
|
|
indented_stream is(cout);
|
|
|
|
begin_superblock(is, "", sb.time_,
|
|
|
|
sb.trans_id_,
|
|
|
|
sb.data_block_size_,
|
2016-02-17 10:42:42 +00:00
|
|
|
nr_data_blocks,
|
2015-03-25 10:09:39 +00:00
|
|
|
sb.metadata_snap_ ?
|
|
|
|
boost::optional<block_address>(sb.metadata_snap_) :
|
|
|
|
boost::optional<block_address>());
|
|
|
|
begin_diff(is, *fs.snap1, *fs.snap2);
|
|
|
|
|
2014-06-17 14:20:33 +01:00
|
|
|
if (fs.verbose) {
|
2015-03-25 10:09:39 +00:00
|
|
|
verbose_emitter e(is);
|
2014-06-17 14:20:33 +01:00
|
|
|
dump_diff(mr1.get_mappings(), mr2.get_mappings(), e);
|
|
|
|
} else {
|
2015-03-25 10:09:39 +00:00
|
|
|
simple_emitter e(is);
|
2014-06-17 14:20:33 +01:00
|
|
|
dump_diff(mr1.get_mappings(), mr2.get_mappings(), e);
|
|
|
|
}
|
2015-03-25 10:09:39 +00:00
|
|
|
|
|
|
|
end_diff(is);
|
|
|
|
end_superblock(is);
|
2014-06-10 16:38:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
int delta(application &app, flags const &fs) {
|
|
|
|
try {
|
|
|
|
delta_(app, fs);
|
|
|
|
} catch (exception const &e) {
|
|
|
|
app.die(e.what());
|
|
|
|
return 1; // never get here
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2014-06-09 10:48:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------
|
|
|
|
|
2014-06-09 13:26:55 +01:00
|
|
|
// FIXME: add metadata snap switch
|
|
|
|
|
2016-01-08 12:51:52 +00:00
|
|
|
thin_delta_cmd::thin_delta_cmd()
|
|
|
|
: command("thin_delta")
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
thin_delta_cmd::usage(std::ostream &out) const
|
|
|
|
{
|
|
|
|
// FIXME: finish
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
thin_delta_cmd::run(int argc, char **argv)
|
2014-06-09 10:37:46 +01:00
|
|
|
{
|
2014-08-27 14:01:31 +01:00
|
|
|
using namespace local;
|
|
|
|
|
2014-06-09 10:37:46 +01:00
|
|
|
int c;
|
2014-06-09 13:26:55 +01:00
|
|
|
flags fs;
|
2014-08-27 14:01:31 +01:00
|
|
|
local::application app(basename(argv[0]));
|
2014-06-09 13:26:55 +01:00
|
|
|
|
2015-07-28 11:26:58 +01:00
|
|
|
char const shortopts[] = "hVm::";
|
2014-06-09 10:37:46 +01:00
|
|
|
option const longopts[] = {
|
2014-06-09 10:48:29 +01:00
|
|
|
{ "help", no_argument, NULL, 'h' },
|
2014-06-09 13:26:55 +01:00
|
|
|
{ "version", no_argument, NULL, 'V' },
|
2015-03-25 10:09:39 +00:00
|
|
|
{ "thin1", required_argument, NULL, 1 },
|
2014-06-09 13:26:55 +01:00
|
|
|
{ "snap1", required_argument, NULL, 1 },
|
2015-03-25 10:09:39 +00:00
|
|
|
{ "thin2", required_argument, NULL, 2 },
|
2014-06-09 13:26:55 +01:00
|
|
|
{ "snap2", required_argument, NULL, 2 },
|
2015-07-28 11:26:58 +01:00
|
|
|
{ "metadata-snap", optional_argument, NULL, 'm' },
|
2017-09-15 12:28:19 +01:00
|
|
|
{ "verbose", no_argument, NULL, 4 },
|
|
|
|
{ NULL, no_argument, NULL, 0 }
|
2014-06-09 10:37:46 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
while ((c = getopt_long(argc, argv, shortopts, longopts, NULL)) != -1) {
|
|
|
|
switch (c) {
|
2014-06-09 10:48:29 +01:00
|
|
|
case 'h':
|
2014-06-09 13:26:55 +01:00
|
|
|
app.usage(cout);
|
2014-06-09 10:48:29 +01:00
|
|
|
return 0;
|
|
|
|
|
2014-06-09 10:37:46 +01:00
|
|
|
case 'V':
|
|
|
|
cout << THIN_PROVISIONING_TOOLS_VERSION << endl;
|
|
|
|
return 0;
|
2014-06-09 10:51:10 +01:00
|
|
|
|
2014-06-09 13:26:55 +01:00
|
|
|
case 1:
|
2015-03-25 11:10:18 +00:00
|
|
|
fs.snap1 = app.parse_int(optarg, "thin id 1");
|
2014-06-09 13:26:55 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 2:
|
2015-03-25 11:10:18 +00:00
|
|
|
fs.snap2 = app.parse_int(optarg, "thin id 2");
|
2014-06-09 13:26:55 +01:00
|
|
|
break;
|
|
|
|
|
2015-03-25 10:09:39 +00:00
|
|
|
case 'm':
|
2015-12-15 10:08:07 +00:00
|
|
|
fs.use_metadata_snap = true;
|
2015-07-28 11:26:58 +01:00
|
|
|
if (optarg)
|
|
|
|
fs.metadata_snap = app.parse_int(optarg, "metadata snapshot block");
|
2014-06-17 14:20:33 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 4:
|
|
|
|
fs.verbose = true;
|
|
|
|
break;
|
|
|
|
|
2014-06-09 10:51:10 +01:00
|
|
|
default:
|
2014-06-09 13:26:55 +01:00
|
|
|
app.usage(cerr);
|
2014-06-09 10:51:10 +01:00
|
|
|
return 1;
|
2014-06-09 10:37:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-09 13:26:55 +01:00
|
|
|
if (argc == optind)
|
|
|
|
app.die("No input device provided.");
|
|
|
|
else
|
|
|
|
fs.dev = argv[optind];
|
|
|
|
|
|
|
|
if (!fs.snap1)
|
|
|
|
app.die("--snap1 not specified.");
|
|
|
|
|
|
|
|
if (!fs.snap2)
|
|
|
|
app.die("--snap2 not specified.");
|
|
|
|
|
2014-06-10 16:38:20 +01:00
|
|
|
return delta(app, fs);
|
2014-06-09 10:37:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------
|