168 lines
4.2 KiB
C++
Raw Normal View History

2011-12-15 19:34:31 +01:00
// Copyright (C) 2011 Red Hat, Inc. All rights reserved.
2011-12-06 13:53:05 +00:00
//
2011-12-06 13:43:56 +00:00
// This file is part of the thin-provisioning-tools source.
//
// thin-provisioning-tools is free software: you can redistribute it
// and/or modify it under the terms of the GNU General Public License
// as published by the Free Software Foundation, either version 3 of
// the License, or (at your option) any later version.
//
// thin-provisioning-tools is distributed in the hope that it will be
// useful, but WITHOUT ANY WARRANTY; without even the implied warranty
// of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License along
// with thin-provisioning-tools. If not, see
// <http://www.gnu.org/licenses/>.
2013-06-28 12:17:43 +01:00
#include <fstream>
2011-08-23 11:55:37 +01:00
#include <iostream>
#include <getopt.h>
#include <libgen.h>
2011-08-23 11:55:37 +01:00
2011-10-10 15:59:37 +01:00
#include "human_readable_format.h"
2011-10-28 12:13:03 +01:00
#include "metadata_dumper.h"
2011-10-28 12:25:06 +01:00
#include "metadata.h"
2011-10-10 15:59:37 +01:00
#include "xml_format.h"
#include "version.h"
2011-10-10 15:59:37 +01:00
2011-08-23 11:55:37 +01:00
using namespace persistent_data;
using namespace std;
using namespace thin_provisioning;
struct flags {
bool find_metadata_snap;
bool repair;
};
2011-08-23 11:55:37 +01:00
namespace {
2013-09-11 11:40:46 +01:00
int dump_(string const &path, ostream &out, string const &format, struct flags &flags,
block_address metadata_snap) {
2012-03-02 10:00:31 +00:00
try {
metadata::ptr md(new metadata(path, metadata_snap));
2012-03-02 10:00:31 +00:00
emitter::ptr e;
if (flags.find_metadata_snap) {
uint64_t metadata_snap_root = md->sb_.metadata_snap_; /* FIXME: use thin_pool method? */
if (metadata_snap_root) {
md.reset();
md = metadata::ptr(new metadata(path, metadata_snap_root));
} else {
cerr << "no metadata snapshot found!" << endl;
exit(1);
}
}
2012-03-02 10:00:31 +00:00
if (format == "xml")
2013-09-11 11:40:46 +01:00
e = create_xml_emitter(out);
2012-03-02 10:00:31 +00:00
else if (format == "human_readable")
2013-09-11 11:40:46 +01:00
e = create_human_readable_emitter(out);
2012-03-02 10:00:31 +00:00
else {
cerr << "unknown format '" << format << "'" << endl;
exit(1);
}
metadata_dump(md, e, flags.repair);
2012-05-17 12:28:23 +01:00
2012-03-02 10:00:31 +00:00
} catch (std::exception &e) {
2012-03-05 19:34:05 +01:00
cerr << e.what() << endl;
2012-03-02 10:00:31 +00:00
return 1;
2011-10-10 15:59:37 +01:00
}
2012-03-02 10:00:31 +00:00
return 0;
2011-08-23 11:55:37 +01:00
}
2013-09-11 11:40:46 +01:00
int dump(string const &path, char const *output, string const &format, struct flags &flags,
block_address metadata_snap = 0) {
if (output) {
ofstream out(output);
return dump_(path, out, format, flags, metadata_snap);
} else
return dump_(path, cout, format, flags, metadata_snap);
}
2012-03-13 14:06:10 +00:00
void usage(ostream &out, string const &cmd) {
out << "Usage: " << cmd << " [options] {device|file}" << endl
<< "Options:" << endl
<< " {-h|--help}" << endl
<< " {-f|--format} {xml|human_readable}" << endl
<< " {-r|--repair}" << endl
<< " {-m|--metadata-snap} [block#]" << endl
2013-06-28 12:17:43 +01:00
<< " {-o <xml file>}" << endl
2012-03-13 14:06:10 +00:00
<< " {-V|--version}" << endl;
2011-08-23 11:55:37 +01:00
}
}
int main(int argc, char **argv)
{
int c;
2013-06-28 12:17:43 +01:00
char const *output = NULL;
const char shortopts[] = "hm::o:f:rV";
char *end_ptr;
2012-03-05 19:34:05 +01:00
string format = "xml";
block_address metadata_snap = 0;
struct flags flags;
flags.find_metadata_snap = flags.repair = false;
2012-05-17 12:28:23 +01:00
const struct option longopts[] = {
{ "help", no_argument, NULL, 'h'},
{ "metadata-snap", optional_argument, NULL, 'm' },
2013-06-28 12:17:43 +01:00
{ "output", required_argument, NULL, 'o'},
{ "format", required_argument, NULL, 'f' },
2012-03-02 10:00:31 +00:00
{ "repair", no_argument, NULL, 'r'},
{ "version", no_argument, NULL, 'V'},
{ NULL, no_argument, NULL, 0 }
};
while ((c = getopt_long(argc, argv, shortopts, longopts, NULL)) != -1) {
switch(c) {
2012-03-02 10:00:31 +00:00
case 'h':
2012-03-13 14:06:10 +00:00
usage(cout, basename(argv[0]));
2012-03-02 10:00:31 +00:00
return 0;
case 'f':
format = optarg;
break;
case 'r':
flags.repair = true;
break;
case 'm':
if (optarg) {
metadata_snap = strtoull(optarg, &end_ptr, 10);
if (end_ptr == optarg) {
cerr << "couldn't parse <metadata_snap>" << endl;
usage(cerr, basename(argv[0]));
return 1;
}
} else
flags.find_metadata_snap = true;
2012-05-17 12:28:23 +01:00
break;
2013-06-28 12:17:43 +01:00
case 'o':
output = optarg;
break;
2012-03-02 10:00:31 +00:00
case 'V':
cout << THIN_PROVISIONING_TOOLS_VERSION << endl;
2012-03-02 10:00:31 +00:00
return 0;
2012-03-05 19:34:05 +01:00
default:
2012-03-13 14:06:10 +00:00
usage(cerr, basename(argv[0]));
2012-03-05 19:34:05 +01:00
return 1;
}
2011-10-10 15:59:37 +01:00
}
2012-03-05 19:34:05 +01:00
if (argc == optind) {
2012-03-13 14:06:10 +00:00
cerr << "No input file provided." << endl;
usage(cerr, basename(argv[0]));
2011-12-14 17:41:43 +01:00
return 1;
}
2013-09-11 11:40:46 +01:00
return dump(argv[optind], output, format, flags, metadata_snap);
2011-08-23 11:55:37 +01:00
}