diff --git a/.gitignore b/.gitignore
index 9af018d..0db515f 100644
--- a/.gitignore
+++ b/.gitignore
@@ -8,19 +8,9 @@
test.data
cachegrind.*
\#*\#
+core
-thin_check
-thin_dump
-thin_restore
-thin_repair
-thin_rmap
-thin_metadata_size
-
-cache_check
-cache_dump
-cache_restore
-cache_repair
-cache_metadata_size
+bin/pdata_tools
*.metadata
bad-metadata
@@ -43,3 +33,5 @@ config.cache
config.log
config.status
configure
+
+callgrind.*
\ No newline at end of file
diff --git a/CHANGES b/CHANGES
new file mode 100644
index 0000000..3de75c1
--- /dev/null
+++ b/CHANGES
@@ -0,0 +1,24 @@
+v0.5
+====
+
+- thin_delta, thin_trim
+
+v0.4
+====
+
+- All tools switch to using libaio. This gives a large performance
+ boost, especially to the write focused tools like thin_restore.
+
+- Added a progress monitor to thin_restore, cache_restore and era_restore
+
+- Added a --quiet/-q option to *_restore to turn off the progress bar
+
+- Removed variable hint size support from cache tools. The kernel
+ still only supports a fixed 32bit width. This will have a side
+ effect of reducing the executable sizes due to less template
+ instatiation.
+
+- Tools rolled into a single executable to save space.
+
+- Fixed some bugs when walking bitsets (possibly effecting cache_dump
+ and cache_check).
\ No newline at end of file
diff --git a/Gemfile.lock b/Gemfile.lock
index 069c068..d198286 100644
--- a/Gemfile.lock
+++ b/Gemfile.lock
@@ -1,30 +1,32 @@
GEM
remote: https://rubygems.org/
specs:
- aruba (0.5.3)
+ aruba (0.6.1)
childprocess (>= 0.3.6)
cucumber (>= 1.1.1)
rspec-expectations (>= 2.7.0)
builder (3.2.2)
- childprocess (0.3.9)
+ childprocess (0.5.3)
ffi (~> 1.0, >= 1.0.11)
- cucumber (1.3.8)
+ cucumber (1.3.16)
builder (>= 2.1.2)
diff-lcs (>= 1.1.3)
- gherkin (~> 2.12.1)
+ gherkin (~> 2.12)
multi_json (>= 1.7.5, < 2.0)
- multi_test (>= 0.0.2)
- diff-lcs (1.2.4)
- ejt_command_line (0.0.2)
- ffi (1.9.0)
+ multi_test (>= 0.1.1)
+ diff-lcs (1.2.5)
+ ejt_command_line (0.0.4)
+ ffi (1.9.3)
gherkin (2.12.2)
multi_json (~> 1.3)
- multi_json (1.8.2)
- multi_test (0.0.2)
- rspec-expectations (2.14.3)
- diff-lcs (>= 1.1.3, < 2.0)
- thinp_xml (0.0.12)
- ejt_command_line (= 0.0.2)
+ multi_json (1.10.1)
+ multi_test (0.1.1)
+ rspec-expectations (3.0.4)
+ diff-lcs (>= 1.2.0, < 2.0)
+ rspec-support (~> 3.0.0)
+ rspec-support (3.0.4)
+ thinp_xml (0.0.20)
+ ejt_command_line (>= 0.0.2)
PLATFORMS
ruby
diff --git a/Makefile.in b/Makefile.in
index 0a93f03..eaff125 100644
--- a/Makefile.in
+++ b/Makefile.in
@@ -16,53 +16,60 @@
# with thin-provisioning-tools. If not, see
# .
-.PHONY: all
-
V=@
-PROGRAMS=\
- cache_check \
- cache_dump \
- cache_restore \
- cache_repair \
- cache_metadata_size \
- \
- thin_check \
- thin_dump \
- thin_restore \
- thin_repair \
- thin_rmap \
- thin_metadata_size
+PROGRAMS=\
+ bin/pdata_tools
+
+.PHONY: all
all: $(PROGRAMS)
SOURCE=\
+ base/application.cc \
base/base64.cc \
+ base/endian_utils.cc \
base/error_state.cc \
- \
+ base/error_string.cc \
+ base/progress_monitor.cc \
+ base/xml_utils.cc \
+ block-cache/block_cache.cc \
+ caching/cache_check.cc \
+ caching/cache_dump.cc \
+ caching/cache_metadata_size.cc \
+ caching/cache_repair.cc \
+ caching/cache_restore.cc \
caching/hint_array.cc \
- caching/superblock.cc \
caching/mapping_array.cc \
caching/metadata.cc \
caching/metadata_dump.cc \
caching/restore_emitter.cc \
+ caching/superblock.cc \
caching/xml_format.cc \
- \
+ era/era_array.cc \
+ era/era_check.cc \
+ era/era_detail.cc \
+ era/era_dump.cc \
+ era/era_invalidate.cc \
+ era/era_restore.cc \
+ era/metadata.cc \
+ era/metadata_dump.cc \
+ era/restore_emitter.cc \
+ era/superblock.cc \
+ era/writeset_tree.cc \
+ era/xml_format.cc \
+ main.cc \
persistent-data/checksum.cc \
- persistent-data/endian_utils.cc \
+ persistent-data/data-structures/bitset.cc \
+ persistent-data/data-structures/bloom_filter.cc \
+ persistent-data/data-structures/btree.cc \
persistent-data/error_set.cc \
persistent-data/file_utils.cc \
persistent-data/hex_dump.cc \
- persistent-data/lock_tracker.cc \
- persistent-data/transaction_manager.cc \
- \
- persistent-data/data-structures/bitset.cc \
- persistent-data/data-structures/btree.cc \
- \
- persistent-data/space_map.cc \
+ persistent-data/space-maps/careful_alloc.cc \
persistent-data/space-maps/disk.cc \
persistent-data/space-maps/recursive.cc \
- persistent-data/space-maps/careful_alloc.cc \
- \
+ persistent-data/space_map.cc \
+ persistent-data/transaction_manager.cc \
thin-provisioning/device_tree.cc \
thin-provisioning/human_readable_format.cc \
thin-provisioning/mapping_tree.cc \
@@ -72,23 +79,16 @@ SOURCE=\
thin-provisioning/restore_emitter.cc \
thin-provisioning/rmap_visitor.cc \
thin-provisioning/superblock.cc \
- thin-provisioning/thin_pool.cc \
- thin-provisioning/xml_format.cc
-
-PDATA_OBJECTS=$(subst .cc,.o,$(SOURCE))
-
-CXX_PROGRAM_SOURCE=\
- caching/cache_check.cc \
- caching/cache_restore.cc \
- \
thin-provisioning/thin_check.cc \
+ thin-provisioning/thin_delta.cc \
thin-provisioning/thin_dump.cc \
- thin-provisioning/thin_restore.cc \
+ thin-provisioning/thin_metadata_size.cc \
+ thin-provisioning/thin_pool.cc \
thin-provisioning/thin_repair.cc \
- thin-provisioning/thin_rmap.cc
-
-C_PROGRAM_SOURCE=\
- thin-provisioning/thin_metadata_size.c
+ thin-provisioning/thin_restore.cc \
+ thin-provisioning/thin_rmap.cc \
+ thin-provisioning/thin_trim.cc \
+ thin-provisioning/xml_format.cc
CC:=@CC@
CXX:=@CXX@
@@ -99,18 +99,19 @@ CFLAGS+=-g -Wall -O3
CXXFLAGS+=-g -Wall -fno-strict-aliasing
CXXFLAGS+=@CXXOPTIMISE_FLAG@
CXXFLAGS+=@CXXDEBUG_FLAG@
+CXXFLAGS+=@CXX_STRERROR_FLAG@
INCLUDES+=-I$(TOP_BUILDDIR) -I$(TOP_DIR) -I$(TOP_DIR)/thin-provisioning
-LIBS:=-lstdc++
-LIBEXPAT:=-lexpat
+LIBS:=-lstdc++ -laio -lexpat
INSTALL:=@INSTALL@
PREFIX:=@prefix@
BINDIR:=$(DESTDIR)$(PREFIX)/sbin
-MANPATH:=$(DESTDIR)$(MANDIR)
+DATADIR:=$(DESTDIR)$(PREFIX)/share
+MANPATH:=$(DATADIR)/man
vpath %.cc $(TOP_DIR)
INSTALL_DIR = $(INSTALL) -m 755 -d
-INSTALL_PROGRAM = $(INSTALL) -m 755
+INSTALL_PROGRAM = $(INSTALL) -m 755 -s
INSTALL_DATA = $(INSTALL) -p -m 644
ifeq ("@TESTING@", "yes")
@@ -123,14 +124,6 @@ endif
.SUFFIXES: .d
-%.o: %.c
- @echo " [CC] $<"
- $(V) $(CC) -c $(INCLUDES) $(CFLAGS) -o $@ $<
- @echo " [DEP] $<"
- $(V) $(CC) -MM -MT $(subst .c,.o,$<) $(INCLUDES) $(CFLAGS) $< > $*.$$$$; \
- sed 's,\([^ :]*\)\.o[ :]*,\1.o \1.gmo $* : Makefile ,g' < $*.$$$$ > $*.d; \
- $(RM) $*.$$$$
-
%.o: %.cc
@echo " [CXX] $<"
$(V) $(CXX) -c $(INCLUDES) $(CXXFLAGS) -o $@ $<
@@ -141,149 +134,15 @@ endif
#----------------------------------------------------------------
-lib/libpdata.a: $(PDATA_OBJECTS)
+lib/libpdata.a: $(OBJECTS)
@echo " [AR] $<"
- $(V)ar -rv $@ $(PDATA_OBJECTS) > /dev/null 2>&1
+ $(V)ar -rv $@ $(OBJECTS) > /dev/null 2>&1
+
+bin/pdata_tools: $(OBJECTS)
+ @echo " [LD] $@"
+ $(V) $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $+ $(LIBS)
#----------------------------------------------------------------
-# Thin provisioning tools
-
-THIN_DEBUG_SOURCE=$(SOURCE)
-THIN_DUMP_SOURCE=$(SOURCE)
-THIN_REPAIR_SOURCE=$(SOURCE)
-THIN_RESTORE_SOURCE=$(SOURCE)
-THIN_CHECK_SOURCE=\
- base/error_state.cc \
- persistent-data/checksum.cc \
- persistent-data/endian_utils.cc \
- persistent-data/error_set.cc \
- persistent-data/file_utils.cc \
- persistent-data/hex_dump.cc \
- persistent-data/lock_tracker.cc \
- persistent-data/data-structures/btree.cc \
- persistent-data/space_map.cc \
- persistent-data/space-maps/disk.cc \
- persistent-data/space-maps/recursive.cc \
- persistent-data/space-maps/careful_alloc.cc \
- persistent-data/transaction_manager.cc \
- thin-provisioning/device_tree.cc \
- thin-provisioning/mapping_tree.cc \
- thin-provisioning/metadata.cc \
- thin-provisioning/metadata_checker.cc \
- thin-provisioning/superblock.cc
-
-THIN_RMAP_SOURCE=\
- persistent-data/checksum.cc \
- persistent-data/endian_utils.cc \
- persistent-data/error_set.cc \
- persistent-data/file_utils.cc \
- persistent-data/hex_dump.cc \
- persistent-data/lock_tracker.cc \
- persistent-data/data-structures/btree.cc \
- persistent-data/space_map.cc \
- persistent-data/space-maps/disk.cc \
- persistent-data/space-maps/recursive.cc \
- persistent-data/space-maps/careful_alloc.cc \
- persistent-data/transaction_manager.cc \
- thin-provisioning/device_tree.cc \
- thin-provisioning/mapping_tree.cc \
- thin-provisioning/metadata.cc \
- thin-provisioning/metadata_checker.cc \
- thin-provisioning/rmap_visitor.cc \
- thin-provisioning/superblock.cc
-
-THIN_DEBUG_OBJECTS=$(subst .cc,.o,$(THIN_DEBUG_SOURCE))
-THIN_DUMP_OBJECTS=$(subst .cc,.o,$(THIN_DUMP_SOURCE))
-THIN_REPAIR_OBJECTS=$(subst .cc,.o,$(THIN_REPAIR_SOURCE))
-THIN_RESTORE_OBJECTS=$(subst .cc,.o,$(THIN_RESTORE_SOURCE))
-THIN_CHECK_OBJECTS=$(subst .cc,.o,$(THIN_CHECK_SOURCE))
-THIN_RMAP_OBJECTS=$(subst .cc,.o,$(THIN_RMAP_SOURCE))
-
-thin_debug: $(THIN_DEBUG_OBJECTS) thin-provisioning/thin_debug.o
- @echo " [LD] $@"
- $(V) $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $+ $(LIBS) $(LIBEXPAT)
-
-thin_repair: $(THIN_REPAIR_OBJECTS) thin-provisioning/thin_repair.o
- @echo " [LD] $@"
- $(V) $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $+ $(LIBS) $(LIBEXPAT)
-
-thin_dump: $(THIN_DUMP_OBJECTS) thin-provisioning/thin_dump.o
- @echo " [LD] $@"
- $(V) $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $+ $(LIBS) $(LIBEXPAT)
-
-thin_restore: $(THIN_RESTORE_OBJECTS) thin-provisioning/thin_restore.o
- @echo " [LD] $@"
- $(V) $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $+ $(LIBS) $(LIBEXPAT)
-
-thin_check: $(THIN_CHECK_OBJECTS) thin-provisioning/thin_check.o
- @echo " [LD] $@"
- $(V) $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $+ $(LIBS)
-
-thin_rmap: $(THIN_RMAP_OBJECTS) thin-provisioning/thin_rmap.o
- @echo " [LD] $@"
- $(V) $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $+ $(LIBS)
-
-thin_metadata_size: thin-provisioning/thin_metadata_size.o
- @echo " [LD] $@"
- $(V) $(CC) $(CFLAGS) $(LDFLAGS) -o $@ $+ -lm
-
-#----------------------------------------------------------------
-# Cache tools
-
-CACHE_CHECK_SOURCE=\
- base/base64.cc \
- base/error_state.cc \
- persistent-data/checksum.cc \
- persistent-data/endian_utils.cc \
- persistent-data/error_set.cc \
- persistent-data/file_utils.cc \
- persistent-data/hex_dump.cc \
- persistent-data/lock_tracker.cc \
- persistent-data/data-structures/btree.cc \
- persistent-data/data-structures/bitset.cc \
- persistent-data/space_map.cc \
- persistent-data/space-maps/disk.cc \
- persistent-data/space-maps/recursive.cc \
- persistent-data/space-maps/careful_alloc.cc \
- persistent-data/transaction_manager.cc \
- caching/hint_array.cc \
- caching/superblock.cc \
- caching/mapping_array.cc \
- caching/metadata.cc \
- caching/metadata_dump.cc \
- caching/restore_emitter.cc \
- caching/xml_format.cc
-
-CACHE_CHECK_OBJECTS=$(subst .cc,.o,$(CACHE_CHECK_SOURCE))
-
-CACHE_DUMP_SOURCE=$(SOURCE)
-CACHE_DUMP_OBJECTS=$(subst .cc,.o,$(CACHE_DUMP_SOURCE))
-
-CACHE_REPAIR_SOURCE=$(SOURCE)
-CACHE_REPAIR_OBJECTS=$(subst .cc,.o,$(CACHE_REPAIR_SOURCE))
-
-CACHE_RESTORE_SOURCE=$(SOURCE)
-CACHE_RESTORE_OBJECTS=$(subst .cc,.o,$(CACHE_RESTORE_SOURCE))
-
-cache_check: $(CACHE_CHECK_OBJECTS) caching/cache_check.o
- @echo " [LD] $@"
- $(V) $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $+ $(LIBS) $(LIBEXPAT)
-
-cache_dump: $(CACHE_DUMP_OBJECTS) caching/cache_dump.o
- @echo " [LD] $@"
- $(V) $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $+ $(LIBS) $(LIBEXPAT)
-
-cache_repair: $(CACHE_REPAIR_OBJECTS) caching/cache_repair.o
- @echo " [LD] $@"
- $(V) $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $+ $(LIBS) $(LIBEXPAT)
-
-cache_restore: $(CACHE_RESTORE_OBJECTS) caching/cache_restore.o
- @echo " [LD] $@"
- $(V) $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $+ $(LIBS) $(LIBEXPAT)
-
-cache_metadata_size: caching/cache_metadata_size.o
- @echo " [LD] $@"
- $(V) $(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $+ $(LIBS)
DEPEND_FILES=\
$(subst .cc,.d,$(SOURCE)) \
@@ -302,29 +161,44 @@ clean:
distclean: clean
$(RM) config.cache config.log config.status configure.h version.h Makefile unit-tests/Makefile
-install: $(PROGRAMS)
+install: bin/pdata_tools
$(INSTALL_DIR) $(BINDIR)
- $(INSTALL_PROGRAM) cache_check $(BINDIR)
- $(INSTALL_PROGRAM) cache_dump $(BINDIR)
- $(INSTALL_PROGRAM) cache_repair $(BINDIR)
- $(INSTALL_PROGRAM) cache_restore $(BINDIR)
- $(INSTALL_PROGRAM) thin_check $(BINDIR)
- $(INSTALL_PROGRAM) thin_dump $(BINDIR)
- $(INSTALL_PROGRAM) thin_repair $(BINDIR)
- $(INSTALL_PROGRAM) thin_restore $(BINDIR)
- $(INSTALL_PROGRAM) thin_rmap $(BINDIR)
- $(INSTALL_PROGRAM) thin_metadata_size $(BINDIR)
+ $(INSTALL_PROGRAM) bin/pdata_tools $(BINDIR)
+ ln -s -f pdata_tools $(BINDIR)/cache_check
+ ln -s -f pdata_tools $(BINDIR)/cache_dump
+ ln -s -f pdata_tools $(BINDIR)/cache_metadata_size
+ ln -s -f pdata_tools $(BINDIR)/cache_repair
+ ln -s -f pdata_tools $(BINDIR)/cache_restore
+ ln -s -f pdata_tools $(BINDIR)/thin_check
+ ln -s -f pdata_tools $(BINDIR)/thin_delta
+ ln -s -f pdata_tools $(BINDIR)/thin_dump
+ ln -s -f pdata_tools $(BINDIR)/thin_repair
+ ln -s -f pdata_tools $(BINDIR)/thin_restore
+ 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)/era_check
+ ln -s -f pdata_tools $(BINDIR)/era_dump
+ ln -s -f pdata_tools $(BINDIR)/era_invalidate
+ ln -s -f pdata_tools $(BINDIR)/era_restore
$(INSTALL_DIR) $(MANPATH)/man8
$(INSTALL_DATA) man8/cache_check.8 $(MANPATH)/man8
$(INSTALL_DATA) man8/cache_dump.8 $(MANPATH)/man8
$(INSTALL_DATA) man8/cache_repair.8 $(MANPATH)/man8
$(INSTALL_DATA) man8/cache_restore.8 $(MANPATH)/man8
$(INSTALL_DATA) man8/thin_check.8 $(MANPATH)/man8
+ $(INSTALL_DATA) man8/thin_delta.8 $(MANPATH)/man8
$(INSTALL_DATA) man8/thin_dump.8 $(MANPATH)/man8
$(INSTALL_DATA) man8/thin_repair.8 $(MANPATH)/man8
$(INSTALL_DATA) man8/thin_restore.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/era_check.8 $(MANPATH)/man8
+ $(INSTALL_DATA) man8/era_dump.8 $(MANPATH)/man8
+ $(INSTALL_DATA) man8/era_invalidate.8 $(MANPATH)/man8
+
+# $(INSTALL_DATA) man8/era_restore.8 $(MANPATH)/man8
.PHONY: install
@@ -333,7 +207,7 @@ include unit-tests/Makefile
.PHONEY: features
-features: $(PROGRAMS)
+features: pdata_tools
cucumber --no-color --format progress
test: features unit-test
diff --git a/README.md b/README.md
index 66c7199..d23dae5 100644
--- a/README.md
+++ b/README.md
@@ -10,6 +10,7 @@ Requirements
A C++ compiler that supports the c++11 standard (eg, g++).
The [Boost C++ library](http://www.boost.org/).
The [expat](http://expat.sourceforge.net/) xml parser library (version 1).
+The libaio library (note this is not the same as the aio library that you get by linking -lrt)
make, autoconf etc.
There are more requirements for testing, detailed below.
diff --git a/VERSION b/VERSION
index a45be46..267577d 100644
--- a/VERSION
+++ b/VERSION
@@ -1 +1 @@
-0.2.8
+0.4.1
diff --git a/base/application.cc b/base/application.cc
new file mode 100644
index 0000000..9e1f0dd
--- /dev/null
+++ b/base/application.cc
@@ -0,0 +1,63 @@
+#include "base/application.h"
+
+#include
+#include
+#include
+
+using namespace base;
+using namespace std;
+
+//----------------------------------------------------------------
+
+int
+application::run(int argc, char **argv)
+{
+ string cmd = get_basename(argv[0]);
+
+ if (cmd == string("pdata_tools")) {
+ argc--;
+ argv++;
+
+ if (!argc) {
+ usage();
+ return 1;
+ }
+
+ cmd = argv[0];
+ }
+
+ std::list::const_iterator it;
+ for (it = cmds_.begin(); it != cmds_.end(); ++it) {
+ if (cmd == (*it)->get_name())
+ return (*it)->run(argc, argv);
+ }
+
+ std::cerr << "Unknown command '" << cmd << "'\n";
+ usage();
+ return 1;
+}
+
+void
+application::usage()
+{
+ std::cerr << "Usage: \n"
+ << "commands:\n";
+
+ std::list::const_iterator it;
+ for (it = cmds_.begin(); it != cmds_.end(); ++it) {
+ std::cerr << " " << (*it)->get_name() << "\n";
+ }
+}
+
+std::string
+application::get_basename(std::string const &path) const
+{
+ char buffer[PATH_MAX + 1];
+
+ memset(buffer, 0, sizeof(buffer));
+ strncpy(buffer, path.c_str(), PATH_MAX);
+
+ return ::basename(buffer);
+}
+
+//----------------------------------------------------------------
diff --git a/base/application.h b/base/application.h
new file mode 100644
index 0000000..d01eb36
--- /dev/null
+++ b/base/application.h
@@ -0,0 +1,52 @@
+#ifndef BASE_APPLICATION_H
+#define BASE_APPLICATION_H
+
+#include
+#include
+#include
+#include
+
+//----------------------------------------------------------------
+
+namespace base {
+ class command {
+ public:
+ typedef int (*cmd_fn)(int, char **);
+
+ command(std::string const &name, cmd_fn fn)
+ : name_(name),
+ fn_(fn) {
+ }
+
+ std::string const &get_name() const {
+ return name_;
+ }
+
+ int run(int argc, char **argv) const {
+ return fn_(argc, argv);
+ }
+
+ private:
+ std::string name_;
+ cmd_fn fn_;
+ };
+
+ class application {
+ public:
+ void add_cmd(command const &c) {
+ cmds_.push_back(&c);
+ }
+
+ int run(int argc, char **argv);
+
+ private:
+ void usage();
+ std::string get_basename(std::string const &path) const;
+
+ std::list cmds_;
+ };
+}
+
+//----------------------------------------------------------------
+
+#endif
diff --git a/persistent-data/endian_utils.cc b/base/endian_utils.cc
similarity index 100%
rename from persistent-data/endian_utils.cc
rename to base/endian_utils.cc
diff --git a/persistent-data/endian_utils.h b/base/endian_utils.h
similarity index 80%
rename from persistent-data/endian_utils.h
rename to base/endian_utils.h
index 39276e6..82b5b59 100644
--- a/persistent-data/endian_utils.h
+++ b/base/endian_utils.h
@@ -16,8 +16,8 @@
// with thin-provisioning-tools. If not, see
// .
-#ifndef ENDIAN_H
-#define ENDIAN_H
+#ifndef BASE_ENDIAN_H
+#define BASE_ENDIAN_H
#include
#include
@@ -25,7 +25,26 @@
//----------------------------------------------------------------
-// FIXME: rename to endian
+/* An old glic doesn't provide these macros */
+#if !defined(htole16) || !defined(le16toh) || !defined(htole32) || !defined(le32toh) || !defined(htole64) || !defined(le64toh)
+#include
+#if __BYTE_ORDER == __LITTLE_ENDIAN
+#define htole16(x) (x)
+#define le16toh(x) (x)
+#define htole32(x) (x)
+#define le32toh(x) (x)
+#define htole64(x) (x)
+#define le64toh(x) (x)
+#else
+#define htole16(x) __bswap_16(x)
+#define le16toh(x) __bswap_16(x)
+#define htole32(x) __bswap_32(x)
+#define le32toh(x) __bswap_32(x)
+#define htole64(x) __bswap_64(x)
+#define le64toh(x) __bswap_64(x)
+#endif
+#endif
+
namespace base {
// These are just little wrapper types to make the compiler
diff --git a/base/error_string.cc b/base/error_string.cc
new file mode 100644
index 0000000..6cb02c4
--- /dev/null
+++ b/base/error_string.cc
@@ -0,0 +1,39 @@
+#include "base/error_string.h"
+
+#include
+#include
+
+#include
+
+using namespace std;
+
+//----------------------------------------------------------------
+
+#ifdef STRERROR_R_CHAR_P
+
+string base::error_string(int err)
+{
+ char *ptr;
+ char buffer[128];
+
+ ptr = strerror_r(errno, buffer, sizeof(buffer));
+ return string(ptr);
+}
+
+#else
+
+string base::error_string(int err)
+{
+ int r;
+ char buffer[128];
+
+ r = strerror_r(errno, buffer, sizeof(buffer));
+ if (r)
+ throw runtime_error("strerror_r failed");
+
+ return string(buffer);
+}
+
+#endif
+
+//----------------------------------------------------------------
diff --git a/base/error_string.h b/base/error_string.h
new file mode 100644
index 0000000..dd7549a
--- /dev/null
+++ b/base/error_string.h
@@ -0,0 +1,16 @@
+#ifndef BASE_ERROR_STRING_H
+#define BASE_ERROR_STRING_H
+
+#include
+
+//----------------------------------------------------------------
+
+namespace base {
+ // There are a couple of version of strerror_r kicking around, so
+ // we wrap it.
+ std::string error_string(int err);
+}
+
+//----------------------------------------------------------------
+
+#endif
diff --git a/base/indented_stream.h b/base/indented_stream.h
new file mode 100644
index 0000000..b47bca2
--- /dev/null
+++ b/base/indented_stream.h
@@ -0,0 +1,48 @@
+#ifndef BASE_INDENTED_STREAM_H
+#define BASE_INDENTED_STREAM_H
+
+#include
+
+//----------------------------------------------------------------
+
+namespace {
+ class indented_stream {
+ public:
+ indented_stream(std::ostream &out)
+ : out_(out),
+ indent_(0) {
+ }
+
+ void indent() {
+ for (unsigned i = 0; i < indent_ * 2; i++)
+ out_ << ' ';
+ }
+
+ void inc() {
+ indent_++;
+ }
+
+ void dec() {
+ indent_--;
+ }
+
+ template
+ indented_stream &operator <<(T const &t) {
+ out_ << t;
+ return *this;
+ }
+
+ indented_stream &operator <<(std::ostream &(*fp)(std::ostream &)) {
+ out_ << fp;
+ return *this;
+ }
+
+ private:
+ std::ostream &out_;
+ unsigned indent_;
+ };
+}
+
+//----------------------------------------------------------------
+
+#endif
diff --git a/base/progress_monitor.cc b/base/progress_monitor.cc
new file mode 100644
index 0000000..1d88302
--- /dev/null
+++ b/base/progress_monitor.cc
@@ -0,0 +1,78 @@
+#include "base/progress_monitor.h"
+
+#include
+
+//----------------------------------------------------------------
+
+namespace {
+ using namespace std;
+
+ class progress_bar : public base::progress_monitor {
+ public:
+ progress_bar(string const &title)
+ : title_(title),
+ progress_width_(50),
+ spinner_(0) {
+
+ update_percent(0);
+ }
+ ~progress_bar() {
+ cout << "\n";
+ }
+
+ void update_percent(unsigned p) {
+ unsigned nr_equals = max(progress_width_ * p / 100, 1);
+ unsigned nr_spaces = progress_width_ - nr_equals;
+
+ cout << title_ << ": [";
+
+ for (unsigned i = 0; i < nr_equals - 1; i++)
+ cout << '=';
+
+ if (nr_equals < progress_width_)
+ cout << '>';
+
+ for (unsigned i = 0; i < nr_spaces; i++)
+ cout << ' ';
+
+ cout << "] " << spinner_char() << " " << p << "%\r" << flush;
+
+ spinner_++;
+ }
+
+ private:
+ char spinner_char() const {
+ char cs[] = {'|', '/', '-', '\\'};
+
+ unsigned index = spinner_ % sizeof(cs);
+ return cs[index];
+ }
+
+ std::string title_;
+ unsigned progress_width_;
+ unsigned spinner_;
+ };
+
+ class quiet_progress : public base::progress_monitor {
+ public:
+ void update_percent(unsigned p) {
+ }
+ };
+
+}
+
+//----------------------------------------------------------------
+
+std::auto_ptr
+base::create_progress_bar(std::string const &title)
+{
+ return auto_ptr(new progress_bar(title));
+}
+
+std::auto_ptr
+base::create_quiet_progress_monitor()
+{
+ return auto_ptr(new quiet_progress());
+}
+
+//----------------------------------------------------------------
diff --git a/base/progress_monitor.h b/base/progress_monitor.h
new file mode 100644
index 0000000..5472343
--- /dev/null
+++ b/base/progress_monitor.h
@@ -0,0 +1,24 @@
+#ifndef BASE_PROGRESS_MONITOR_H
+#define BASE_PROGRESS_MONITOR_H
+
+#include
+#include
+#include
+
+//----------------------------------------------------------------
+
+namespace base {
+ class progress_monitor {
+ public:
+ virtual ~progress_monitor() {}
+
+ virtual void update_percent(unsigned) = 0;
+ };
+
+ std::auto_ptr create_progress_bar(std::string const &title);
+ std::auto_ptr create_quiet_progress_monitor();
+}
+
+//----------------------------------------------------------------
+
+#endif
diff --git a/base/xml_utils.cc b/base/xml_utils.cc
new file mode 100644
index 0000000..fb34153
--- /dev/null
+++ b/base/xml_utils.cc
@@ -0,0 +1,86 @@
+#include "xml_utils.h"
+
+#include "persistent-data/file_utils.h"
+#include
+#include
+
+using namespace xml_utils;
+
+//----------------------------------------------------------------
+
+void
+xml_parser::parse(std::string const &backup_file, bool quiet)
+{
+ persistent_data::check_file_exists(backup_file);
+ ifstream in(backup_file.c_str(), ifstream::in);
+
+ std::auto_ptr monitor = create_monitor(quiet);
+
+ size_t total = 0;
+ size_t input_length = get_file_length(backup_file);
+
+ while (!in.eof()) {
+ char buffer[4096];
+ in.read(buffer, sizeof(buffer));
+ size_t len = in.gcount();
+ int done = in.eof();
+
+ if (!XML_Parse(parser_, buffer, len, done)) {
+ ostringstream out;
+ out << "Parse error at line "
+ << XML_GetCurrentLineNumber(parser_)
+ << ":\n"
+ << XML_ErrorString(XML_GetErrorCode(parser_))
+ << endl;
+ throw runtime_error(out.str());
+ }
+
+ total += len;
+ monitor->update_percent(total * 100 / input_length);
+ }
+}
+
+size_t
+xml_parser::get_file_length(string const &file) const
+{
+ struct stat info;
+ int r;
+
+ r = ::stat(file.c_str(), &info);
+ if (r)
+ throw runtime_error("Couldn't stat backup path");
+
+ return info.st_size;
+}
+
+auto_ptr
+xml_parser::create_monitor(bool quiet)
+{
+ if (!quiet && isatty(fileno(stdout)))
+ return base::create_progress_bar("Restoring");
+ else
+ return base::create_quiet_progress_monitor();
+}
+
+//----------------------------------------------------------------
+
+void
+xml_utils::build_attributes(attributes &a, char const **attr)
+{
+ while (*attr) {
+ char const *key = *attr;
+
+ attr++;
+ if (!*attr) {
+ ostringstream out;
+ out << "No value given for xml attribute: " << key;
+ throw runtime_error(out.str());
+ }
+
+ char const *value = *attr;
+ a.insert(make_pair(string(key), string(value)));
+ attr++;
+ }
+}
+
+//----------------------------------------------------------------
diff --git a/base/xml_utils.h b/base/xml_utils.h
new file mode 100644
index 0000000..f867f56
--- /dev/null
+++ b/base/xml_utils.h
@@ -0,0 +1,74 @@
+#ifndef BASE_XML_UTILS_H
+#define BASE_XML_UTILS_H
+
+#include
+#include
+#include
+#include
+#include
+#include