b7d418131d
* [file_utils] spin-off syscall-related file operations 1. Eliminate the potential circular dependency between persistent-data/block.h and persistent-data/file_utils.h, if the former one wants to include the latter. 2. Avoid namespace pollution by removing the "using namespace std" declaration in block.tcc. 3. Correct the header hierarchy: base/xml_utils.h now no longer depends on the higher-level persistent-data/file_utils.h * [file_utils] support block files in get_file_length()
224 lines
4.7 KiB
C++
224 lines
4.7 KiB
C++
#include <iostream>
|
|
#include <getopt.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/stat.h>
|
|
#include <linux/fs.h>
|
|
#include <libgen.h>
|
|
#include <fcntl.h>
|
|
|
|
#undef BLOCK_SIZE
|
|
|
|
#ifndef BLKDISCARD
|
|
#define BLKDISCARD _IO(0x12,119)
|
|
#endif
|
|
|
|
#include "persistent-data/file_utils.h"
|
|
#include "thin-provisioning/commands.h"
|
|
#include "metadata.h"
|
|
#include "version.h"
|
|
|
|
using namespace persistent_data;
|
|
using namespace std;
|
|
using namespace thin_provisioning;
|
|
|
|
//----------------------------------------------------------------
|
|
|
|
namespace {
|
|
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];
|
|
|
|
cerr << "emitting discard for blocks (" << b << ", " << e << "]\n";
|
|
|
|
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_iterator : public space_map::iterator {
|
|
public:
|
|
trim_iterator(discard_emitter &e)
|
|
: emitter_(e) {
|
|
}
|
|
|
|
virtual void operator() (block_address b, ref_t count) {
|
|
highest_ = b;
|
|
|
|
if (count) {
|
|
if (last_referenced_ && (b > *last_referenced_ + 1))
|
|
emitter_.emit(*last_referenced_ + 1, b);
|
|
|
|
last_referenced_ = b;
|
|
}
|
|
}
|
|
|
|
void complete() {
|
|
if (last_referenced_) {
|
|
if (*last_referenced_ != *highest_)
|
|
emitter_.emit(*last_referenced_, *highest_ + 1ull);
|
|
|
|
} else if (highest_)
|
|
emitter_.emit(0ull, *highest_ + 1);
|
|
}
|
|
|
|
private:
|
|
discard_emitter &emitter_;
|
|
boost::optional<block_address> last_referenced_;
|
|
boost::optional<block_address> highest_;
|
|
};
|
|
|
|
int trim(string const &metadata_dev, string const &data_dev) {
|
|
cerr << "in trim\n";
|
|
|
|
// We can trim any block that has zero count in the data
|
|
// space map.
|
|
block_manager<>::ptr bm = open_bm(metadata_dev, block_manager<>::READ_ONLY);
|
|
metadata md(bm);
|
|
|
|
if (!md.data_sm_->get_nr_free()) {
|
|
cerr << "All data blocks allocated, nothing to discard\n";
|
|
return 0;
|
|
}
|
|
|
|
discard_emitter de(data_dev, md.sb_.data_block_size_,
|
|
md.data_sm_->get_nr_blocks());
|
|
trim_iterator ti(de);
|
|
|
|
md.data_sm_->iterate(ti);
|
|
ti.complete();
|
|
|
|
return 0;
|
|
}
|
|
|
|
struct flags {
|
|
boost::optional<string> metadata_dev;
|
|
boost::optional<string> data_dev;
|
|
};
|
|
}
|
|
|
|
//----------------------------------------------------------------
|
|
|
|
thin_trim_cmd::thin_trim_cmd()
|
|
: command("thin_trim")
|
|
{
|
|
}
|
|
|
|
void
|
|
thin_trim_cmd::usage(std::ostream &out) const
|
|
{
|
|
out << "Usage: " << get_name() << " [options] --metadata-dev {device|file} --data-dev {device|file}\n"
|
|
<< "Options:\n"
|
|
<< " {-h|--help}\n"
|
|
<< " {-V|--version}" << endl;
|
|
}
|
|
|
|
int
|
|
thin_trim_cmd::run(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 2:
|
|
cerr << "--pool-inactive no longer required since we ensure the metadata device is opened exclusively.\n";
|
|
break;
|
|
|
|
case 'h':
|
|
usage(cout);
|
|
return 0;
|
|
|
|
case 'V':
|
|
cout << THIN_PROVISIONING_TOOLS_VERSION << endl;
|
|
return 0;
|
|
|
|
default:
|
|
usage(cerr);
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
if (!fs.metadata_dev || !fs.data_dev) {
|
|
usage(cerr);
|
|
return 1;
|
|
}
|
|
|
|
return trim(*fs.metadata_dev, *fs.data_dev);
|
|
}
|
|
|
|
//----------------------------------------------------------------
|