[thin_trim] first code drop.
No testing done as yet.
This commit is contained in:
parent
e476515087
commit
8e92158055
@ -86,6 +86,7 @@ SOURCE=\
|
|||||||
thin-provisioning/thin_repair.cc \
|
thin-provisioning/thin_repair.cc \
|
||||||
thin-provisioning/thin_restore.cc \
|
thin-provisioning/thin_restore.cc \
|
||||||
thin-provisioning/thin_rmap.cc \
|
thin-provisioning/thin_rmap.cc \
|
||||||
|
thin-provisioning/thin_trim.cc \
|
||||||
thin-provisioning/xml_format.cc
|
thin-provisioning/xml_format.cc
|
||||||
|
|
||||||
CC:=@CC@
|
CC:=@CC@
|
||||||
@ -171,6 +172,7 @@ install: bin/pdata_tools
|
|||||||
ln -s -f pdata_tools $(BINDIR)/thin_repair
|
ln -s -f pdata_tools $(BINDIR)/thin_repair
|
||||||
ln -s -f pdata_tools $(BINDIR)/thin_restore
|
ln -s -f pdata_tools $(BINDIR)/thin_restore
|
||||||
ln -s -f pdata_tools $(BINDIR)/thin_rmap
|
ln -s -f pdata_tools $(BINDIR)/thin_rmap
|
||||||
|
ln -s -f pdata_tools $(BINDIR)/thin_trim
|
||||||
ln -s -f pdata_tools $(BINDIR)/thin_metadata_size
|
ln -s -f pdata_tools $(BINDIR)/thin_metadata_size
|
||||||
ln -s -f pdata_tools $(BINDIR)/era_check
|
ln -s -f pdata_tools $(BINDIR)/era_check
|
||||||
ln -s -f pdata_tools $(BINDIR)/era_dump
|
ln -s -f pdata_tools $(BINDIR)/era_dump
|
||||||
@ -186,6 +188,7 @@ install: bin/pdata_tools
|
|||||||
$(INSTALL_DATA) man8/thin_repair.8 $(MANPATH)/man8
|
$(INSTALL_DATA) man8/thin_repair.8 $(MANPATH)/man8
|
||||||
$(INSTALL_DATA) man8/thin_restore.8 $(MANPATH)/man8
|
$(INSTALL_DATA) man8/thin_restore.8 $(MANPATH)/man8
|
||||||
$(INSTALL_DATA) man8/thin_rmap.8 $(MANPATH)/man8
|
$(INSTALL_DATA) man8/thin_rmap.8 $(MANPATH)/man8
|
||||||
|
$(INSTALL_DATA) man8/thin_trim.8 $(MANPATH)/man8
|
||||||
$(INSTALL_DATA) man8/thin_metadata_size.8 $(MANPATH)/man8
|
$(INSTALL_DATA) man8/thin_metadata_size.8 $(MANPATH)/man8
|
||||||
$(INSTALL_DATA) man8/era_check.8 $(MANPATH)/man8
|
$(INSTALL_DATA) man8/era_check.8 $(MANPATH)/man8
|
||||||
$(INSTALL_DATA) man8/era_dump.8 $(MANPATH)/man8
|
$(INSTALL_DATA) man8/era_dump.8 $(MANPATH)/man8
|
||||||
|
@ -13,6 +13,7 @@ namespace thin_provisioning {
|
|||||||
extern base::command thin_restore_cmd;
|
extern base::command thin_restore_cmd;
|
||||||
extern base::command thin_repair_cmd;
|
extern base::command thin_repair_cmd;
|
||||||
extern base::command thin_rmap_cmd;
|
extern base::command thin_rmap_cmd;
|
||||||
|
extern base::command thin_trim_cmd;
|
||||||
extern base::command thin_metadata_size_cmd;
|
extern base::command thin_metadata_size_cmd;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
198
thin-provisioning/thin_trim.cc
Normal file
198
thin-provisioning/thin_trim.cc
Normal file
@ -0,0 +1,198 @@
|
|||||||
|
#include <iostream>
|
||||||
|
#include <getopt.h>
|
||||||
|
#include <sys/ioctl.h>
|
||||||
|
#include <linux/fs.h>
|
||||||
|
|
||||||
|
#undef BLOCK_SIZE
|
||||||
|
|
||||||
|
#include "thin-provisioning/commands.h"
|
||||||
|
#include "metadata.h"
|
||||||
|
#include "version.h"
|
||||||
|
|
||||||
|
using namespace persistent_data;
|
||||||
|
using namespace std;
|
||||||
|
using namespace thin_provisioning;
|
||||||
|
|
||||||
|
//----------------------------------------------------------------
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
void confirm_pool_is_not_active() {
|
||||||
|
cout << "The pool must *not* be active when running this tool.\n"
|
||||||
|
<< "Do you wish to continue? [Y/N]\n"
|
||||||
|
<< endl;
|
||||||
|
|
||||||
|
string input;
|
||||||
|
cin >> input;
|
||||||
|
if (input != "Y")
|
||||||
|
exit(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
class discard_emitter {
|
||||||
|
public:
|
||||||
|
discard_emitter(string const &data_dev, unsigned block_size, uint64_t nr_blocks)
|
||||||
|
: fd_(open_dev(data_dev, block_size * nr_blocks)),
|
||||||
|
block_size_(block_size) {
|
||||||
|
}
|
||||||
|
|
||||||
|
~discard_emitter() {
|
||||||
|
::close(fd_);
|
||||||
|
}
|
||||||
|
|
||||||
|
void emit(block_address b, block_address e) {
|
||||||
|
uint64_t range[2];
|
||||||
|
|
||||||
|
range[0] = block_to_byte(b);
|
||||||
|
range[1] = block_to_byte(e) - range[0];
|
||||||
|
|
||||||
|
if (ioctl(fd_, BLKDISCARD, &range))
|
||||||
|
throw runtime_error("discard ioctl failed");
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
static int open_dev(string const &data_dev, uint64_t expected_size) {
|
||||||
|
int r, fd;
|
||||||
|
uint64_t blksize;
|
||||||
|
struct stat info;
|
||||||
|
|
||||||
|
fd = ::open(data_dev.c_str(), O_WRONLY);
|
||||||
|
if (fd < 0) {
|
||||||
|
ostringstream out;
|
||||||
|
out << "Couldn't open data device '" << data_dev << "'";
|
||||||
|
throw runtime_error(out.str());
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
r = fstat(fd, &info);
|
||||||
|
if (r)
|
||||||
|
throw runtime_error("Couldn't stat data device");
|
||||||
|
|
||||||
|
if (!S_ISBLK(info.st_mode))
|
||||||
|
throw runtime_error("Data device is not a block device");
|
||||||
|
|
||||||
|
r = ioctl(fd, BLKGETSIZE64, &blksize);
|
||||||
|
if (r)
|
||||||
|
throw runtime_error("Couldn't get data device size");
|
||||||
|
|
||||||
|
if (blksize != (expected_size << 9))
|
||||||
|
throw runtime_error("Data device is not the expected size");
|
||||||
|
|
||||||
|
} catch (...) {
|
||||||
|
::close(fd);
|
||||||
|
throw;
|
||||||
|
}
|
||||||
|
|
||||||
|
return fd;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint64_t block_to_byte(block_address b) {
|
||||||
|
return (b * block_size_) << 9;
|
||||||
|
}
|
||||||
|
|
||||||
|
int fd_;
|
||||||
|
unsigned block_size_;
|
||||||
|
};
|
||||||
|
|
||||||
|
class trim_visitor : public space_map_detail::visitor {
|
||||||
|
public:
|
||||||
|
trim_visitor(discard_emitter &e)
|
||||||
|
: emitter_(e) {
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void visit(space_map_detail::missing_counts const &mc) {
|
||||||
|
throw std::runtime_error("corrupt metadata, please use thin_check for details");
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void visit(block_address b, uint32_t count) {
|
||||||
|
if (last_visited_ && (b > *last_visited_ + 1))
|
||||||
|
emitter_.emit(*last_visited_ + 1, b);
|
||||||
|
|
||||||
|
last_visited_ = b;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
discard_emitter &emitter_;
|
||||||
|
boost::optional<block_address> last_visited_;
|
||||||
|
};
|
||||||
|
|
||||||
|
int trim(string const &metadata_dev, string const &data_dev) {
|
||||||
|
// We can trim any block that has zero count in the data
|
||||||
|
// space map.
|
||||||
|
metadata md(metadata_dev, 0);
|
||||||
|
|
||||||
|
if (!md.data_sm_->get_nr_free())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
discard_emitter de(data_dev, md.sb_.data_block_size_,
|
||||||
|
md.data_sm_->get_nr_blocks());
|
||||||
|
trim_visitor tv(de);
|
||||||
|
|
||||||
|
confirm_pool_is_not_active();
|
||||||
|
md.data_sm_->visit(tv);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void usage(ostream &out, string const &cmd) {
|
||||||
|
out << "Usage: " << cmd << " [options] {device|file}\n"
|
||||||
|
<< "Options:\n"
|
||||||
|
<< " {--pool-inactive}\n"
|
||||||
|
<< " {-h|--help}\n"
|
||||||
|
<< " {-V|--version}" << endl;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct flags {
|
||||||
|
boost::optional<string> metadata_dev;
|
||||||
|
boost::optional<string> data_dev;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
//----------------------------------------------------------------
|
||||||
|
|
||||||
|
int thin_trim_main(int argc, char **argv)
|
||||||
|
{
|
||||||
|
int c;
|
||||||
|
flags fs;
|
||||||
|
const char shortopts[] = "hV";
|
||||||
|
|
||||||
|
const struct option longopts[] = {
|
||||||
|
{ "help", no_argument, NULL, 'h' },
|
||||||
|
{ "version", no_argument, NULL, 'V' },
|
||||||
|
{ "metadata-dev", required_argument, NULL, 0 },
|
||||||
|
{ "data-dev", required_argument, NULL, 1 },
|
||||||
|
{ "pool-inactive", no_argument, NULL, 2 },
|
||||||
|
{ NULL, no_argument, NULL, 0 }
|
||||||
|
};
|
||||||
|
|
||||||
|
while ((c = getopt_long(argc, argv, shortopts, longopts, NULL)) != -1) {
|
||||||
|
switch(c) {
|
||||||
|
case 0:
|
||||||
|
fs.metadata_dev = optarg;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 1:
|
||||||
|
fs.data_dev = optarg;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'h':
|
||||||
|
usage(cout, basename(argv[0]));
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
case 'V':
|
||||||
|
cout << THIN_PROVISIONING_TOOLS_VERSION << endl;
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
default:
|
||||||
|
usage(cerr, basename(argv[0]));
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!fs.metadata_dev || !fs.data_dev) {
|
||||||
|
usage(cerr, basename(argv[0]));
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return trim(*fs.metadata_dev, *fs.data_dev);
|
||||||
|
}
|
||||||
|
|
||||||
|
//----------------------------------------------------------------
|
Loading…
Reference in New Issue
Block a user