summaryrefslogtreecommitdiffstats
path: root/resize/utils.ml
diff options
context:
space:
mode:
authorRichard W.M. Jones <rjones@redhat.com>2011-04-08 14:07:26 +0100
committerRichard W.M. Jones <rjones@redhat.com>2011-04-09 14:28:22 +0100
commitca03635a4c83afbe9b51fe846a8b3d5361462a90 (patch)
treeb7faa804b620854f9b8ef982f91238221819c4bb /resize/utils.ml
parent3a84e0784e1e3ab7b56850d0f8c9aa42f1ae3da1 (diff)
downloadlibguestfs-ca03635a4c83afbe9b51fe846a8b3d5361462a90.tar.gz
libguestfs-ca03635a4c83afbe9b51fe846a8b3d5361462a90.tar.xz
libguestfs-ca03635a4c83afbe9b51fe846a8b3d5361462a90.zip
Rewrite virt-resize in OCaml.
This is a fairly straightforward translation of Perl virt-resize into OCaml. It is bug-for-bug and feature-for-feature identical to the Perl version, except as noted below. The motivation is to have a more solid, high-level, statically safe compiled language to go forwards with fixing some of the harder bugs in virt-resize. In particular contracts between different parts of the program are now handled by statically typed structures checked at compile time, instead of the very ad-hoc unchecked hash tables used by the Perl version. OCaml and the ocaml-pcre library (Perl-Compatible Regular Expressions bindings for OCaml) are required. Extra features in this version: - 32 bit hosts are now supported. - We try hard to handle the case where the target disk is not "clean" (ie. all zeroes). It usually works for this case, whereas the previous version would usually fail. However it is still recommended that the system administrator creates a fresh blank disk for the target before running the program. - User messages are a bit more verbose and helpful. You can turn these off with the -q (--quiet) option. There is one lost feature: - Ability to specify >= T (terabytes) sizes in command line size expressions has been removed. This probably didn't work in the Perl version. Other differences: - The first partition on the target is no longer aligned; instead we place it at the same sector as on the source. I suspect that aligning it was causing the bootloader failures. - Because it's easier, we do more sanity checking on the source disk. This might lead to more failures, but they'd be failures you'd want to know about. - The order in which operations are performed has been changed to make it more logical. The user should not notice any functional difference, but debug messages will be quite a bit different. - virt-resize is a compiled binary, not a script.
Diffstat (limited to 'resize/utils.ml')
-rw-r--r--resize/utils.ml154
1 files changed, 154 insertions, 0 deletions
diff --git a/resize/utils.ml b/resize/utils.ml
new file mode 100644
index 00000000..38519757
--- /dev/null
+++ b/resize/utils.ml
@@ -0,0 +1,154 @@
+(* virt-resize
+ * Copyright (C) 2010-2011 Red Hat Inc.
+ *
+ * This program 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 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program 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 this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+ *)
+
+open Printf
+
+module G = Guestfs
+
+let ( +^ ) = Int64.add
+let ( -^ ) = Int64.sub
+let ( *^ ) = Int64.mul
+let ( /^ ) = Int64.div
+let ( &^ ) = Int64.logand
+let ( ~^ ) = Int64.lognot
+
+let output_spaces chan n = for i = 0 to n-1 do output_char chan ' ' done
+
+let wrap ?(chan = stdout) ?(hanging = 0) str =
+ let rec _wrap col str =
+ let n = String.length str in
+ let i = try String.index str ' ' with Not_found -> n in
+ let col =
+ if col+i >= 72 then (
+ output_char chan '\n';
+ output_spaces chan hanging;
+ i+hanging+1
+ ) else col+i+1 in
+ output_string chan (String.sub str 0 i);
+ if i < n then (
+ output_char chan ' ';
+ _wrap col (String.sub str (i+1) (n-(i+1)))
+ )
+ in
+ _wrap 0 str
+
+let error fs =
+ let display str =
+ wrap ~chan:stderr ("virt-resize: error: " ^ str);
+ prerr_newline ();
+ prerr_newline ();
+ wrap ~chan:stderr
+ "If reporting bugs, run virt-resize with the '-d' option and include the complete output.";
+ prerr_newline ();
+ exit 1
+ in
+ ksprintf display fs
+
+(* The reverse of device name translation, see
+ * BLOCK DEVICE NAMING in guestfs(3).
+ *)
+let canonicalize dev =
+ if String.length dev >= 8 &&
+ dev.[0] = '/' && dev.[1] = 'd' && dev.[2] = 'e' && dev.[3] = 'v' &&
+ dev.[4] = '/' && (dev.[5] = 'h' || dev.[5] = 'v') && dev.[6] = 'd' then (
+ let dev = String.copy dev in
+ dev.[5] <- 's';
+ dev
+ )
+ else
+ dev
+
+let feature_available (g : Guestfs.guestfs) names =
+ try g#available names; true
+ with G.Error _ -> false
+
+(* Parse the size field from --resize and --resize-force options. *)
+let parse_size =
+ let const_re = Pcre.regexp "^([.\\d]+)([bKMG])$"
+ and plus_const_re = Pcre.regexp "^\\+([.\\d]+)([bKMG])$"
+ and minus_const_re = Pcre.regexp "^-([.\\d]+)([bKMG])$"
+ and percent_re = Pcre.regexp "^([.\\d]+)%$"
+ and plus_percent_re = Pcre.regexp "^\\+([.\\d]+)%$"
+ and minus_percent_re = Pcre.regexp "^-([.\\d]+)%$"
+ in
+ fun oldsize field ->
+ let subs = ref None in
+ let matches rex =
+ try subs := Some (Pcre.exec ~rex field); true
+ with Not_found -> false
+ in
+ let sub i =
+ match !subs with None -> assert false
+ | Some subs -> Pcre.get_substring subs i
+ in
+ let size_scaled f = function
+ | "b" -> Int64.of_float f
+ | "K" -> Int64.of_float (f *. 1024.)
+ | "M" -> Int64.of_float (f *. 1024. *. 1024.)
+ | "G" -> Int64.of_float (f *. 1024. *. 1024. *. 1024.)
+ | _ -> assert false
+ in
+
+ if matches const_re then (
+ size_scaled (float_of_string (sub 1)) (sub 2)
+ )
+ else if matches plus_const_re then (
+ let incr = size_scaled (float_of_string (sub 1)) (sub 2) in
+ oldsize +^ incr
+ )
+ else if matches minus_const_re then (
+ let incr = size_scaled (float_of_string (sub 1)) (sub 2) in
+ oldsize -^ incr
+ )
+ else if matches percent_re then (
+ let percent = Int64.of_float (10. *. float_of_string (sub 1)) in
+ oldsize *^ percent /^ 1000L
+ )
+ else if matches plus_percent_re then (
+ let percent = Int64.of_float (10. *. float_of_string (sub 1)) in
+ oldsize +^ oldsize *^ percent /^ 1000L
+ )
+ else if matches minus_percent_re then (
+ let percent = Int64.of_float (10. *. float_of_string (sub 1)) in
+ oldsize -^ oldsize *^ percent /^ 1000L
+ )
+ else
+ error "virt-resize: %s: cannot parse size field" field
+
+let human_size i =
+ let sign, i = if i < 0L then "-", Int64.neg i else "", i in
+
+ if i < 1024L then
+ sprintf "%s%Ld" sign i
+ else (
+ let f = Int64.to_float i /. 1024. in
+ let i = i /^ 1024L in
+ if i < 1024L then
+ sprintf "%s%.1fK" sign f
+ else (
+ let f = Int64.to_float i /. 1024. in
+ let i = i /^ 1024L in
+ if i < 1024L then
+ sprintf "%s%.1fM" sign f
+ else (
+ let f = Int64.to_float i /. 1024. in
+ (*let i = i /^ 1024L in*)
+ sprintf "%s%.1fG" sign f
+ )
+ )
+ )