fish: Print extended help when the user types an unknown command first.
[libguestfs.git] / src / generator.ml
1 #!/usr/bin/env ocaml
2 (* libguestfs
3  * Copyright (C) 2009-2010 Red Hat Inc.
4  *
5  * This program is free software; you can redistribute it and/or modify
6  * it under the terms of the GNU General Public License as published by
7  * the Free Software Foundation; either version 2 of the License, or
8  * (at your option) any later version.
9  *
10  * This program is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13  * GNU General Public License for more details.
14  *
15  * You should have received a copy of the GNU General Public License
16  * along with this program; if not, write to the Free Software
17  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
18  *)
19
20 (* This script generates a large amount of code and documentation for
21  * all the daemon actions.
22  *
23  * To add a new action there are only two files you need to change,
24  * this one to describe the interface (see the big table of
25  * 'daemon_functions' below), and daemon/<somefile>.c to write the
26  * implementation.
27  *
28  * After editing this file, run it (./src/generator.ml) to regenerate
29  * all the output files.  'make' will rerun this automatically when
30  * necessary.  Note that if you are using a separate build directory
31  * you must run generator.ml from the _source_ directory.
32  *
33  * IMPORTANT: This script should NOT print any warnings.  If it prints
34  * warnings, you should treat them as errors.
35  *
36  * OCaml tips:
37  * (1) In emacs, install tuareg-mode to display and format OCaml code
38  * correctly.  'vim' comes with a good OCaml editing mode by default.
39  * (2) Read the resources at http://ocaml-tutorial.org/
40  *)
41
42 #load "unix.cma";;
43 #load "str.cma";;
44 #directory "+xml-light";;
45 #directory "+../pkg-lib/xml-light";; (* for GODI users *)
46 #load "xml-light.cma";;
47
48 open Unix
49 open Printf
50
51 type style = ret * args
52 and ret =
53     (* "RErr" as a return value means an int used as a simple error
54      * indication, ie. 0 or -1.
55      *)
56   | RErr
57
58     (* "RInt" as a return value means an int which is -1 for error
59      * or any value >= 0 on success.  Only use this for smallish
60      * positive ints (0 <= i < 2^30).
61      *)
62   | RInt of string
63
64     (* "RInt64" is the same as RInt, but is guaranteed to be able
65      * to return a full 64 bit value, _except_ that -1 means error
66      * (so -1 cannot be a valid, non-error return value).
67      *)
68   | RInt64 of string
69
70     (* "RBool" is a bool return value which can be true/false or
71      * -1 for error.
72      *)
73   | RBool of string
74
75     (* "RConstString" is a string that refers to a constant value.
76      * The return value must NOT be NULL (since NULL indicates
77      * an error).
78      *
79      * Try to avoid using this.  In particular you cannot use this
80      * for values returned from the daemon, because there is no
81      * thread-safe way to return them in the C API.
82      *)
83   | RConstString of string
84
85     (* "RConstOptString" is an even more broken version of
86      * "RConstString".  The returned string may be NULL and there
87      * is no way to return an error indication.  Avoid using this!
88      *)
89   | RConstOptString of string
90
91     (* "RString" is a returned string.  It must NOT be NULL, since
92      * a NULL return indicates an error.  The caller frees this.
93      *)
94   | RString of string
95
96     (* "RStringList" is a list of strings.  No string in the list
97      * can be NULL.  The caller frees the strings and the array.
98      *)
99   | RStringList of string
100
101     (* "RStruct" is a function which returns a single named structure
102      * or an error indication (in C, a struct, and in other languages
103      * with varying representations, but usually very efficient).  See
104      * after the function list below for the structures.
105      *)
106   | RStruct of string * string          (* name of retval, name of struct *)
107
108     (* "RStructList" is a function which returns either a list/array
109      * of structures (could be zero-length), or an error indication.
110      *)
111   | RStructList of string * string      (* name of retval, name of struct *)
112
113     (* Key-value pairs of untyped strings.  Turns into a hashtable or
114      * dictionary in languages which support it.  DON'T use this as a
115      * general "bucket" for results.  Prefer a stronger typed return
116      * value if one is available, or write a custom struct.  Don't use
117      * this if the list could potentially be very long, since it is
118      * inefficient.  Keys should be unique.  NULLs are not permitted.
119      *)
120   | RHashtable of string
121
122     (* "RBufferOut" is handled almost exactly like RString, but
123      * it allows the string to contain arbitrary 8 bit data including
124      * ASCII NUL.  In the C API this causes an implicit extra parameter
125      * to be added of type <size_t *size_r>.  The extra parameter
126      * returns the actual size of the return buffer in bytes.
127      *
128      * Other programming languages support strings with arbitrary 8 bit
129      * data.
130      *
131      * At the RPC layer we have to use the opaque<> type instead of
132      * string<>.  Returned data is still limited to the max message
133      * size (ie. ~ 2 MB).
134      *)
135   | RBufferOut of string
136
137 and args = argt list    (* Function parameters, guestfs handle is implicit. *)
138
139     (* Note in future we should allow a "variable args" parameter as
140      * the final parameter, to allow commands like
141      *   chmod mode file [file(s)...]
142      * This is not implemented yet, but many commands (such as chmod)
143      * are currently defined with the argument order keeping this future
144      * possibility in mind.
145      *)
146 and argt =
147   | String of string    (* const char *name, cannot be NULL *)
148   | Device of string    (* /dev device name, cannot be NULL *)
149   | Pathname of string  (* file name, cannot be NULL *)
150   | Dev_or_Path of string (* /dev device name or Pathname, cannot be NULL *)
151   | OptString of string (* const char *name, may be NULL *)
152   | StringList of string(* list of strings (each string cannot be NULL) *)
153   | DeviceList of string(* list of Device names (each cannot be NULL) *)
154   | Bool of string      (* boolean *)
155   | Int of string       (* int (smallish ints, signed, <= 31 bits) *)
156   | Int64 of string     (* any 64 bit int *)
157     (* These are treated as filenames (simple string parameters) in
158      * the C API and bindings.  But in the RPC protocol, we transfer
159      * the actual file content up to or down from the daemon.
160      * FileIn: local machine -> daemon (in request)
161      * FileOut: daemon -> local machine (in reply)
162      * In guestfish (only), the special name "-" means read from
163      * stdin or write to stdout.
164      *)
165   | FileIn of string
166   | FileOut of string
167 (* Not implemented:
168     (* Opaque buffer which can contain arbitrary 8 bit data.
169      * In the C API, this is expressed as <char *, int> pair.
170      * Most other languages have a string type which can contain
171      * ASCII NUL.  We use whatever type is appropriate for each
172      * language.
173      * Buffers are limited by the total message size.  To transfer
174      * large blocks of data, use FileIn/FileOut parameters instead.
175      * To return an arbitrary buffer, use RBufferOut.
176      *)
177   | BufferIn of string
178 *)
179
180 type flags =
181   | ProtocolLimitWarning  (* display warning about protocol size limits *)
182   | DangerWillRobinson    (* flags particularly dangerous commands *)
183   | FishAlias of string   (* provide an alias for this cmd in guestfish *)
184   | FishAction of string  (* call this function in guestfish *)
185   | NotInFish             (* do not export via guestfish *)
186   | NotInDocs             (* do not add this function to documentation *)
187   | DeprecatedBy of string (* function is deprecated, use .. instead *)
188   | Optional of string    (* function is part of an optional group *)
189
190 (* You can supply zero or as many tests as you want per API call.
191  *
192  * Note that the test environment has 3 block devices, of size 500MB,
193  * 50MB and 10MB (respectively /dev/sda, /dev/sdb, /dev/sdc), and
194  * a fourth ISO block device with some known files on it (/dev/sdd).
195  *
196  * Note for partitioning purposes, the 500MB device has 1015 cylinders.
197  * Number of cylinders was 63 for IDE emulated disks with precisely
198  * the same size.  How exactly this is calculated is a mystery.
199  *
200  * The ISO block device (/dev/sdd) comes from images/test.iso.
201  *
202  * To be able to run the tests in a reasonable amount of time,
203  * the virtual machine and block devices are reused between tests.
204  * So don't try testing kill_subprocess :-x
205  *
206  * Between each test we blockdev-setrw, umount-all, lvm-remove-all.
207  *
208  * Don't assume anything about the previous contents of the block
209  * devices.  Use 'Init*' to create some initial scenarios.
210  *
211  * You can add a prerequisite clause to any individual test.  This
212  * is a run-time check, which, if it fails, causes the test to be
213  * skipped.  Useful if testing a command which might not work on
214  * all variations of libguestfs builds.  A test that has prerequisite
215  * of 'Always' is run unconditionally.
216  *
217  * In addition, packagers can skip individual tests by setting the
218  * environment variables:     eg:
219  *   SKIP_TEST_<CMD>_<NUM>=1  SKIP_TEST_COMMAND_3=1  (skips test #3 of command)
220  *   SKIP_TEST_<CMD>=1        SKIP_TEST_ZEROFREE=1   (skips all zerofree tests)
221  *)
222 type tests = (test_init * test_prereq * test) list
223 and test =
224     (* Run the command sequence and just expect nothing to fail. *)
225   | TestRun of seq
226
227     (* Run the command sequence and expect the output of the final
228      * command to be the string.
229      *)
230   | TestOutput of seq * string
231
232     (* Run the command sequence and expect the output of the final
233      * command to be the list of strings.
234      *)
235   | TestOutputList of seq * string list
236
237     (* Run the command sequence and expect the output of the final
238      * command to be the list of block devices (could be either
239      * "/dev/sd.." or "/dev/hd.." form - we don't check the 5th
240      * character of each string).
241      *)
242   | TestOutputListOfDevices of seq * string list
243
244     (* Run the command sequence and expect the output of the final
245      * command to be the integer.
246      *)
247   | TestOutputInt of seq * int
248
249     (* Run the command sequence and expect the output of the final
250      * command to be <op> <int>, eg. ">=", "1".
251      *)
252   | TestOutputIntOp of seq * string * int
253
254     (* Run the command sequence and expect the output of the final
255      * command to be a true value (!= 0 or != NULL).
256      *)
257   | TestOutputTrue of seq
258
259     (* Run the command sequence and expect the output of the final
260      * command to be a false value (== 0 or == NULL, but not an error).
261      *)
262   | TestOutputFalse of seq
263
264     (* Run the command sequence and expect the output of the final
265      * command to be a list of the given length (but don't care about
266      * content).
267      *)
268   | TestOutputLength of seq * int
269
270     (* Run the command sequence and expect the output of the final
271      * command to be a buffer (RBufferOut), ie. string + size.
272      *)
273   | TestOutputBuffer of seq * string
274
275     (* Run the command sequence and expect the output of the final
276      * command to be a structure.
277      *)
278   | TestOutputStruct of seq * test_field_compare list
279
280     (* Run the command sequence and expect the final command (only)
281      * to fail.
282      *)
283   | TestLastFail of seq
284
285 and test_field_compare =
286   | CompareWithInt of string * int
287   | CompareWithIntOp of string * string * int
288   | CompareWithString of string * string
289   | CompareFieldsIntEq of string * string
290   | CompareFieldsStrEq of string * string
291
292 (* Test prerequisites. *)
293 and test_prereq =
294     (* Test always runs. *)
295   | Always
296
297     (* Test is currently disabled - eg. it fails, or it tests some
298      * unimplemented feature.
299      *)
300   | Disabled
301
302     (* 'string' is some C code (a function body) that should return
303      * true or false.  The test will run if the code returns true.
304      *)
305   | If of string
306
307     (* As for 'If' but the test runs _unless_ the code returns true. *)
308   | Unless of string
309
310 (* Some initial scenarios for testing. *)
311 and test_init =
312     (* Do nothing, block devices could contain random stuff including
313      * LVM PVs, and some filesystems might be mounted.  This is usually
314      * a bad idea.
315      *)
316   | InitNone
317
318     (* Block devices are empty and no filesystems are mounted. *)
319   | InitEmpty
320
321     (* /dev/sda contains a single partition /dev/sda1, with random
322      * content.  /dev/sdb and /dev/sdc may have random content.
323      * No LVM.
324      *)
325   | InitPartition
326
327     (* /dev/sda contains a single partition /dev/sda1, which is formatted
328      * as ext2, empty [except for lost+found] and mounted on /.
329      * /dev/sdb and /dev/sdc may have random content.
330      * No LVM.
331      *)
332   | InitBasicFS
333
334     (* /dev/sda:
335      *   /dev/sda1 (is a PV):
336      *     /dev/VG/LV (size 8MB):
337      *       formatted as ext2, empty [except for lost+found], mounted on /
338      * /dev/sdb and /dev/sdc may have random content.
339      *)
340   | InitBasicFSonLVM
341
342     (* /dev/sdd (the ISO, see images/ directory in source)
343      * is mounted on /
344      *)
345   | InitISOFS
346
347 (* Sequence of commands for testing. *)
348 and seq = cmd list
349 and cmd = string list
350
351 (* Note about long descriptions: When referring to another
352  * action, use the format C<guestfs_other> (ie. the full name of
353  * the C function).  This will be replaced as appropriate in other
354  * language bindings.
355  *
356  * Apart from that, long descriptions are just perldoc paragraphs.
357  *)
358
359 (* Generate a random UUID (used in tests). *)
360 let uuidgen () =
361   let chan = open_process_in "uuidgen" in
362   let uuid = input_line chan in
363   (match close_process_in chan with
364    | WEXITED 0 -> ()
365    | WEXITED _ ->
366        failwith "uuidgen: process exited with non-zero status"
367    | WSIGNALED _ | WSTOPPED _ ->
368        failwith "uuidgen: process signalled or stopped by signal"
369   );
370   uuid
371
372 (* These test functions are used in the language binding tests. *)
373
374 let test_all_args = [
375   String "str";
376   OptString "optstr";
377   StringList "strlist";
378   Bool "b";
379   Int "integer";
380   Int64 "integer64";
381   FileIn "filein";
382   FileOut "fileout";
383 ]
384
385 let test_all_rets = [
386   (* except for RErr, which is tested thoroughly elsewhere *)
387   "test0rint",         RInt "valout";
388   "test0rint64",       RInt64 "valout";
389   "test0rbool",        RBool "valout";
390   "test0rconststring", RConstString "valout";
391   "test0rconstoptstring", RConstOptString "valout";
392   "test0rstring",      RString "valout";
393   "test0rstringlist",  RStringList "valout";
394   "test0rstruct",      RStruct ("valout", "lvm_pv");
395   "test0rstructlist",  RStructList ("valout", "lvm_pv");
396   "test0rhashtable",   RHashtable "valout";
397 ]
398
399 let test_functions = [
400   ("test0", (RErr, test_all_args), -1, [NotInFish; NotInDocs],
401    [],
402    "internal test function - do not use",
403    "\
404 This is an internal test function which is used to test whether
405 the automatically generated bindings can handle every possible
406 parameter type correctly.
407
408 It echos the contents of each parameter to stdout.
409
410 You probably don't want to call this function.");
411 ] @ List.flatten (
412   List.map (
413     fun (name, ret) ->
414       [(name, (ret, [String "val"]), -1, [NotInFish; NotInDocs],
415         [],
416         "internal test function - do not use",
417         "\
418 This is an internal test function which is used to test whether
419 the automatically generated bindings can handle every possible
420 return type correctly.
421
422 It converts string C<val> to the return type.
423
424 You probably don't want to call this function.");
425        (name ^ "err", (ret, []), -1, [NotInFish; NotInDocs],
426         [],
427         "internal test function - do not use",
428         "\
429 This is an internal test function which is used to test whether
430 the automatically generated bindings can handle every possible
431 return type correctly.
432
433 This function always returns an error.
434
435 You probably don't want to call this function.")]
436   ) test_all_rets
437 )
438
439 (* non_daemon_functions are any functions which don't get processed
440  * in the daemon, eg. functions for setting and getting local
441  * configuration values.
442  *)
443
444 let non_daemon_functions = test_functions @ [
445   ("launch", (RErr, []), -1, [FishAlias "run"; FishAction "launch"],
446    [],
447    "launch the qemu subprocess",
448    "\
449 Internally libguestfs is implemented by running a virtual machine
450 using L<qemu(1)>.
451
452 You should call this after configuring the handle
453 (eg. adding drives) but before performing any actions.");
454
455   ("wait_ready", (RErr, []), -1, [NotInFish],
456    [],
457    "wait until the qemu subprocess launches (no op)",
458    "\
459 This function is a no op.
460
461 In versions of the API E<lt> 1.0.71 you had to call this function
462 just after calling C<guestfs_launch> to wait for the launch
463 to complete.  However this is no longer necessary because
464 C<guestfs_launch> now does the waiting.
465
466 If you see any calls to this function in code then you can just
467 remove them, unless you want to retain compatibility with older
468 versions of the API.");
469
470   ("kill_subprocess", (RErr, []), -1, [],
471    [],
472    "kill the qemu subprocess",
473    "\
474 This kills the qemu subprocess.  You should never need to call this.");
475
476   ("add_drive", (RErr, [String "filename"]), -1, [FishAlias "add"],
477    [],
478    "add an image to examine or modify",
479    "\
480 This function adds a virtual machine disk image C<filename> to the
481 guest.  The first time you call this function, the disk appears as IDE
482 disk 0 (C</dev/sda>) in the guest, the second time as C</dev/sdb>, and
483 so on.
484
485 You don't necessarily need to be root when using libguestfs.  However
486 you obviously do need sufficient permissions to access the filename
487 for whatever operations you want to perform (ie. read access if you
488 just want to read the image or write access if you want to modify the
489 image).
490
491 This is equivalent to the qemu parameter
492 C<-drive file=filename,cache=off,if=...>.
493
494 C<cache=off> is omitted in cases where it is not supported by
495 the underlying filesystem.
496
497 C<if=...> is set at compile time by the configuration option
498 C<./configure --with-drive-if=...>.  In the rare case where you
499 might need to change this at run time, use C<guestfs_add_drive_with_if>
500 or C<guestfs_add_drive_ro_with_if>.
501
502 Note that this call checks for the existence of C<filename>.  This
503 stops you from specifying other types of drive which are supported
504 by qemu such as C<nbd:> and C<http:> URLs.  To specify those, use
505 the general C<guestfs_config> call instead.");
506
507   ("add_cdrom", (RErr, [String "filename"]), -1, [FishAlias "cdrom"],
508    [],
509    "add a CD-ROM disk image to examine",
510    "\
511 This function adds a virtual CD-ROM disk image to the guest.
512
513 This is equivalent to the qemu parameter C<-cdrom filename>.
514
515 Notes:
516
517 =over 4
518
519 =item *
520
521 This call checks for the existence of C<filename>.  This
522 stops you from specifying other types of drive which are supported
523 by qemu such as C<nbd:> and C<http:> URLs.  To specify those, use
524 the general C<guestfs_config> call instead.
525
526 =item *
527
528 If you just want to add an ISO file (often you use this as an
529 efficient way to transfer large files into the guest), then you
530 should probably use C<guestfs_add_drive_ro> instead.
531
532 =back");
533
534   ("add_drive_ro", (RErr, [String "filename"]), -1, [FishAlias "add-ro"],
535    [],
536    "add a drive in snapshot mode (read-only)",
537    "\
538 This adds a drive in snapshot mode, making it effectively
539 read-only.
540
541 Note that writes to the device are allowed, and will be seen for
542 the duration of the guestfs handle, but they are written
543 to a temporary file which is discarded as soon as the guestfs
544 handle is closed.  We don't currently have any method to enable
545 changes to be committed, although qemu can support this.
546
547 This is equivalent to the qemu parameter
548 C<-drive file=filename,snapshot=on,readonly=on,if=...>.
549
550 C<if=...> is set at compile time by the configuration option
551 C<./configure --with-drive-if=...>.  In the rare case where you
552 might need to change this at run time, use C<guestfs_add_drive_with_if>
553 or C<guestfs_add_drive_ro_with_if>.
554
555 C<readonly=on> is only added where qemu supports this option.
556
557 Note that this call checks for the existence of C<filename>.  This
558 stops you from specifying other types of drive which are supported
559 by qemu such as C<nbd:> and C<http:> URLs.  To specify those, use
560 the general C<guestfs_config> call instead.");
561
562   ("config", (RErr, [String "qemuparam"; OptString "qemuvalue"]), -1, [],
563    [],
564    "add qemu parameters",
565    "\
566 This can be used to add arbitrary qemu command line parameters
567 of the form C<-param value>.  Actually it's not quite arbitrary - we
568 prevent you from setting some parameters which would interfere with
569 parameters that we use.
570
571 The first character of C<param> string must be a C<-> (dash).
572
573 C<value> can be NULL.");
574
575   ("set_qemu", (RErr, [String "qemu"]), -1, [FishAlias "qemu"],
576    [],
577    "set the qemu binary",
578    "\
579 Set the qemu binary that we will use.
580
581 The default is chosen when the library was compiled by the
582 configure script.
583
584 You can also override this by setting the C<LIBGUESTFS_QEMU>
585 environment variable.
586
587 Setting C<qemu> to C<NULL> restores the default qemu binary.
588
589 Note that you should call this function as early as possible
590 after creating the handle.  This is because some pre-launch
591 operations depend on testing qemu features (by running C<qemu -help>).
592 If the qemu binary changes, we don't retest features, and
593 so you might see inconsistent results.  Using the environment
594 variable C<LIBGUESTFS_QEMU> is safest of all since that picks
595 the qemu binary at the same time as the handle is created.");
596
597   ("get_qemu", (RConstString "qemu", []), -1, [],
598    [InitNone, Always, TestRun (
599       [["get_qemu"]])],
600    "get the qemu binary",
601    "\
602 Return the current qemu binary.
603
604 This is always non-NULL.  If it wasn't set already, then this will
605 return the default qemu binary name.");
606
607   ("set_path", (RErr, [String "searchpath"]), -1, [FishAlias "path"],
608    [],
609    "set the search path",
610    "\
611 Set the path that libguestfs searches for kernel and initrd.img.
612
613 The default is C<$libdir/guestfs> unless overridden by setting
614 C<LIBGUESTFS_PATH> environment variable.
615
616 Setting C<path> to C<NULL> restores the default path.");
617
618   ("get_path", (RConstString "path", []), -1, [],
619    [InitNone, Always, TestRun (
620       [["get_path"]])],
621    "get the search path",
622    "\
623 Return the current search path.
624
625 This is always non-NULL.  If it wasn't set already, then this will
626 return the default path.");
627
628   ("set_append", (RErr, [OptString "append"]), -1, [FishAlias "append"],
629    [],
630    "add options to kernel command line",
631    "\
632 This function is used to add additional options to the
633 guest kernel command line.
634
635 The default is C<NULL> unless overridden by setting
636 C<LIBGUESTFS_APPEND> environment variable.
637
638 Setting C<append> to C<NULL> means I<no> additional options
639 are passed (libguestfs always adds a few of its own).");
640
641   ("get_append", (RConstOptString "append", []), -1, [],
642    (* This cannot be tested with the current framework.  The
643     * function can return NULL in normal operations, which the
644     * test framework interprets as an error.
645     *)
646    [],
647    "get the additional kernel options",
648    "\
649 Return the additional kernel options which are added to the
650 guest kernel command line.
651
652 If C<NULL> then no options are added.");
653
654   ("set_autosync", (RErr, [Bool "autosync"]), -1, [FishAlias "autosync"],
655    [],
656    "set autosync mode",
657    "\
658 If C<autosync> is true, this enables autosync.  Libguestfs will make a
659 best effort attempt to run C<guestfs_umount_all> followed by
660 C<guestfs_sync> when the handle is closed
661 (also if the program exits without closing handles).
662
663 This is disabled by default (except in guestfish where it is
664 enabled by default).");
665
666   ("get_autosync", (RBool "autosync", []), -1, [],
667    [InitNone, Always, TestRun (
668       [["get_autosync"]])],
669    "get autosync mode",
670    "\
671 Get the autosync flag.");
672
673   ("set_verbose", (RErr, [Bool "verbose"]), -1, [FishAlias "verbose"],
674    [],
675    "set verbose mode",
676    "\
677 If C<verbose> is true, this turns on verbose messages (to C<stderr>).
678
679 Verbose messages are disabled unless the environment variable
680 C<LIBGUESTFS_DEBUG> is defined and set to C<1>.");
681
682   ("get_verbose", (RBool "verbose", []), -1, [],
683    [],
684    "get verbose mode",
685    "\
686 This returns the verbose messages flag.");
687
688   ("is_ready", (RBool "ready", []), -1, [],
689    [InitNone, Always, TestOutputTrue (
690       [["is_ready"]])],
691    "is ready to accept commands",
692    "\
693 This returns true iff this handle is ready to accept commands
694 (in the C<READY> state).
695
696 For more information on states, see L<guestfs(3)>.");
697
698   ("is_config", (RBool "config", []), -1, [],
699    [InitNone, Always, TestOutputFalse (
700       [["is_config"]])],
701    "is in configuration state",
702    "\
703 This returns true iff this handle is being configured
704 (in the C<CONFIG> state).
705
706 For more information on states, see L<guestfs(3)>.");
707
708   ("is_launching", (RBool "launching", []), -1, [],
709    [InitNone, Always, TestOutputFalse (
710       [["is_launching"]])],
711    "is launching subprocess",
712    "\
713 This returns true iff this handle is launching the subprocess
714 (in the C<LAUNCHING> state).
715
716 For more information on states, see L<guestfs(3)>.");
717
718   ("is_busy", (RBool "busy", []), -1, [],
719    [InitNone, Always, TestOutputFalse (
720       [["is_busy"]])],
721    "is busy processing a command",
722    "\
723 This returns true iff this handle is busy processing a command
724 (in the C<BUSY> state).
725
726 For more information on states, see L<guestfs(3)>.");
727
728   ("get_state", (RInt "state", []), -1, [],
729    [],
730    "get the current state",
731    "\
732 This returns the current state as an opaque integer.  This is
733 only useful for printing debug and internal error messages.
734
735 For more information on states, see L<guestfs(3)>.");
736
737   ("set_memsize", (RErr, [Int "memsize"]), -1, [FishAlias "memsize"],
738    [InitNone, Always, TestOutputInt (
739       [["set_memsize"; "500"];
740        ["get_memsize"]], 500)],
741    "set memory allocated to the qemu subprocess",
742    "\
743 This sets the memory size in megabytes allocated to the
744 qemu subprocess.  This only has any effect if called before
745 C<guestfs_launch>.
746
747 You can also change this by setting the environment
748 variable C<LIBGUESTFS_MEMSIZE> before the handle is
749 created.
750
751 For more information on the architecture of libguestfs,
752 see L<guestfs(3)>.");
753
754   ("get_memsize", (RInt "memsize", []), -1, [],
755    [InitNone, Always, TestOutputIntOp (
756       [["get_memsize"]], ">=", 256)],
757    "get memory allocated to the qemu subprocess",
758    "\
759 This gets the memory size in megabytes allocated to the
760 qemu subprocess.
761
762 If C<guestfs_set_memsize> was not called
763 on this handle, and if C<LIBGUESTFS_MEMSIZE> was not set,
764 then this returns the compiled-in default value for memsize.
765
766 For more information on the architecture of libguestfs,
767 see L<guestfs(3)>.");
768
769   ("get_pid", (RInt "pid", []), -1, [FishAlias "pid"],
770    [InitNone, Always, TestOutputIntOp (
771       [["get_pid"]], ">=", 1)],
772    "get PID of qemu subprocess",
773    "\
774 Return the process ID of the qemu subprocess.  If there is no
775 qemu subprocess, then this will return an error.
776
777 This is an internal call used for debugging and testing.");
778
779   ("version", (RStruct ("version", "version"), []), -1, [],
780    [InitNone, Always, TestOutputStruct (
781       [["version"]], [CompareWithInt ("major", 1)])],
782    "get the library version number",
783    "\
784 Return the libguestfs version number that the program is linked
785 against.
786
787 Note that because of dynamic linking this is not necessarily
788 the version of libguestfs that you compiled against.  You can
789 compile the program, and then at runtime dynamically link
790 against a completely different C<libguestfs.so> library.
791
792 This call was added in version C<1.0.58>.  In previous
793 versions of libguestfs there was no way to get the version
794 number.  From C code you can use ELF weak linking tricks to find out if
795 this symbol exists (if it doesn't, then it's an earlier version).
796
797 The call returns a structure with four elements.  The first
798 three (C<major>, C<minor> and C<release>) are numbers and
799 correspond to the usual version triplet.  The fourth element
800 (C<extra>) is a string and is normally empty, but may be
801 used for distro-specific information.
802
803 To construct the original version string:
804 C<$major.$minor.$release$extra>
805
806 I<Note:> Don't use this call to test for availability
807 of features.  Distro backports makes this unreliable.  Use
808 C<guestfs_available> instead.");
809
810   ("set_selinux", (RErr, [Bool "selinux"]), -1, [FishAlias "selinux"],
811    [InitNone, Always, TestOutputTrue (
812       [["set_selinux"; "true"];
813        ["get_selinux"]])],
814    "set SELinux enabled or disabled at appliance boot",
815    "\
816 This sets the selinux flag that is passed to the appliance
817 at boot time.  The default is C<selinux=0> (disabled).
818
819 Note that if SELinux is enabled, it is always in
820 Permissive mode (C<enforcing=0>).
821
822 For more information on the architecture of libguestfs,
823 see L<guestfs(3)>.");
824
825   ("get_selinux", (RBool "selinux", []), -1, [],
826    [],
827    "get SELinux enabled flag",
828    "\
829 This returns the current setting of the selinux flag which
830 is passed to the appliance at boot time.  See C<guestfs_set_selinux>.
831
832 For more information on the architecture of libguestfs,
833 see L<guestfs(3)>.");
834
835   ("set_trace", (RErr, [Bool "trace"]), -1, [FishAlias "trace"],
836    [InitNone, Always, TestOutputFalse (
837       [["set_trace"; "false"];
838        ["get_trace"]])],
839    "enable or disable command traces",
840    "\
841 If the command trace flag is set to 1, then commands are
842 printed on stdout before they are executed in a format
843 which is very similar to the one used by guestfish.  In
844 other words, you can run a program with this enabled, and
845 you will get out a script which you can feed to guestfish
846 to perform the same set of actions.
847
848 If you want to trace C API calls into libguestfs (and
849 other libraries) then possibly a better way is to use
850 the external ltrace(1) command.
851
852 Command traces are disabled unless the environment variable
853 C<LIBGUESTFS_TRACE> is defined and set to C<1>.");
854
855   ("get_trace", (RBool "trace", []), -1, [],
856    [],
857    "get command trace enabled flag",
858    "\
859 Return the command trace flag.");
860
861   ("set_direct", (RErr, [Bool "direct"]), -1, [FishAlias "direct"],
862    [InitNone, Always, TestOutputFalse (
863       [["set_direct"; "false"];
864        ["get_direct"]])],
865    "enable or disable direct appliance mode",
866    "\
867 If the direct appliance mode flag is enabled, then stdin and
868 stdout are passed directly through to the appliance once it
869 is launched.
870
871 One consequence of this is that log messages aren't caught
872 by the library and handled by C<guestfs_set_log_message_callback>,
873 but go straight to stdout.
874
875 You probably don't want to use this unless you know what you
876 are doing.
877
878 The default is disabled.");
879
880   ("get_direct", (RBool "direct", []), -1, [],
881    [],
882    "get direct appliance mode flag",
883    "\
884 Return the direct appliance mode flag.");
885
886   ("set_recovery_proc", (RErr, [Bool "recoveryproc"]), -1, [FishAlias "recovery-proc"],
887    [InitNone, Always, TestOutputTrue (
888       [["set_recovery_proc"; "true"];
889        ["get_recovery_proc"]])],
890    "enable or disable the recovery process",
891    "\
892 If this is called with the parameter C<false> then
893 C<guestfs_launch> does not create a recovery process.  The
894 purpose of the recovery process is to stop runaway qemu
895 processes in the case where the main program aborts abruptly.
896
897 This only has any effect if called before C<guestfs_launch>,
898 and the default is true.
899
900 About the only time when you would want to disable this is
901 if the main process will fork itself into the background
902 (\"daemonize\" itself).  In this case the recovery process
903 thinks that the main program has disappeared and so kills
904 qemu, which is not very helpful.");
905
906   ("get_recovery_proc", (RBool "recoveryproc", []), -1, [],
907    [],
908    "get recovery process enabled flag",
909    "\
910 Return the recovery process enabled flag.");
911
912   ("add_drive_with_if", (RErr, [String "filename"; String "iface"]), -1, [],
913    [],
914    "add a drive specifying the QEMU block emulation to use",
915    "\
916 This is the same as C<guestfs_add_drive> but it allows you
917 to specify the QEMU interface emulation to use at run time.");
918
919   ("add_drive_ro_with_if", (RErr, [String "filename"; String "iface"]), -1, [],
920    [],
921    "add a drive read-only specifying the QEMU block emulation to use",
922    "\
923 This is the same as C<guestfs_add_drive_ro> but it allows you
924 to specify the QEMU interface emulation to use at run time.");
925
926 ]
927
928 (* daemon_functions are any functions which cause some action
929  * to take place in the daemon.
930  *)
931
932 let daemon_functions = [
933   ("mount", (RErr, [Device "device"; String "mountpoint"]), 1, [],
934    [InitEmpty, Always, TestOutput (
935       [["part_disk"; "/dev/sda"; "mbr"];
936        ["mkfs"; "ext2"; "/dev/sda1"];
937        ["mount"; "/dev/sda1"; "/"];
938        ["write_file"; "/new"; "new file contents"; "0"];
939        ["cat"; "/new"]], "new file contents")],
940    "mount a guest disk at a position in the filesystem",
941    "\
942 Mount a guest disk at a position in the filesystem.  Block devices
943 are named C</dev/sda>, C</dev/sdb> and so on, as they were added to
944 the guest.  If those block devices contain partitions, they will have
945 the usual names (eg. C</dev/sda1>).  Also LVM C</dev/VG/LV>-style
946 names can be used.
947
948 The rules are the same as for L<mount(2)>:  A filesystem must
949 first be mounted on C</> before others can be mounted.  Other
950 filesystems can only be mounted on directories which already
951 exist.
952
953 The mounted filesystem is writable, if we have sufficient permissions
954 on the underlying device.
955
956 The filesystem options C<sync> and C<noatime> are set with this
957 call, in order to improve reliability.");
958
959   ("sync", (RErr, []), 2, [],
960    [ InitEmpty, Always, TestRun [["sync"]]],
961    "sync disks, writes are flushed through to the disk image",
962    "\
963 This syncs the disk, so that any writes are flushed through to the
964 underlying disk image.
965
966 You should always call this if you have modified a disk image, before
967 closing the handle.");
968
969   ("touch", (RErr, [Pathname "path"]), 3, [],
970    [InitBasicFS, Always, TestOutputTrue (
971       [["touch"; "/new"];
972        ["exists"; "/new"]])],
973    "update file timestamps or create a new file",
974    "\
975 Touch acts like the L<touch(1)> command.  It can be used to
976 update the timestamps on a file, or, if the file does not exist,
977 to create a new zero-length file.");
978
979   ("cat", (RString "content", [Pathname "path"]), 4, [ProtocolLimitWarning],
980    [InitISOFS, Always, TestOutput (
981       [["cat"; "/known-2"]], "abcdef\n")],
982    "list the contents of a file",
983    "\
984 Return the contents of the file named C<path>.
985
986 Note that this function cannot correctly handle binary files
987 (specifically, files containing C<\\0> character which is treated
988 as end of string).  For those you need to use the C<guestfs_read_file>
989 or C<guestfs_download> functions which have a more complex interface.");
990
991   ("ll", (RString "listing", [Pathname "directory"]), 5, [],
992    [], (* XXX Tricky to test because it depends on the exact format
993         * of the 'ls -l' command, which changes between F10 and F11.
994         *)
995    "list the files in a directory (long format)",
996    "\
997 List the files in C<directory> (relative to the root directory,
998 there is no cwd) in the format of 'ls -la'.
999
1000 This command is mostly useful for interactive sessions.  It
1001 is I<not> intended that you try to parse the output string.");
1002
1003   ("ls", (RStringList "listing", [Pathname "directory"]), 6, [],
1004    [InitBasicFS, Always, TestOutputList (
1005       [["touch"; "/new"];
1006        ["touch"; "/newer"];
1007        ["touch"; "/newest"];
1008        ["ls"; "/"]], ["lost+found"; "new"; "newer"; "newest"])],
1009    "list the files in a directory",
1010    "\
1011 List the files in C<directory> (relative to the root directory,
1012 there is no cwd).  The '.' and '..' entries are not returned, but
1013 hidden files are shown.
1014
1015 This command is mostly useful for interactive sessions.  Programs
1016 should probably use C<guestfs_readdir> instead.");
1017
1018   ("list_devices", (RStringList "devices", []), 7, [],
1019    [InitEmpty, Always, TestOutputListOfDevices (
1020       [["list_devices"]], ["/dev/sda"; "/dev/sdb"; "/dev/sdc"; "/dev/sdd"])],
1021    "list the block devices",
1022    "\
1023 List all the block devices.
1024
1025 The full block device names are returned, eg. C</dev/sda>");
1026
1027   ("list_partitions", (RStringList "partitions", []), 8, [],
1028    [InitBasicFS, Always, TestOutputListOfDevices (
1029       [["list_partitions"]], ["/dev/sda1"]);
1030     InitEmpty, Always, TestOutputListOfDevices (
1031       [["sfdiskM"; "/dev/sda"; ",100 ,200 ,"];
1032        ["list_partitions"]], ["/dev/sda1"; "/dev/sda2"; "/dev/sda3"])],
1033    "list the partitions",
1034    "\
1035 List all the partitions detected on all block devices.
1036
1037 The full partition device names are returned, eg. C</dev/sda1>
1038
1039 This does not return logical volumes.  For that you will need to
1040 call C<guestfs_lvs>.");
1041
1042   ("pvs", (RStringList "physvols", []), 9, [Optional "lvm2"],
1043    [InitBasicFSonLVM, Always, TestOutputListOfDevices (
1044       [["pvs"]], ["/dev/sda1"]);
1045     InitEmpty, Always, TestOutputListOfDevices (
1046       [["sfdiskM"; "/dev/sda"; ",100 ,200 ,"];
1047        ["pvcreate"; "/dev/sda1"];
1048        ["pvcreate"; "/dev/sda2"];
1049        ["pvcreate"; "/dev/sda3"];
1050        ["pvs"]], ["/dev/sda1"; "/dev/sda2"; "/dev/sda3"])],
1051    "list the LVM physical volumes (PVs)",
1052    "\
1053 List all the physical volumes detected.  This is the equivalent
1054 of the L<pvs(8)> command.
1055
1056 This returns a list of just the device names that contain
1057 PVs (eg. C</dev/sda2>).
1058
1059 See also C<guestfs_pvs_full>.");
1060
1061   ("vgs", (RStringList "volgroups", []), 10, [Optional "lvm2"],
1062    [InitBasicFSonLVM, Always, TestOutputList (
1063       [["vgs"]], ["VG"]);
1064     InitEmpty, Always, TestOutputList (
1065       [["sfdiskM"; "/dev/sda"; ",100 ,200 ,"];
1066        ["pvcreate"; "/dev/sda1"];
1067        ["pvcreate"; "/dev/sda2"];
1068        ["pvcreate"; "/dev/sda3"];
1069        ["vgcreate"; "VG1"; "/dev/sda1 /dev/sda2"];
1070        ["vgcreate"; "VG2"; "/dev/sda3"];
1071        ["vgs"]], ["VG1"; "VG2"])],
1072    "list the LVM volume groups (VGs)",
1073    "\
1074 List all the volumes groups detected.  This is the equivalent
1075 of the L<vgs(8)> command.
1076
1077 This returns a list of just the volume group names that were
1078 detected (eg. C<VolGroup00>).
1079
1080 See also C<guestfs_vgs_full>.");
1081
1082   ("lvs", (RStringList "logvols", []), 11, [Optional "lvm2"],
1083    [InitBasicFSonLVM, Always, TestOutputList (
1084       [["lvs"]], ["/dev/VG/LV"]);
1085     InitEmpty, Always, TestOutputList (
1086       [["sfdiskM"; "/dev/sda"; ",100 ,200 ,"];
1087        ["pvcreate"; "/dev/sda1"];
1088        ["pvcreate"; "/dev/sda2"];
1089        ["pvcreate"; "/dev/sda3"];
1090        ["vgcreate"; "VG1"; "/dev/sda1 /dev/sda2"];
1091        ["vgcreate"; "VG2"; "/dev/sda3"];
1092        ["lvcreate"; "LV1"; "VG1"; "50"];
1093        ["lvcreate"; "LV2"; "VG1"; "50"];
1094        ["lvcreate"; "LV3"; "VG2"; "50"];
1095        ["lvs"]], ["/dev/VG1/LV1"; "/dev/VG1/LV2"; "/dev/VG2/LV3"])],
1096    "list the LVM logical volumes (LVs)",
1097    "\
1098 List all the logical volumes detected.  This is the equivalent
1099 of the L<lvs(8)> command.
1100
1101 This returns a list of the logical volume device names
1102 (eg. C</dev/VolGroup00/LogVol00>).
1103
1104 See also C<guestfs_lvs_full>.");
1105
1106   ("pvs_full", (RStructList ("physvols", "lvm_pv"), []), 12, [Optional "lvm2"],
1107    [], (* XXX how to test? *)
1108    "list the LVM physical volumes (PVs)",
1109    "\
1110 List all the physical volumes detected.  This is the equivalent
1111 of the L<pvs(8)> command.  The \"full\" version includes all fields.");
1112
1113   ("vgs_full", (RStructList ("volgroups", "lvm_vg"), []), 13, [Optional "lvm2"],
1114    [], (* XXX how to test? *)
1115    "list the LVM volume groups (VGs)",
1116    "\
1117 List all the volumes groups detected.  This is the equivalent
1118 of the L<vgs(8)> command.  The \"full\" version includes all fields.");
1119
1120   ("lvs_full", (RStructList ("logvols", "lvm_lv"), []), 14, [Optional "lvm2"],
1121    [], (* XXX how to test? *)
1122    "list the LVM logical volumes (LVs)",
1123    "\
1124 List all the logical volumes detected.  This is the equivalent
1125 of the L<lvs(8)> command.  The \"full\" version includes all fields.");
1126
1127   ("read_lines", (RStringList "lines", [Pathname "path"]), 15, [],
1128    [InitISOFS, Always, TestOutputList (
1129       [["read_lines"; "/known-4"]], ["abc"; "def"; "ghi"]);
1130     InitISOFS, Always, TestOutputList (
1131       [["read_lines"; "/empty"]], [])],
1132    "read file as lines",
1133    "\
1134 Return the contents of the file named C<path>.
1135
1136 The file contents are returned as a list of lines.  Trailing
1137 C<LF> and C<CRLF> character sequences are I<not> returned.
1138
1139 Note that this function cannot correctly handle binary files
1140 (specifically, files containing C<\\0> character which is treated
1141 as end of line).  For those you need to use the C<guestfs_read_file>
1142 function which has a more complex interface.");
1143
1144   ("aug_init", (RErr, [Pathname "root"; Int "flags"]), 16, [Optional "augeas"],
1145    [], (* XXX Augeas code needs tests. *)
1146    "create a new Augeas handle",
1147    "\
1148 Create a new Augeas handle for editing configuration files.
1149 If there was any previous Augeas handle associated with this
1150 guestfs session, then it is closed.
1151
1152 You must call this before using any other C<guestfs_aug_*>
1153 commands.
1154
1155 C<root> is the filesystem root.  C<root> must not be NULL,
1156 use C</> instead.
1157
1158 The flags are the same as the flags defined in
1159 E<lt>augeas.hE<gt>, the logical I<or> of the following
1160 integers:
1161
1162 =over 4
1163
1164 =item C<AUG_SAVE_BACKUP> = 1
1165
1166 Keep the original file with a C<.augsave> extension.
1167
1168 =item C<AUG_SAVE_NEWFILE> = 2
1169
1170 Save changes into a file with extension C<.augnew>, and
1171 do not overwrite original.  Overrides C<AUG_SAVE_BACKUP>.
1172
1173 =item C<AUG_TYPE_CHECK> = 4
1174
1175 Typecheck lenses (can be expensive).
1176
1177 =item C<AUG_NO_STDINC> = 8
1178
1179 Do not use standard load path for modules.
1180
1181 =item C<AUG_SAVE_NOOP> = 16
1182
1183 Make save a no-op, just record what would have been changed.
1184
1185 =item C<AUG_NO_LOAD> = 32
1186
1187 Do not load the tree in C<guestfs_aug_init>.
1188
1189 =back
1190
1191 To close the handle, you can call C<guestfs_aug_close>.
1192
1193 To find out more about Augeas, see L<http://augeas.net/>.");
1194
1195   ("aug_close", (RErr, []), 26, [Optional "augeas"],
1196    [], (* XXX Augeas code needs tests. *)
1197    "close the current Augeas handle",
1198    "\
1199 Close the current Augeas handle and free up any resources
1200 used by it.  After calling this, you have to call
1201 C<guestfs_aug_init> again before you can use any other
1202 Augeas functions.");
1203
1204   ("aug_defvar", (RInt "nrnodes", [String "name"; OptString "expr"]), 17, [Optional "augeas"],
1205    [], (* XXX Augeas code needs tests. *)
1206    "define an Augeas variable",
1207    "\
1208 Defines an Augeas variable C<name> whose value is the result
1209 of evaluating C<expr>.  If C<expr> is NULL, then C<name> is
1210 undefined.
1211
1212 On success this returns the number of nodes in C<expr>, or
1213 C<0> if C<expr> evaluates to something which is not a nodeset.");
1214
1215   ("aug_defnode", (RStruct ("nrnodescreated", "int_bool"), [String "name"; String "expr"; String "val"]), 18, [Optional "augeas"],
1216    [], (* XXX Augeas code needs tests. *)
1217    "define an Augeas node",
1218    "\
1219 Defines a variable C<name> whose value is the result of
1220 evaluating C<expr>.
1221
1222 If C<expr> evaluates to an empty nodeset, a node is created,
1223 equivalent to calling C<guestfs_aug_set> C<expr>, C<value>.
1224 C<name> will be the nodeset containing that single node.
1225
1226 On success this returns a pair containing the
1227 number of nodes in the nodeset, and a boolean flag
1228 if a node was created.");
1229
1230   ("aug_get", (RString "val", [String "augpath"]), 19, [Optional "augeas"],
1231    [], (* XXX Augeas code needs tests. *)
1232    "look up the value of an Augeas path",
1233    "\
1234 Look up the value associated with C<path>.  If C<path>
1235 matches exactly one node, the C<value> is returned.");
1236
1237   ("aug_set", (RErr, [String "augpath"; String "val"]), 20, [Optional "augeas"],
1238    [], (* XXX Augeas code needs tests. *)
1239    "set Augeas path to value",
1240    "\
1241 Set the value associated with C<path> to C<value>.");
1242
1243   ("aug_insert", (RErr, [String "augpath"; String "label"; Bool "before"]), 21, [Optional "augeas"],
1244    [], (* XXX Augeas code needs tests. *)
1245    "insert a sibling Augeas node",
1246    "\
1247 Create a new sibling C<label> for C<path>, inserting it into
1248 the tree before or after C<path> (depending on the boolean
1249 flag C<before>).
1250
1251 C<path> must match exactly one existing node in the tree, and
1252 C<label> must be a label, ie. not contain C</>, C<*> or end
1253 with a bracketed index C<[N]>.");
1254
1255   ("aug_rm", (RInt "nrnodes", [String "augpath"]), 22, [Optional "augeas"],
1256    [], (* XXX Augeas code needs tests. *)
1257    "remove an Augeas path",
1258    "\
1259 Remove C<path> and all of its children.
1260
1261 On success this returns the number of entries which were removed.");
1262
1263   ("aug_mv", (RErr, [String "src"; String "dest"]), 23, [Optional "augeas"],
1264    [], (* XXX Augeas code needs tests. *)
1265    "move Augeas node",
1266    "\
1267 Move the node C<src> to C<dest>.  C<src> must match exactly
1268 one node.  C<dest> is overwritten if it exists.");
1269
1270   ("aug_match", (RStringList "matches", [String "augpath"]), 24, [Optional "augeas"],
1271    [], (* XXX Augeas code needs tests. *)
1272    "return Augeas nodes which match augpath",
1273    "\
1274 Returns a list of paths which match the path expression C<path>.
1275 The returned paths are sufficiently qualified so that they match
1276 exactly one node in the current tree.");
1277
1278   ("aug_save", (RErr, []), 25, [Optional "augeas"],
1279    [], (* XXX Augeas code needs tests. *)
1280    "write all pending Augeas changes to disk",
1281    "\
1282 This writes all pending changes to disk.
1283
1284 The flags which were passed to C<guestfs_aug_init> affect exactly
1285 how files are saved.");
1286
1287   ("aug_load", (RErr, []), 27, [Optional "augeas"],
1288    [], (* XXX Augeas code needs tests. *)
1289    "load files into the tree",
1290    "\
1291 Load files into the tree.
1292
1293 See C<aug_load> in the Augeas documentation for the full gory
1294 details.");
1295
1296   ("aug_ls", (RStringList "matches", [String "augpath"]), 28, [Optional "augeas"],
1297    [], (* XXX Augeas code needs tests. *)
1298    "list Augeas nodes under augpath",
1299    "\
1300 This is just a shortcut for listing C<guestfs_aug_match>
1301 C<path/*> and sorting the resulting nodes into alphabetical order.");
1302
1303   ("rm", (RErr, [Pathname "path"]), 29, [],
1304    [InitBasicFS, Always, TestRun
1305       [["touch"; "/new"];
1306        ["rm"; "/new"]];
1307     InitBasicFS, Always, TestLastFail
1308       [["rm"; "/new"]];
1309     InitBasicFS, Always, TestLastFail
1310       [["mkdir"; "/new"];
1311        ["rm"; "/new"]]],
1312    "remove a file",
1313    "\
1314 Remove the single file C<path>.");
1315
1316   ("rmdir", (RErr, [Pathname "path"]), 30, [],
1317    [InitBasicFS, Always, TestRun
1318       [["mkdir"; "/new"];
1319        ["rmdir"; "/new"]];
1320     InitBasicFS, Always, TestLastFail
1321       [["rmdir"; "/new"]];
1322     InitBasicFS, Always, TestLastFail
1323       [["touch"; "/new"];
1324        ["rmdir"; "/new"]]],
1325    "remove a directory",
1326    "\
1327 Remove the single directory C<path>.");
1328
1329   ("rm_rf", (RErr, [Pathname "path"]), 31, [],
1330    [InitBasicFS, Always, TestOutputFalse
1331       [["mkdir"; "/new"];
1332        ["mkdir"; "/new/foo"];
1333        ["touch"; "/new/foo/bar"];
1334        ["rm_rf"; "/new"];
1335        ["exists"; "/new"]]],
1336    "remove a file or directory recursively",
1337    "\
1338 Remove the file or directory C<path>, recursively removing the
1339 contents if its a directory.  This is like the C<rm -rf> shell
1340 command.");
1341
1342   ("mkdir", (RErr, [Pathname "path"]), 32, [],
1343    [InitBasicFS, Always, TestOutputTrue
1344       [["mkdir"; "/new"];
1345        ["is_dir"; "/new"]];
1346     InitBasicFS, Always, TestLastFail
1347       [["mkdir"; "/new/foo/bar"]]],
1348    "create a directory",
1349    "\
1350 Create a directory named C<path>.");
1351
1352   ("mkdir_p", (RErr, [Pathname "path"]), 33, [],
1353    [InitBasicFS, Always, TestOutputTrue
1354       [["mkdir_p"; "/new/foo/bar"];
1355        ["is_dir"; "/new/foo/bar"]];
1356     InitBasicFS, Always, TestOutputTrue
1357       [["mkdir_p"; "/new/foo/bar"];
1358        ["is_dir"; "/new/foo"]];
1359     InitBasicFS, Always, TestOutputTrue
1360       [["mkdir_p"; "/new/foo/bar"];
1361        ["is_dir"; "/new"]];
1362     (* Regression tests for RHBZ#503133: *)
1363     InitBasicFS, Always, TestRun
1364       [["mkdir"; "/new"];
1365        ["mkdir_p"; "/new"]];
1366     InitBasicFS, Always, TestLastFail
1367       [["touch"; "/new"];
1368        ["mkdir_p"; "/new"]]],
1369    "create a directory and parents",
1370    "\
1371 Create a directory named C<path>, creating any parent directories
1372 as necessary.  This is like the C<mkdir -p> shell command.");
1373
1374   ("chmod", (RErr, [Int "mode"; Pathname "path"]), 34, [],
1375    [], (* XXX Need stat command to test *)
1376    "change file mode",
1377    "\
1378 Change the mode (permissions) of C<path> to C<mode>.  Only
1379 numeric modes are supported.
1380
1381 I<Note>: When using this command from guestfish, C<mode>
1382 by default would be decimal, unless you prefix it with
1383 C<0> to get octal, ie. use C<0700> not C<700>.");
1384
1385   ("chown", (RErr, [Int "owner"; Int "group"; Pathname "path"]), 35, [],
1386    [], (* XXX Need stat command to test *)
1387    "change file owner and group",
1388    "\
1389 Change the file owner to C<owner> and group to C<group>.
1390
1391 Only numeric uid and gid are supported.  If you want to use
1392 names, you will need to locate and parse the password file
1393 yourself (Augeas support makes this relatively easy).");
1394
1395   ("exists", (RBool "existsflag", [Pathname "path"]), 36, [],
1396    [InitISOFS, Always, TestOutputTrue (
1397       [["exists"; "/empty"]]);
1398     InitISOFS, Always, TestOutputTrue (
1399       [["exists"; "/directory"]])],
1400    "test if file or directory exists",
1401    "\
1402 This returns C<true> if and only if there is a file, directory
1403 (or anything) with the given C<path> name.
1404
1405 See also C<guestfs_is_file>, C<guestfs_is_dir>, C<guestfs_stat>.");
1406
1407   ("is_file", (RBool "fileflag", [Pathname "path"]), 37, [],
1408    [InitISOFS, Always, TestOutputTrue (
1409       [["is_file"; "/known-1"]]);
1410     InitISOFS, Always, TestOutputFalse (
1411       [["is_file"; "/directory"]])],
1412    "test if file exists",
1413    "\
1414 This returns C<true> if and only if there is a file
1415 with the given C<path> name.  Note that it returns false for
1416 other objects like directories.
1417
1418 See also C<guestfs_stat>.");
1419
1420   ("is_dir", (RBool "dirflag", [Pathname "path"]), 38, [],
1421    [InitISOFS, Always, TestOutputFalse (
1422       [["is_dir"; "/known-3"]]);
1423     InitISOFS, Always, TestOutputTrue (
1424       [["is_dir"; "/directory"]])],
1425    "test if file exists",
1426    "\
1427 This returns C<true> if and only if there is a directory
1428 with the given C<path> name.  Note that it returns false for
1429 other objects like files.
1430
1431 See also C<guestfs_stat>.");
1432
1433   ("pvcreate", (RErr, [Device "device"]), 39, [Optional "lvm2"],
1434    [InitEmpty, Always, TestOutputListOfDevices (
1435       [["sfdiskM"; "/dev/sda"; ",100 ,200 ,"];
1436        ["pvcreate"; "/dev/sda1"];
1437        ["pvcreate"; "/dev/sda2"];
1438        ["pvcreate"; "/dev/sda3"];
1439        ["pvs"]], ["/dev/sda1"; "/dev/sda2"; "/dev/sda3"])],
1440    "create an LVM physical volume",
1441    "\
1442 This creates an LVM physical volume on the named C<device>,
1443 where C<device> should usually be a partition name such
1444 as C</dev/sda1>.");
1445
1446   ("vgcreate", (RErr, [String "volgroup"; DeviceList "physvols"]), 40, [Optional "lvm2"],
1447    [InitEmpty, Always, TestOutputList (
1448       [["sfdiskM"; "/dev/sda"; ",100 ,200 ,"];
1449        ["pvcreate"; "/dev/sda1"];
1450        ["pvcreate"; "/dev/sda2"];
1451        ["pvcreate"; "/dev/sda3"];
1452        ["vgcreate"; "VG1"; "/dev/sda1 /dev/sda2"];
1453        ["vgcreate"; "VG2"; "/dev/sda3"];
1454        ["vgs"]], ["VG1"; "VG2"])],
1455    "create an LVM volume group",
1456    "\
1457 This creates an LVM volume group called C<volgroup>
1458 from the non-empty list of physical volumes C<physvols>.");
1459
1460   ("lvcreate", (RErr, [String "logvol"; String "volgroup"; Int "mbytes"]), 41, [Optional "lvm2"],
1461    [InitEmpty, Always, TestOutputList (
1462       [["sfdiskM"; "/dev/sda"; ",100 ,200 ,"];
1463        ["pvcreate"; "/dev/sda1"];
1464        ["pvcreate"; "/dev/sda2"];
1465        ["pvcreate"; "/dev/sda3"];
1466        ["vgcreate"; "VG1"; "/dev/sda1 /dev/sda2"];
1467        ["vgcreate"; "VG2"; "/dev/sda3"];
1468        ["lvcreate"; "LV1"; "VG1"; "50"];
1469        ["lvcreate"; "LV2"; "VG1"; "50"];
1470        ["lvcreate"; "LV3"; "VG2"; "50"];
1471        ["lvcreate"; "LV4"; "VG2"; "50"];
1472        ["lvcreate"; "LV5"; "VG2"; "50"];
1473        ["lvs"]],
1474       ["/dev/VG1/LV1"; "/dev/VG1/LV2";
1475        "/dev/VG2/LV3"; "/dev/VG2/LV4"; "/dev/VG2/LV5"])],
1476    "create an LVM volume group",
1477    "\
1478 This creates an LVM volume group called C<logvol>
1479 on the volume group C<volgroup>, with C<size> megabytes.");
1480
1481   ("mkfs", (RErr, [String "fstype"; Device "device"]), 42, [],
1482    [InitEmpty, Always, TestOutput (
1483       [["part_disk"; "/dev/sda"; "mbr"];
1484        ["mkfs"; "ext2"; "/dev/sda1"];
1485        ["mount_options"; ""; "/dev/sda1"; "/"];
1486        ["write_file"; "/new"; "new file contents"; "0"];
1487        ["cat"; "/new"]], "new file contents")],
1488    "make a filesystem",
1489    "\
1490 This creates a filesystem on C<device> (usually a partition
1491 or LVM logical volume).  The filesystem type is C<fstype>, for
1492 example C<ext3>.");
1493
1494   ("sfdisk", (RErr, [Device "device";
1495                      Int "cyls"; Int "heads"; Int "sectors";
1496                      StringList "lines"]), 43, [DangerWillRobinson],
1497    [],
1498    "create partitions on a block device",
1499    "\
1500 This is a direct interface to the L<sfdisk(8)> program for creating
1501 partitions on block devices.
1502
1503 C<device> should be a block device, for example C</dev/sda>.
1504
1505 C<cyls>, C<heads> and C<sectors> are the number of cylinders, heads
1506 and sectors on the device, which are passed directly to sfdisk as
1507 the I<-C>, I<-H> and I<-S> parameters.  If you pass C<0> for any
1508 of these, then the corresponding parameter is omitted.  Usually for
1509 'large' disks, you can just pass C<0> for these, but for small
1510 (floppy-sized) disks, sfdisk (or rather, the kernel) cannot work
1511 out the right geometry and you will need to tell it.
1512
1513 C<lines> is a list of lines that we feed to C<sfdisk>.  For more
1514 information refer to the L<sfdisk(8)> manpage.
1515
1516 To create a single partition occupying the whole disk, you would
1517 pass C<lines> as a single element list, when the single element being
1518 the string C<,> (comma).
1519
1520 See also: C<guestfs_sfdisk_l>, C<guestfs_sfdisk_N>,
1521 C<guestfs_part_init>");
1522
1523   ("write_file", (RErr, [Pathname "path"; String "content"; Int "size"]), 44, [ProtocolLimitWarning],
1524    [InitBasicFS, Always, TestOutput (
1525       [["write_file"; "/new"; "new file contents"; "0"];
1526        ["cat"; "/new"]], "new file contents");
1527     InitBasicFS, Always, TestOutput (
1528       [["write_file"; "/new"; "\nnew file contents\n"; "0"];
1529        ["cat"; "/new"]], "\nnew file contents\n");
1530     InitBasicFS, Always, TestOutput (
1531       [["write_file"; "/new"; "\n\n"; "0"];
1532        ["cat"; "/new"]], "\n\n");
1533     InitBasicFS, Always, TestOutput (
1534       [["write_file"; "/new"; ""; "0"];
1535        ["cat"; "/new"]], "");
1536     InitBasicFS, Always, TestOutput (
1537       [["write_file"; "/new"; "\n\n\n"; "0"];
1538        ["cat"; "/new"]], "\n\n\n");
1539     InitBasicFS, Always, TestOutput (
1540       [["write_file"; "/new"; "\n"; "0"];
1541        ["cat"; "/new"]], "\n")],
1542    "create a file",
1543    "\
1544 This call creates a file called C<path>.  The contents of the
1545 file is the string C<content> (which can contain any 8 bit data),
1546 with length C<size>.
1547
1548 As a special case, if C<size> is C<0>
1549 then the length is calculated using C<strlen> (so in this case
1550 the content cannot contain embedded ASCII NULs).
1551
1552 I<NB.> Owing to a bug, writing content containing ASCII NUL
1553 characters does I<not> work, even if the length is specified.
1554 We hope to resolve this bug in a future version.  In the meantime
1555 use C<guestfs_upload>.");
1556
1557   ("umount", (RErr, [String "pathordevice"]), 45, [FishAlias "unmount"],
1558    [InitEmpty, Always, TestOutputListOfDevices (
1559       [["part_disk"; "/dev/sda"; "mbr"];
1560        ["mkfs"; "ext2"; "/dev/sda1"];
1561        ["mount_options"; ""; "/dev/sda1"; "/"];
1562        ["mounts"]], ["/dev/sda1"]);
1563     InitEmpty, Always, TestOutputList (
1564       [["part_disk"; "/dev/sda"; "mbr"];
1565        ["mkfs"; "ext2"; "/dev/sda1"];
1566        ["mount_options"; ""; "/dev/sda1"; "/"];
1567        ["umount"; "/"];
1568        ["mounts"]], [])],
1569    "unmount a filesystem",
1570    "\
1571 This unmounts the given filesystem.  The filesystem may be
1572 specified either by its mountpoint (path) or the device which
1573 contains the filesystem.");
1574
1575   ("mounts", (RStringList "devices", []), 46, [],
1576    [InitBasicFS, Always, TestOutputListOfDevices (
1577       [["mounts"]], ["/dev/sda1"])],
1578    "show mounted filesystems",
1579    "\
1580 This returns the list of currently mounted filesystems.  It returns
1581 the list of devices (eg. C</dev/sda1>, C</dev/VG/LV>).
1582
1583 Some internal mounts are not shown.
1584
1585 See also: C<guestfs_mountpoints>");
1586
1587   ("umount_all", (RErr, []), 47, [FishAlias "unmount-all"],
1588    [InitBasicFS, Always, TestOutputList (
1589       [["umount_all"];
1590        ["mounts"]], []);
1591     (* check that umount_all can unmount nested mounts correctly: *)
1592     InitEmpty, Always, TestOutputList (
1593       [["sfdiskM"; "/dev/sda"; ",100 ,200 ,"];
1594        ["mkfs"; "ext2"; "/dev/sda1"];
1595        ["mkfs"; "ext2"; "/dev/sda2"];
1596        ["mkfs"; "ext2"; "/dev/sda3"];
1597        ["mount_options"; ""; "/dev/sda1"; "/"];
1598        ["mkdir"; "/mp1"];
1599        ["mount_options"; ""; "/dev/sda2"; "/mp1"];
1600        ["mkdir"; "/mp1/mp2"];
1601        ["mount_options"; ""; "/dev/sda3"; "/mp1/mp2"];
1602        ["mkdir"; "/mp1/mp2/mp3"];
1603        ["umount_all"];
1604        ["mounts"]], [])],
1605    "unmount all filesystems",
1606    "\
1607 This unmounts all mounted filesystems.
1608
1609 Some internal mounts are not unmounted by this call.");
1610
1611   ("lvm_remove_all", (RErr, []), 48, [DangerWillRobinson; Optional "lvm2"],
1612    [],
1613    "remove all LVM LVs, VGs and PVs",
1614    "\
1615 This command removes all LVM logical volumes, volume groups
1616 and physical volumes.");
1617
1618   ("file", (RString "description", [Dev_or_Path "path"]), 49, [],
1619    [InitISOFS, Always, TestOutput (
1620       [["file"; "/empty"]], "empty");
1621     InitISOFS, Always, TestOutput (
1622       [["file"; "/known-1"]], "ASCII text");
1623     InitISOFS, Always, TestLastFail (
1624       [["file"; "/notexists"]])],
1625    "determine file type",
1626    "\
1627 This call uses the standard L<file(1)> command to determine
1628 the type or contents of the file.  This also works on devices,
1629 for example to find out whether a partition contains a filesystem.
1630
1631 This call will also transparently look inside various types
1632 of compressed file.
1633
1634 The exact command which runs is C<file -zbsL path>.  Note in
1635 particular that the filename is not prepended to the output
1636 (the C<-b> option).");
1637
1638   ("command", (RString "output", [StringList "arguments"]), 50, [ProtocolLimitWarning],
1639    [InitBasicFS, Always, TestOutput (
1640       [["upload"; "test-command"; "/test-command"];
1641        ["chmod"; "0o755"; "/test-command"];
1642        ["command"; "/test-command 1"]], "Result1");
1643     InitBasicFS, Always, TestOutput (
1644       [["upload"; "test-command"; "/test-command"];
1645        ["chmod"; "0o755"; "/test-command"];
1646        ["command"; "/test-command 2"]], "Result2\n");
1647     InitBasicFS, Always, TestOutput (
1648       [["upload"; "test-command"; "/test-command"];
1649        ["chmod"; "0o755"; "/test-command"];
1650        ["command"; "/test-command 3"]], "\nResult3");
1651     InitBasicFS, Always, TestOutput (
1652       [["upload"; "test-command"; "/test-command"];
1653        ["chmod"; "0o755"; "/test-command"];
1654        ["command"; "/test-command 4"]], "\nResult4\n");
1655     InitBasicFS, Always, TestOutput (
1656       [["upload"; "test-command"; "/test-command"];
1657        ["chmod"; "0o755"; "/test-command"];
1658        ["command"; "/test-command 5"]], "\nResult5\n\n");
1659     InitBasicFS, Always, TestOutput (
1660       [["upload"; "test-command"; "/test-command"];
1661        ["chmod"; "0o755"; "/test-command"];
1662        ["command"; "/test-command 6"]], "\n\nResult6\n\n");
1663     InitBasicFS, Always, TestOutput (
1664       [["upload"; "test-command"; "/test-command"];
1665        ["chmod"; "0o755"; "/test-command"];
1666        ["command"; "/test-command 7"]], "");
1667     InitBasicFS, Always, TestOutput (
1668       [["upload"; "test-command"; "/test-command"];
1669        ["chmod"; "0o755"; "/test-command"];
1670        ["command"; "/test-command 8"]], "\n");
1671     InitBasicFS, Always, TestOutput (
1672       [["upload"; "test-command"; "/test-command"];
1673        ["chmod"; "0o755"; "/test-command"];
1674        ["command"; "/test-command 9"]], "\n\n");
1675     InitBasicFS, Always, TestOutput (
1676       [["upload"; "test-command"; "/test-command"];
1677        ["chmod"; "0o755"; "/test-command"];
1678        ["command"; "/test-command 10"]], "Result10-1\nResult10-2\n");
1679     InitBasicFS, Always, TestOutput (
1680       [["upload"; "test-command"; "/test-command"];
1681        ["chmod"; "0o755"; "/test-command"];
1682        ["command"; "/test-command 11"]], "Result11-1\nResult11-2");
1683     InitBasicFS, Always, TestLastFail (
1684       [["upload"; "test-command"; "/test-command"];
1685        ["chmod"; "0o755"; "/test-command"];
1686        ["command"; "/test-command"]])],
1687    "run a command from the guest filesystem",
1688    "\
1689 This call runs a command from the guest filesystem.  The
1690 filesystem must be mounted, and must contain a compatible
1691 operating system (ie. something Linux, with the same
1692 or compatible processor architecture).
1693
1694 The single parameter is an argv-style list of arguments.
1695 The first element is the name of the program to run.
1696 Subsequent elements are parameters.  The list must be
1697 non-empty (ie. must contain a program name).  Note that
1698 the command runs directly, and is I<not> invoked via
1699 the shell (see C<guestfs_sh>).
1700
1701 The return value is anything printed to I<stdout> by
1702 the command.
1703
1704 If the command returns a non-zero exit status, then
1705 this function returns an error message.  The error message
1706 string is the content of I<stderr> from the command.
1707
1708 The C<$PATH> environment variable will contain at least
1709 C</usr/bin> and C</bin>.  If you require a program from
1710 another location, you should provide the full path in the
1711 first parameter.
1712
1713 Shared libraries and data files required by the program
1714 must be available on filesystems which are mounted in the
1715 correct places.  It is the caller's responsibility to ensure
1716 all filesystems that are needed are mounted at the right
1717 locations.");
1718
1719   ("command_lines", (RStringList "lines", [StringList "arguments"]), 51, [ProtocolLimitWarning],
1720    [InitBasicFS, Always, TestOutputList (
1721       [["upload"; "test-command"; "/test-command"];
1722        ["chmod"; "0o755"; "/test-command"];
1723        ["command_lines"; "/test-command 1"]], ["Result1"]);
1724     InitBasicFS, Always, TestOutputList (
1725       [["upload"; "test-command"; "/test-command"];
1726        ["chmod"; "0o755"; "/test-command"];
1727        ["command_lines"; "/test-command 2"]], ["Result2"]);
1728     InitBasicFS, Always, TestOutputList (
1729       [["upload"; "test-command"; "/test-command"];
1730        ["chmod"; "0o755"; "/test-command"];
1731        ["command_lines"; "/test-command 3"]], ["";"Result3"]);
1732     InitBasicFS, Always, TestOutputList (
1733       [["upload"; "test-command"; "/test-command"];
1734        ["chmod"; "0o755"; "/test-command"];
1735        ["command_lines"; "/test-command 4"]], ["";"Result4"]);
1736     InitBasicFS, Always, TestOutputList (
1737       [["upload"; "test-command"; "/test-command"];
1738        ["chmod"; "0o755"; "/test-command"];
1739        ["command_lines"; "/test-command 5"]], ["";"Result5";""]);
1740     InitBasicFS, Always, TestOutputList (
1741       [["upload"; "test-command"; "/test-command"];
1742        ["chmod"; "0o755"; "/test-command"];
1743        ["command_lines"; "/test-command 6"]], ["";"";"Result6";""]);
1744     InitBasicFS, Always, TestOutputList (
1745       [["upload"; "test-command"; "/test-command"];
1746        ["chmod"; "0o755"; "/test-command"];
1747        ["command_lines"; "/test-command 7"]], []);
1748     InitBasicFS, Always, TestOutputList (
1749       [["upload"; "test-command"; "/test-command"];
1750        ["chmod"; "0o755"; "/test-command"];
1751        ["command_lines"; "/test-command 8"]], [""]);
1752     InitBasicFS, Always, TestOutputList (
1753       [["upload"; "test-command"; "/test-command"];
1754        ["chmod"; "0o755"; "/test-command"];
1755        ["command_lines"; "/test-command 9"]], ["";""]);
1756     InitBasicFS, Always, TestOutputList (
1757       [["upload"; "test-command"; "/test-command"];
1758        ["chmod"; "0o755"; "/test-command"];
1759        ["command_lines"; "/test-command 10"]], ["Result10-1";"Result10-2"]);
1760     InitBasicFS, Always, TestOutputList (
1761       [["upload"; "test-command"; "/test-command"];
1762        ["chmod"; "0o755"; "/test-command"];
1763        ["command_lines"; "/test-command 11"]], ["Result11-1";"Result11-2"])],
1764    "run a command, returning lines",
1765    "\
1766 This is the same as C<guestfs_command>, but splits the
1767 result into a list of lines.
1768
1769 See also: C<guestfs_sh_lines>");
1770
1771   ("stat", (RStruct ("statbuf", "stat"), [Pathname "path"]), 52, [],
1772    [InitISOFS, Always, TestOutputStruct (
1773       [["stat"; "/empty"]], [CompareWithInt ("size", 0)])],
1774    "get file information",
1775    "\
1776 Returns file information for the given C<path>.
1777
1778 This is the same as the C<stat(2)> system call.");
1779
1780   ("lstat", (RStruct ("statbuf", "stat"), [Pathname "path"]), 53, [],
1781    [InitISOFS, Always, TestOutputStruct (
1782       [["lstat"; "/empty"]], [CompareWithInt ("size", 0)])],
1783    "get file information for a symbolic link",
1784    "\
1785 Returns file information for the given C<path>.
1786
1787 This is the same as C<guestfs_stat> except that if C<path>
1788 is a symbolic link, then the link is stat-ed, not the file it
1789 refers to.
1790
1791 This is the same as the C<lstat(2)> system call.");
1792
1793   ("statvfs", (RStruct ("statbuf", "statvfs"), [Pathname "path"]), 54, [],
1794    [InitISOFS, Always, TestOutputStruct (
1795       [["statvfs"; "/"]], [CompareWithInt ("namemax", 255)])],
1796    "get file system statistics",
1797    "\
1798 Returns file system statistics for any mounted file system.
1799 C<path> should be a file or directory in the mounted file system
1800 (typically it is the mount point itself, but it doesn't need to be).
1801
1802 This is the same as the C<statvfs(2)> system call.");
1803
1804   ("tune2fs_l", (RHashtable "superblock", [Device "device"]), 55, [],
1805    [], (* XXX test *)
1806    "get ext2/ext3/ext4 superblock details",
1807    "\
1808 This returns the contents of the ext2, ext3 or ext4 filesystem
1809 superblock on C<device>.
1810
1811 It is the same as running C<tune2fs -l device>.  See L<tune2fs(8)>
1812 manpage for more details.  The list of fields returned isn't
1813 clearly defined, and depends on both the version of C<tune2fs>
1814 that libguestfs was built against, and the filesystem itself.");
1815
1816   ("blockdev_setro", (RErr, [Device "device"]), 56, [],
1817    [InitEmpty, Always, TestOutputTrue (
1818       [["blockdev_setro"; "/dev/sda"];
1819        ["blockdev_getro"; "/dev/sda"]])],
1820    "set block device to read-only",
1821    "\
1822 Sets the block device named C<device> to read-only.
1823
1824 This uses the L<blockdev(8)> command.");
1825
1826   ("blockdev_setrw", (RErr, [Device "device"]), 57, [],
1827    [InitEmpty, Always, TestOutputFalse (
1828       [["blockdev_setrw"; "/dev/sda"];
1829        ["blockdev_getro"; "/dev/sda"]])],
1830    "set block device to read-write",
1831    "\
1832 Sets the block device named C<device> to read-write.
1833
1834 This uses the L<blockdev(8)> command.");
1835
1836   ("blockdev_getro", (RBool "ro", [Device "device"]), 58, [],
1837    [InitEmpty, Always, TestOutputTrue (
1838       [["blockdev_setro"; "/dev/sda"];
1839        ["blockdev_getro"; "/dev/sda"]])],
1840    "is block device set to read-only",
1841    "\
1842 Returns a boolean indicating if the block device is read-only
1843 (true if read-only, false if not).
1844
1845 This uses the L<blockdev(8)> command.");
1846
1847   ("blockdev_getss", (RInt "sectorsize", [Device "device"]), 59, [],
1848    [InitEmpty, Always, TestOutputInt (
1849       [["blockdev_getss"; "/dev/sda"]], 512)],
1850    "get sectorsize of block device",
1851    "\
1852 This returns the size of sectors on a block device.
1853 Usually 512, but can be larger for modern devices.
1854
1855 (Note, this is not the size in sectors, use C<guestfs_blockdev_getsz>
1856 for that).
1857
1858 This uses the L<blockdev(8)> command.");
1859
1860   ("blockdev_getbsz", (RInt "blocksize", [Device "device"]), 60, [],
1861    [InitEmpty, Always, TestOutputInt (
1862       [["blockdev_getbsz"; "/dev/sda"]], 4096)],
1863    "get blocksize of block device",
1864    "\
1865 This returns the block size of a device.
1866
1867 (Note this is different from both I<size in blocks> and
1868 I<filesystem block size>).
1869
1870 This uses the L<blockdev(8)> command.");
1871
1872   ("blockdev_setbsz", (RErr, [Device "device"; Int "blocksize"]), 61, [],
1873    [], (* XXX test *)
1874    "set blocksize of block device",
1875    "\
1876 This sets the block size of a device.
1877
1878 (Note this is different from both I<size in blocks> and
1879 I<filesystem block size>).
1880
1881 This uses the L<blockdev(8)> command.");
1882
1883   ("blockdev_getsz", (RInt64 "sizeinsectors", [Device "device"]), 62, [],
1884    [InitEmpty, Always, TestOutputInt (
1885       [["blockdev_getsz"; "/dev/sda"]], 1024000)],
1886    "get total size of device in 512-byte sectors",
1887    "\
1888 This returns the size of the device in units of 512-byte sectors
1889 (even if the sectorsize isn't 512 bytes ... weird).
1890
1891 See also C<guestfs_blockdev_getss> for the real sector size of
1892 the device, and C<guestfs_blockdev_getsize64> for the more
1893 useful I<size in bytes>.
1894
1895 This uses the L<blockdev(8)> command.");
1896
1897   ("blockdev_getsize64", (RInt64 "sizeinbytes", [Device "device"]), 63, [],
1898    [InitEmpty, Always, TestOutputInt (
1899       [["blockdev_getsize64"; "/dev/sda"]], 524288000)],
1900    "get total size of device in bytes",
1901    "\
1902 This returns the size of the device in bytes.
1903
1904 See also C<guestfs_blockdev_getsz>.
1905
1906 This uses the L<blockdev(8)> command.");
1907
1908   ("blockdev_flushbufs", (RErr, [Device "device"]), 64, [],
1909    [InitEmpty, Always, TestRun
1910       [["blockdev_flushbufs"; "/dev/sda"]]],
1911    "flush device buffers",
1912    "\
1913 This tells the kernel to flush internal buffers associated
1914 with C<device>.
1915
1916 This uses the L<blockdev(8)> command.");
1917
1918   ("blockdev_rereadpt", (RErr, [Device "device"]), 65, [],
1919    [InitEmpty, Always, TestRun
1920       [["blockdev_rereadpt"; "/dev/sda"]]],
1921    "reread partition table",
1922    "\
1923 Reread the partition table on C<device>.
1924
1925 This uses the L<blockdev(8)> command.");
1926
1927   ("upload", (RErr, [FileIn "filename"; Dev_or_Path "remotefilename"]), 66, [],
1928    [InitBasicFS, Always, TestOutput (
1929       (* Pick a file from cwd which isn't likely to change. *)
1930       [["upload"; "../COPYING.LIB"; "/COPYING.LIB"];
1931        ["checksum"; "md5"; "/COPYING.LIB"]],
1932       Digest.to_hex (Digest.file "COPYING.LIB"))],
1933    "upload a file from the local machine",
1934    "\
1935 Upload local file C<filename> to C<remotefilename> on the
1936 filesystem.
1937
1938 C<filename> can also be a named pipe.
1939
1940 See also C<guestfs_download>.");
1941
1942   ("download", (RErr, [Dev_or_Path "remotefilename"; FileOut "filename"]), 67, [],
1943    [InitBasicFS, Always, TestOutput (
1944       (* Pick a file from cwd which isn't likely to change. *)
1945       [["upload"; "../COPYING.LIB"; "/COPYING.LIB"];
1946        ["download"; "/COPYING.LIB"; "testdownload.tmp"];
1947        ["upload"; "testdownload.tmp"; "/upload"];
1948        ["checksum"; "md5"; "/upload"]],
1949       Digest.to_hex (Digest.file "COPYING.LIB"))],
1950    "download a file to the local machine",
1951    "\
1952 Download file C<remotefilename> and save it as C<filename>
1953 on the local machine.
1954
1955 C<filename> can also be a named pipe.
1956
1957 See also C<guestfs_upload>, C<guestfs_cat>.");
1958
1959   ("checksum", (RString "checksum", [String "csumtype"; Pathname "path"]), 68, [],
1960    [InitISOFS, Always, TestOutput (
1961       [["checksum"; "crc"; "/known-3"]], "2891671662");
1962     InitISOFS, Always, TestLastFail (
1963       [["checksum"; "crc"; "/notexists"]]);
1964     InitISOFS, Always, TestOutput (
1965       [["checksum"; "md5"; "/known-3"]], "46d6ca27ee07cdc6fa99c2e138cc522c");
1966     InitISOFS, Always, TestOutput (
1967       [["checksum"; "sha1"; "/known-3"]], "b7ebccc3ee418311091c3eda0a45b83c0a770f15");
1968     InitISOFS, Always, TestOutput (
1969       [["checksum"; "sha224"; "/known-3"]], "d2cd1774b28f3659c14116be0a6dc2bb5c4b350ce9cd5defac707741");
1970     InitISOFS, Always, TestOutput (
1971       [["checksum"; "sha256"; "/known-3"]], "75bb71b90cd20cb13f86d2bea8dad63ac7194e7517c3b52b8d06ff52d3487d30");
1972     InitISOFS, Always, TestOutput (
1973       [["checksum"; "sha384"; "/known-3"]], "5fa7883430f357b5d7b7271d3a1d2872b51d73cba72731de6863d3dea55f30646af2799bef44d5ea776a5ec7941ac640");
1974     InitISOFS, Always, TestOutput (
1975       [["checksum"; "sha512"; "/known-3"]], "2794062c328c6b216dca90443b7f7134c5f40e56bd0ed7853123275a09982a6f992e6ca682f9d2fba34a4c5e870d8fe077694ff831e3032a004ee077e00603f6")],
1976    "compute MD5, SHAx or CRC checksum of file",
1977    "\
1978 This call computes the MD5, SHAx or CRC checksum of the
1979 file named C<path>.
1980
1981 The type of checksum to compute is given by the C<csumtype>
1982 parameter which must have one of the following values:
1983
1984 =over 4
1985
1986 =item C<crc>
1987
1988 Compute the cyclic redundancy check (CRC) specified by POSIX
1989 for the C<cksum> command.
1990
1991 =item C<md5>
1992
1993 Compute the MD5 hash (using the C<md5sum> program).
1994
1995 =item C<sha1>
1996
1997 Compute the SHA1 hash (using the C<sha1sum> program).
1998
1999 =item C<sha224>
2000
2001 Compute the SHA224 hash (using the C<sha224sum> program).
2002
2003 =item C<sha256>
2004
2005 Compute the SHA256 hash (using the C<sha256sum> program).
2006
2007 =item C<sha384>
2008
2009 Compute the SHA384 hash (using the C<sha384sum> program).
2010
2011 =item C<sha512>
2012
2013 Compute the SHA512 hash (using the C<sha512sum> program).
2014
2015 =back
2016
2017 The checksum is returned as a printable string.");
2018
2019   ("tar_in", (RErr, [FileIn "tarfile"; String "directory"]), 69, [],
2020    [InitBasicFS, Always, TestOutput (
2021       [["tar_in"; "../images/helloworld.tar"; "/"];
2022        ["cat"; "/hello"]], "hello\n")],
2023    "unpack tarfile to directory",
2024    "\
2025 This command uploads and unpacks local file C<tarfile> (an
2026 I<uncompressed> tar file) into C<directory>.
2027
2028 To upload a compressed tarball, use C<guestfs_tgz_in>.");
2029
2030   ("tar_out", (RErr, [String "directory"; FileOut "tarfile"]), 70, [],
2031    [],
2032    "pack directory into tarfile",
2033    "\
2034 This command packs the contents of C<directory> and downloads
2035 it to local file C<tarfile>.
2036
2037 To download a compressed tarball, use C<guestfs_tgz_out>.");
2038
2039   ("tgz_in", (RErr, [FileIn "tarball"; String "directory"]), 71, [],
2040    [InitBasicFS, Always, TestOutput (
2041       [["tgz_in"; "../images/helloworld.tar.gz"; "/"];
2042        ["cat"; "/hello"]], "hello\n")],
2043    "unpack compressed tarball to directory",
2044    "\
2045 This command uploads and unpacks local file C<tarball> (a
2046 I<gzip compressed> tar file) into C<directory>.
2047
2048 To upload an uncompressed tarball, use C<guestfs_tar_in>.");
2049
2050   ("tgz_out", (RErr, [Pathname "directory"; FileOut "tarball"]), 72, [],
2051    [],
2052    "pack directory into compressed tarball",
2053    "\
2054 This command packs the contents of C<directory> and downloads
2055 it to local file C<tarball>.
2056
2057 To download an uncompressed tarball, use C<guestfs_tar_out>.");
2058
2059   ("mount_ro", (RErr, [Device "device"; String "mountpoint"]), 73, [],
2060    [InitBasicFS, Always, TestLastFail (
2061       [["umount"; "/"];
2062        ["mount_ro"; "/dev/sda1"; "/"];
2063        ["touch"; "/new"]]);
2064     InitBasicFS, Always, TestOutput (
2065       [["write_file"; "/new"; "data"; "0"];
2066        ["umount"; "/"];
2067        ["mount_ro"; "/dev/sda1"; "/"];
2068        ["cat"; "/new"]], "data")],
2069    "mount a guest disk, read-only",
2070    "\
2071 This is the same as the C<guestfs_mount> command, but it
2072 mounts the filesystem with the read-only (I<-o ro>) flag.");
2073
2074   ("mount_options", (RErr, [String "options"; Device "device"; String "mountpoint"]), 74, [],
2075    [],
2076    "mount a guest disk with mount options",
2077    "\
2078 This is the same as the C<guestfs_mount> command, but it
2079 allows you to set the mount options as for the
2080 L<mount(8)> I<-o> flag.");
2081
2082   ("mount_vfs", (RErr, [String "options"; String "vfstype"; Device "device"; String "mountpoint"]), 75, [],
2083    [],
2084    "mount a guest disk with mount options and vfstype",
2085    "\
2086 This is the same as the C<guestfs_mount> command, but it
2087 allows you to set both the mount options and the vfstype
2088 as for the L<mount(8)> I<-o> and I<-t> flags.");
2089
2090   ("debug", (RString "result", [String "subcmd"; StringList "extraargs"]), 76, [],
2091    [],
2092    "debugging and internals",
2093    "\
2094 The C<guestfs_debug> command exposes some internals of
2095 C<guestfsd> (the guestfs daemon) that runs inside the
2096 qemu subprocess.
2097
2098 There is no comprehensive help for this command.  You have
2099 to look at the file C<daemon/debug.c> in the libguestfs source
2100 to find out what you can do.");
2101
2102   ("lvremove", (RErr, [Device "device"]), 77, [Optional "lvm2"],
2103    [InitEmpty, Always, TestOutputList (
2104       [["part_disk"; "/dev/sda"; "mbr"];
2105        ["pvcreate"; "/dev/sda1"];
2106        ["vgcreate"; "VG"; "/dev/sda1"];
2107        ["lvcreate"; "LV1"; "VG"; "50"];
2108        ["lvcreate"; "LV2"; "VG"; "50"];
2109        ["lvremove"; "/dev/VG/LV1"];
2110        ["lvs"]], ["/dev/VG/LV2"]);
2111     InitEmpty, Always, TestOutputList (
2112       [["part_disk"; "/dev/sda"; "mbr"];
2113        ["pvcreate"; "/dev/sda1"];
2114        ["vgcreate"; "VG"; "/dev/sda1"];
2115        ["lvcreate"; "LV1"; "VG"; "50"];
2116        ["lvcreate"; "LV2"; "VG"; "50"];
2117        ["lvremove"; "/dev/VG"];
2118        ["lvs"]], []);
2119     InitEmpty, Always, TestOutputList (
2120       [["part_disk"; "/dev/sda"; "mbr"];
2121        ["pvcreate"; "/dev/sda1"];
2122        ["vgcreate"; "VG"; "/dev/sda1"];
2123        ["lvcreate"; "LV1"; "VG"; "50"];
2124        ["lvcreate"; "LV2"; "VG"; "50"];
2125        ["lvremove"; "/dev/VG"];
2126        ["vgs"]], ["VG"])],
2127    "remove an LVM logical volume",
2128    "\
2129 Remove an LVM logical volume C<device>, where C<device> is
2130 the path to the LV, such as C</dev/VG/LV>.
2131
2132 You can also remove all LVs in a volume group by specifying
2133 the VG name, C</dev/VG>.");
2134
2135   ("vgremove", (RErr, [String "vgname"]), 78, [Optional "lvm2"],
2136    [InitEmpty, Always, TestOutputList (
2137       [["part_disk"; "/dev/sda"; "mbr"];
2138        ["pvcreate"; "/dev/sda1"];
2139        ["vgcreate"; "VG"; "/dev/sda1"];
2140        ["lvcreate"; "LV1"; "VG"; "50"];
2141        ["lvcreate"; "LV2"; "VG"; "50"];
2142        ["vgremove"; "VG"];
2143        ["lvs"]], []);
2144     InitEmpty, Always, TestOutputList (
2145       [["part_disk"; "/dev/sda"; "mbr"];
2146        ["pvcreate"; "/dev/sda1"];
2147        ["vgcreate"; "VG"; "/dev/sda1"];
2148        ["lvcreate"; "LV1"; "VG"; "50"];
2149        ["lvcreate"; "LV2"; "VG"; "50"];
2150        ["vgremove"; "VG"];
2151        ["vgs"]], [])],
2152    "remove an LVM volume group",
2153    "\
2154 Remove an LVM volume group C<vgname>, (for example C<VG>).
2155
2156 This also forcibly removes all logical volumes in the volume
2157 group (if any).");
2158
2159   ("pvremove", (RErr, [Device "device"]), 79, [Optional "lvm2"],
2160    [InitEmpty, Always, TestOutputListOfDevices (
2161       [["part_disk"; "/dev/sda"; "mbr"];
2162        ["pvcreate"; "/dev/sda1"];
2163        ["vgcreate"; "VG"; "/dev/sda1"];
2164        ["lvcreate"; "LV1"; "VG"; "50"];
2165        ["lvcreate"; "LV2"; "VG"; "50"];
2166        ["vgremove"; "VG"];
2167        ["pvremove"; "/dev/sda1"];
2168        ["lvs"]], []);
2169     InitEmpty, Always, TestOutputListOfDevices (
2170       [["part_disk"; "/dev/sda"; "mbr"];
2171        ["pvcreate"; "/dev/sda1"];
2172        ["vgcreate"; "VG"; "/dev/sda1"];
2173        ["lvcreate"; "LV1"; "VG"; "50"];
2174        ["lvcreate"; "LV2"; "VG"; "50"];
2175        ["vgremove"; "VG"];
2176        ["pvremove"; "/dev/sda1"];
2177        ["vgs"]], []);
2178     InitEmpty, Always, TestOutputListOfDevices (
2179       [["part_disk"; "/dev/sda"; "mbr"];
2180        ["pvcreate"; "/dev/sda1"];
2181        ["vgcreate"; "VG"; "/dev/sda1"];
2182        ["lvcreate"; "LV1"; "VG"; "50"];
2183        ["lvcreate"; "LV2"; "VG"; "50"];
2184        ["vgremove"; "VG"];
2185        ["pvremove"; "/dev/sda1"];
2186        ["pvs"]], [])],
2187    "remove an LVM physical volume",
2188    "\
2189 This wipes a physical volume C<device> so that LVM will no longer
2190 recognise it.
2191
2192 The implementation uses the C<pvremove> command which refuses to
2193 wipe physical volumes that contain any volume groups, so you have
2194 to remove those first.");
2195
2196   ("set_e2label", (RErr, [Device "device"; String "label"]), 80, [],
2197    [InitBasicFS, Always, TestOutput (
2198       [["set_e2label"; "/dev/sda1"; "testlabel"];
2199        ["get_e2label"; "/dev/sda1"]], "testlabel")],
2200    "set the ext2/3/4 filesystem label",
2201    "\
2202 This sets the ext2/3/4 filesystem label of the filesystem on
2203 C<device> to C<label>.  Filesystem labels are limited to
2204 16 characters.
2205
2206 You can use either C<guestfs_tune2fs_l> or C<guestfs_get_e2label>
2207 to return the existing label on a filesystem.");
2208
2209   ("get_e2label", (RString "label", [Device "device"]), 81, [],
2210    [],
2211    "get the ext2/3/4 filesystem label",
2212    "\
2213 This returns the ext2/3/4 filesystem label of the filesystem on
2214 C<device>.");
2215
2216   ("set_e2uuid", (RErr, [Device "device"; String "uuid"]), 82, [],
2217    (let uuid = uuidgen () in
2218     [InitBasicFS, Always, TestOutput (
2219        [["set_e2uuid"; "/dev/sda1"; uuid];
2220         ["get_e2uuid"; "/dev/sda1"]], uuid);
2221      InitBasicFS, Always, TestOutput (
2222        [["set_e2uuid"; "/dev/sda1"; "clear"];
2223         ["get_e2uuid"; "/dev/sda1"]], "");
2224      (* We can't predict what UUIDs will be, so just check the commands run. *)
2225      InitBasicFS, Always, TestRun (
2226        [["set_e2uuid"; "/dev/sda1"; "random"]]);
2227      InitBasicFS, Always, TestRun (
2228        [["set_e2uuid"; "/dev/sda1"; "time"]])]),
2229    "set the ext2/3/4 filesystem UUID",
2230    "\
2231 This sets the ext2/3/4 filesystem UUID of the filesystem on
2232 C<device> to C<uuid>.  The format of the UUID and alternatives
2233 such as C<clear>, C<random> and C<time> are described in the
2234 L<tune2fs(8)> manpage.
2235
2236 You can use either C<guestfs_tune2fs_l> or C<guestfs_get_e2uuid>
2237 to return the existing UUID of a filesystem.");
2238
2239   ("get_e2uuid", (RString "uuid", [Device "device"]), 83, [],
2240    [],
2241    "get the ext2/3/4 filesystem UUID",
2242    "\
2243 This returns the ext2/3/4 filesystem UUID of the filesystem on
2244 C<device>.");
2245
2246   ("fsck", (RInt "status", [String "fstype"; Device "device"]), 84, [],
2247    [InitBasicFS, Always, TestOutputInt (
2248       [["umount"; "/dev/sda1"];
2249        ["fsck"; "ext2"; "/dev/sda1"]], 0);
2250     InitBasicFS, Always, TestOutputInt (
2251       [["umount"; "/dev/sda1"];
2252        ["zero"; "/dev/sda1"];
2253        ["fsck"; "ext2"; "/dev/sda1"]], 8)],
2254    "run the filesystem checker",
2255    "\
2256 This runs the filesystem checker (fsck) on C<device> which
2257 should have filesystem type C<fstype>.
2258
2259 The returned integer is the status.  See L<fsck(8)> for the
2260 list of status codes from C<fsck>.
2261
2262 Notes:
2263
2264 =over 4
2265
2266 =item *
2267
2268 Multiple status codes can be summed together.
2269
2270 =item *
2271
2272 A non-zero return code can mean \"success\", for example if
2273 errors have been corrected on the filesystem.
2274
2275 =item *
2276
2277 Checking or repairing NTFS volumes is not supported
2278 (by linux-ntfs).
2279
2280 =back
2281
2282 This command is entirely equivalent to running C<fsck -a -t fstype device>.");
2283
2284   ("zero", (RErr, [Device "device"]), 85, [],
2285    [InitBasicFS, Always, TestOutput (
2286       [["umount"; "/dev/sda1"];
2287        ["zero"; "/dev/sda1"];
2288        ["file"; "/dev/sda1"]], "data")],
2289    "write zeroes to the device",
2290    "\
2291 This command writes zeroes over the first few blocks of C<device>.
2292
2293 How many blocks are zeroed isn't specified (but it's I<not> enough
2294 to securely wipe the device).  It should be sufficient to remove
2295 any partition tables, filesystem superblocks and so on.
2296
2297 See also: C<guestfs_scrub_device>.");
2298
2299   ("grub_install", (RErr, [Pathname "root"; Device "device"]), 86, [],
2300    (* Test disabled because grub-install incompatible with virtio-blk driver.
2301     * See also: https://bugzilla.redhat.com/show_bug.cgi?id=479760
2302     *)
2303    [InitBasicFS, Disabled, TestOutputTrue (
2304       [["grub_install"; "/"; "/dev/sda1"];
2305        ["is_dir"; "/boot"]])],
2306    "install GRUB",
2307    "\
2308 This command installs GRUB (the Grand Unified Bootloader) on
2309 C<device>, with the root directory being C<root>.");
2310
2311   ("cp", (RErr, [Pathname "src"; Pathname "dest"]), 87, [],
2312    [InitBasicFS, Always, TestOutput (
2313       [["write_file"; "/old"; "file content"; "0"];
2314        ["cp"; "/old"; "/new"];
2315        ["cat"; "/new"]], "file content");
2316     InitBasicFS, Always, TestOutputTrue (
2317       [["write_file"; "/old"; "file content"; "0"];
2318        ["cp"; "/old"; "/new"];
2319        ["is_file"; "/old"]]);
2320     InitBasicFS, Always, TestOutput (
2321       [["write_file"; "/old"; "file content"; "0"];
2322        ["mkdir"; "/dir"];
2323        ["cp"; "/old"; "/dir/new"];
2324        ["cat"; "/dir/new"]], "file content")],
2325    "copy a file",
2326    "\
2327 This copies a file from C<src> to C<dest> where C<dest> is
2328 either a destination filename or destination directory.");
2329
2330   ("cp_a", (RErr, [Pathname "src"; Pathname "dest"]), 88, [],
2331    [InitBasicFS, Always, TestOutput (
2332       [["mkdir"; "/olddir"];
2333        ["mkdir"; "/newdir"];
2334        ["write_file"; "/olddir/file"; "file content"; "0"];
2335        ["cp_a"; "/olddir"; "/newdir"];
2336        ["cat"; "/newdir/olddir/file"]], "file content")],
2337    "copy a file or directory recursively",
2338    "\
2339 This copies a file or directory from C<src> to C<dest>
2340 recursively using the C<cp -a> command.");
2341
2342   ("mv", (RErr, [Pathname "src"; Pathname "dest"]), 89, [],
2343    [InitBasicFS, Always, TestOutput (
2344       [["write_file"; "/old"; "file content"; "0"];
2345        ["mv"; "/old"; "/new"];
2346        ["cat"; "/new"]], "file content");
2347     InitBasicFS, Always, TestOutputFalse (
2348       [["write_file"; "/old"; "file content"; "0"];
2349        ["mv"; "/old"; "/new"];
2350        ["is_file"; "/old"]])],
2351    "move a file",
2352    "\
2353 This moves a file from C<src> to C<dest> where C<dest> is
2354 either a destination filename or destination directory.");
2355
2356   ("drop_caches", (RErr, [Int "whattodrop"]), 90, [],
2357    [InitEmpty, Always, TestRun (
2358       [["drop_caches"; "3"]])],
2359    "drop kernel page cache, dentries and inodes",
2360    "\
2361 This instructs the guest kernel to drop its page cache,
2362 and/or dentries and inode caches.  The parameter C<whattodrop>
2363 tells the kernel what precisely to drop, see
2364 L<http://linux-mm.org/Drop_Caches>
2365
2366 Setting C<whattodrop> to 3 should drop everything.
2367
2368 This automatically calls L<sync(2)> before the operation,
2369 so that the maximum guest memory is freed.");
2370
2371   ("dmesg", (RString "kmsgs", []), 91, [],
2372    [InitEmpty, Always, TestRun (
2373       [["dmesg"]])],
2374    "return kernel messages",
2375    "\
2376 This returns the kernel messages (C<dmesg> output) from
2377 the guest kernel.  This is sometimes useful for extended
2378 debugging of problems.
2379
2380 Another way to get the same information is to enable
2381 verbose messages with C<guestfs_set_verbose> or by setting
2382 the environment variable C<LIBGUESTFS_DEBUG=1> before
2383 running the program.");
2384
2385   ("ping_daemon", (RErr, []), 92, [],
2386    [InitEmpty, Always, TestRun (
2387       [["ping_daemon"]])],
2388    "ping the guest daemon",
2389    "\
2390 This is a test probe into the guestfs daemon running inside
2391 the qemu subprocess.  Calling this function checks that the
2392 daemon responds to the ping message, without affecting the daemon
2393 or attached block device(s) in any other way.");
2394
2395   ("equal", (RBool "equality", [Pathname "file1"; Pathname "file2"]), 93, [],
2396    [InitBasicFS, Always, TestOutputTrue (
2397       [["write_file"; "/file1"; "contents of a file"; "0"];
2398        ["cp"; "/file1"; "/file2"];
2399        ["equal"; "/file1"; "/file2"]]);
2400     InitBasicFS, Always, TestOutputFalse (
2401       [["write_file"; "/file1"; "contents of a file"; "0"];
2402        ["write_file"; "/file2"; "contents of another file"; "0"];
2403        ["equal"; "/file1"; "/file2"]]);
2404     InitBasicFS, Always, TestLastFail (
2405       [["equal"; "/file1"; "/file2"]])],
2406    "test if two files have equal contents",
2407    "\
2408 This compares the two files C<file1> and C<file2> and returns
2409 true if their content is exactly equal, or false otherwise.
2410
2411 The external L<cmp(1)> program is used for the comparison.");
2412
2413   ("strings", (RStringList "stringsout", [Pathname "path"]), 94, [ProtocolLimitWarning],
2414    [InitISOFS, Always, TestOutputList (
2415       [["strings"; "/known-5"]], ["abcdefghi"; "jklmnopqr"]);
2416     InitISOFS, Always, TestOutputList (
2417       [["strings"; "/empty"]], [])],
2418    "print the printable strings in a file",
2419    "\
2420 This runs the L<strings(1)> command on a file and returns
2421 the list of printable strings found.");
2422
2423   ("strings_e", (RStringList "stringsout", [String "encoding"; Pathname "path"]), 95, [ProtocolLimitWarning],
2424    [InitISOFS, Always, TestOutputList (
2425       [["strings_e"; "b"; "/known-5"]], []);
2426     InitBasicFS, Disabled, TestOutputList (
2427       [["write_file"; "/new"; "\000h\000e\000l\000l\000o\000\n\000w\000o\000r\000l\000d\000\n"; "24"];
2428        ["strings_e"; "b"; "/new"]], ["hello"; "world"])],
2429    "print the printable strings in a file",
2430    "\
2431 This is like the C<guestfs_strings> command, but allows you to
2432 specify the encoding.
2433
2434 See the L<strings(1)> manpage for the full list of encodings.
2435
2436 Commonly useful encodings are C<l> (lower case L) which will
2437 show strings inside Windows/x86 files.
2438
2439 The returned strings are transcoded to UTF-8.");
2440
2441   ("hexdump", (RString "dump", [Pathname "path"]), 96, [ProtocolLimitWarning],
2442    [InitISOFS, Always, TestOutput (
2443       [["hexdump"; "/known-4"]], "00000000  61 62 63 0a 64 65 66 0a  67 68 69                 |abc.def.ghi|\n0000000b\n");
2444     (* Test for RHBZ#501888c2 regression which caused large hexdump
2445      * commands to segfault.
2446      *)
2447     InitISOFS, Always, TestRun (
2448       [["hexdump"; "/100krandom"]])],
2449    "dump a file in hexadecimal",
2450    "\
2451 This runs C<hexdump -C> on the given C<path>.  The result is
2452 the human-readable, canonical hex dump of the file.");
2453
2454   ("zerofree", (RErr, [Device "device"]), 97, [Optional "zerofree"],
2455    [InitNone, Always, TestOutput (
2456       [["part_disk"; "/dev/sda"; "mbr"];
2457        ["mkfs"; "ext3"; "/dev/sda1"];
2458        ["mount_options"; ""; "/dev/sda1"; "/"];
2459        ["write_file"; "/new"; "test file"; "0"];
2460        ["umount"; "/dev/sda1"];
2461        ["zerofree"; "/dev/sda1"];
2462        ["mount_options"; ""; "/dev/sda1"; "/"];
2463        ["cat"; "/new"]], "test file")],
2464    "zero unused inodes and disk blocks on ext2/3 filesystem",
2465    "\
2466 This runs the I<zerofree> program on C<device>.  This program
2467 claims to zero unused inodes and disk blocks on an ext2/3
2468 filesystem, thus making it possible to compress the filesystem
2469 more effectively.
2470
2471 You should B<not> run this program if the filesystem is
2472 mounted.
2473
2474 It is possible that using this program can damage the filesystem
2475 or data on the filesystem.");
2476
2477   ("pvresize", (RErr, [Device "device"]), 98, [Optional "lvm2"],
2478    [],
2479    "resize an LVM physical volume",
2480    "\
2481 This resizes (expands or shrinks) an existing LVM physical
2482 volume to match the new size of the underlying device.");
2483
2484   ("sfdisk_N", (RErr, [Device "device"; Int "partnum";
2485                        Int "cyls"; Int "heads"; Int "sectors";
2486                        String "line"]), 99, [DangerWillRobinson],
2487    [],
2488    "modify a single partition on a block device",
2489    "\
2490 This runs L<sfdisk(8)> option to modify just the single
2491 partition C<n> (note: C<n> counts from 1).
2492
2493 For other parameters, see C<guestfs_sfdisk>.  You should usually
2494 pass C<0> for the cyls/heads/sectors parameters.
2495
2496 See also: C<guestfs_part_add>");
2497
2498   ("sfdisk_l", (RString "partitions", [Device "device"]), 100, [],
2499    [],
2500    "display the partition table",
2501    "\
2502 This displays the partition table on C<device>, in the
2503 human-readable output of the L<sfdisk(8)> command.  It is
2504 not intended to be parsed.
2505
2506 See also: C<guestfs_part_list>");
2507
2508   ("sfdisk_kernel_geometry", (RString "partitions", [Device "device"]), 101, [],
2509    [],
2510    "display the kernel geometry",
2511    "\
2512 This displays the kernel's idea of the geometry of C<device>.
2513
2514 The result is in human-readable format, and not designed to
2515 be parsed.");
2516
2517   ("sfdisk_disk_geometry", (RString "partitions", [Device "device"]), 102, [],
2518    [],
2519    "display the disk geometry from the partition table",
2520    "\
2521 This displays the disk geometry of C<device> read from the
2522 partition table.  Especially in the case where the underlying
2523 block device has been resized, this can be different from the
2524 kernel's idea of the geometry (see C<guestfs_sfdisk_kernel_geometry>).
2525
2526 The result is in human-readable format, and not designed to
2527 be parsed.");
2528
2529   ("vg_activate_all", (RErr, [Bool "activate"]), 103, [Optional "lvm2"],
2530    [],
2531    "activate or deactivate all volume groups",
2532    "\
2533 This command activates or (if C<activate> is false) deactivates
2534 all logical volumes in all volume groups.
2535 If activated, then they are made known to the
2536 kernel, ie. they appear as C</dev/mapper> devices.  If deactivated,
2537 then those devices disappear.
2538
2539 This command is the same as running C<vgchange -a y|n>");
2540
2541   ("vg_activate", (RErr, [Bool "activate"; StringList "volgroups"]), 104, [Optional "lvm2"],
2542    [],
2543    "activate or deactivate some volume groups",
2544    "\
2545 This command activates or (if C<activate> is false) deactivates
2546 all logical volumes in the listed volume groups C<volgroups>.
2547 If activated, then they are made known to the
2548 kernel, ie. they appear as C</dev/mapper> devices.  If deactivated,
2549 then those devices disappear.
2550
2551 This command is the same as running C<vgchange -a y|n volgroups...>
2552
2553 Note that if C<volgroups> is an empty list then B<all> volume groups
2554 are activated or deactivated.");
2555
2556   ("lvresize", (RErr, [Device "device"; Int "mbytes"]), 105, [Optional "lvm2"],
2557    [InitNone, Always, TestOutput (
2558       [["part_disk"; "/dev/sda"; "mbr"];
2559        ["pvcreate"; "/dev/sda1"];
2560        ["vgcreate"; "VG"; "/dev/sda1"];
2561        ["lvcreate"; "LV"; "VG"; "10"];
2562        ["mkfs"; "ext2"; "/dev/VG/LV"];
2563        ["mount_options"; ""; "/dev/VG/LV"; "/"];
2564        ["write_file"; "/new"; "test content"; "0"];
2565        ["umount"; "/"];
2566        ["lvresize"; "/dev/VG/LV"; "20"];
2567        ["e2fsck_f"; "/dev/VG/LV"];
2568        ["resize2fs"; "/dev/VG/LV"];
2569        ["mount_options"; ""; "/dev/VG/LV"; "/"];
2570        ["cat"; "/new"]], "test content")],
2571    "resize an LVM logical volume",
2572    "\
2573 This resizes (expands or shrinks) an existing LVM logical
2574 volume to C<mbytes>.  When reducing, data in the reduced part
2575 is lost.");
2576
2577   ("resize2fs", (RErr, [Device "device"]), 106, [],
2578    [], (* lvresize tests this *)
2579    "resize an ext2/ext3 filesystem",
2580    "\
2581 This resizes an ext2 or ext3 filesystem to match the size of
2582 the underlying device.
2583
2584 I<Note:> It is sometimes required that you run C<guestfs_e2fsck_f>
2585 on the C<device> before calling this command.  For unknown reasons
2586 C<resize2fs> sometimes gives an error about this and sometimes not.
2587 In any case, it is always safe to call C<guestfs_e2fsck_f> before
2588 calling this function.");
2589
2590   ("find", (RStringList "names", [Pathname "directory"]), 107, [ProtocolLimitWarning],
2591    [InitBasicFS, Always, TestOutputList (
2592       [["find"; "/"]], ["lost+found"]);
2593     InitBasicFS, Always, TestOutputList (
2594       [["touch"; "/a"];
2595        ["mkdir"; "/b"];
2596        ["touch"; "/b/c"];
2597        ["find"; "/"]], ["a"; "b"; "b/c"; "lost+found"]);
2598     InitBasicFS, Always, TestOutputList (
2599       [["mkdir_p"; "/a/b/c"];
2600        ["touch"; "/a/b/c/d"];
2601        ["find"; "/a/b/"]], ["c"; "c/d"])],
2602    "find all files and directories",
2603    "\
2604 This command lists out all files and directories, recursively,
2605 starting at C<directory>.  It is essentially equivalent to
2606 running the shell command C<find directory -print> but some
2607 post-processing happens on the output, described below.
2608
2609 This returns a list of strings I<without any prefix>.  Thus
2610 if the directory structure was:
2611
2612  /tmp/a
2613  /tmp/b
2614  /tmp/c/d
2615
2616 then the returned list from C<guestfs_find> C</tmp> would be
2617 4 elements:
2618
2619  a
2620  b
2621  c
2622  c/d
2623
2624 If C<directory> is not a directory, then this command returns
2625 an error.
2626
2627 The returned list is sorted.
2628
2629 See also C<guestfs_find0>.");
2630
2631   ("e2fsck_f", (RErr, [Device "device"]), 108, [],
2632    [], (* lvresize tests this *)
2633    "check an ext2/ext3 filesystem",
2634    "\
2635 This runs C<e2fsck -p -f device>, ie. runs the ext2/ext3
2636 filesystem checker on C<device>, noninteractively (C<-p>),
2637 even if the filesystem appears to be clean (C<-f>).
2638
2639 This command is only needed because of C<guestfs_resize2fs>
2640 (q.v.).  Normally you should use C<guestfs_fsck>.");
2641
2642   ("sleep", (RErr, [Int "secs"]), 109, [],
2643    [InitNone, Always, TestRun (
2644       [["sleep"; "1"]])],
2645    "sleep for some seconds",
2646    "\
2647 Sleep for C<secs> seconds.");
2648
2649   ("ntfs_3g_probe", (RInt "status", [Bool "rw"; Device "device"]), 110, [Optional "ntfs3g"],
2650    [InitNone, Always, TestOutputInt (
2651       [["part_disk"; "/dev/sda"; "mbr"];
2652        ["mkfs"; "ntfs"; "/dev/sda1"];
2653        ["ntfs_3g_probe"; "true"; "/dev/sda1"]], 0);
2654     InitNone, Always, TestOutputInt (
2655       [["part_disk"; "/dev/sda"; "mbr"];
2656        ["mkfs"; "ext2"; "/dev/sda1"];
2657        ["ntfs_3g_probe"; "true"; "/dev/sda1"]], 12)],
2658    "probe NTFS volume",
2659    "\
2660 This command runs the L<ntfs-3g.probe(8)> command which probes
2661 an NTFS C<device> for mountability.  (Not all NTFS volumes can
2662 be mounted read-write, and some cannot be mounted at all).
2663
2664 C<rw> is a boolean flag.  Set it to true if you want to test
2665 if the volume can be mounted read-write.  Set it to false if
2666 you want to test if the volume can be mounted read-only.
2667
2668 The return value is an integer which C<0> if the operation
2669 would succeed, or some non-zero value documented in the
2670 L<ntfs-3g.probe(8)> manual page.");
2671
2672   ("sh", (RString "output", [String "command"]), 111, [],
2673    [], (* XXX needs tests *)
2674    "run a command via the shell",
2675    "\
2676 This call runs a command from the guest filesystem via the
2677 guest's C</bin/sh>.
2678
2679 This is like C<guestfs_command>, but passes the command to:
2680
2681  /bin/sh -c \"command\"
2682
2683 Depending on the guest's shell, this usually results in
2684 wildcards being expanded, shell expressions being interpolated
2685 and so on.
2686
2687 All the provisos about C<guestfs_command> apply to this call.");
2688
2689   ("sh_lines", (RStringList "lines", [String "command"]), 112, [],
2690    [], (* XXX needs tests *)
2691    "run a command via the shell returning lines",
2692    "\
2693 This is the same as C<guestfs_sh>, but splits the result
2694 into a list of lines.
2695
2696 See also: C<guestfs_command_lines>");
2697
2698   ("glob_expand", (RStringList "paths", [Pathname "pattern"]), 113, [],
2699    (* Use Pathname here, and hence ABS_PATH (pattern,... in generated
2700     * code in stubs.c, since all valid glob patterns must start with "/".
2701     * There is no concept of "cwd" in libguestfs, hence no "."-relative names.
2702     *)
2703    [InitBasicFS, Always, TestOutputList (
2704       [["mkdir_p"; "/a/b/c"];
2705        ["touch"; "/a/b/c/d"];
2706        ["touch"; "/a/b/c/e"];
2707        ["glob_expand"; "/a/b/c/*"]], ["/a/b/c/d"; "/a/b/c/e"]);
2708     InitBasicFS, Always, TestOutputList (
2709       [["mkdir_p"; "/a/b/c"];
2710        ["touch"; "/a/b/c/d"];
2711        ["touch"; "/a/b/c/e"];
2712        ["glob_expand"; "/a/*/c/*"]], ["/a/b/c/d"; "/a/b/c/e"]);
2713     InitBasicFS, Always, TestOutputList (
2714       [["mkdir_p"; "/a/b/c"];
2715        ["touch"; "/a/b/c/d"];
2716        ["touch"; "/a/b/c/e"];
2717        ["glob_expand"; "/a/*/x/*"]], [])],
2718    "expand a wildcard path",
2719    "\
2720 This command searches for all the pathnames matching
2721 C<pattern> according to the wildcard expansion rules
2722 used by the shell.
2723
2724 If no paths match, then this returns an empty list
2725 (note: not an error).
2726
2727 It is just a wrapper around the C L<glob(3)> function
2728 with flags C<GLOB_MARK|GLOB_BRACE>.
2729 See that manual page for more details.");
2730
2731   ("scrub_device", (RErr, [Device "device"]), 114, [DangerWillRobinson; Optional "scrub"],
2732    [InitNone, Always, TestRun ( (* use /dev/sdc because it's smaller *)
2733       [["scrub_device"; "/dev/sdc"]])],
2734    "scrub (securely wipe) a device",
2735    "\
2736 This command writes patterns over C<device> to make data retrieval
2737 more difficult.
2738
2739 It is an interface to the L<scrub(1)> program.  See that
2740 manual page for more details.");
2741
2742   ("scrub_file", (RErr, [Pathname "file"]), 115, [Optional "scrub"],
2743    [InitBasicFS, Always, TestRun (
2744       [["write_file"; "/file"; "content"; "0"];
2745        ["scrub_file"; "/file"]])],
2746    "scrub (securely wipe) a file",
2747    "\
2748 This command writes patterns over a file to make data retrieval
2749 more difficult.
2750
2751 The file is I<removed> after scrubbing.
2752
2753 It is an interface to the L<scrub(1)> program.  See that
2754 manual page for more details.");
2755
2756   ("scrub_freespace", (RErr, [Pathname "dir"]), 116, [Optional "scrub"],
2757    [], (* XXX needs testing *)
2758    "scrub (securely wipe) free space",
2759    "\
2760 This command creates the directory C<dir> and then fills it
2761 with files until the filesystem is full, and scrubs the files
2762 as for C<guestfs_scrub_file>, and deletes them.
2763 The intention is to scrub any free space on the partition
2764 containing C<dir>.
2765
2766 It is an interface to the L<scrub(1)> program.  See that
2767 manual page for more details.");
2768
2769   ("mkdtemp", (RString "dir", [Pathname "template"]), 117, [],
2770    [InitBasicFS, Always, TestRun (
2771       [["mkdir"; "/tmp"];
2772        ["mkdtemp"; "/tmp/tmpXXXXXX"]])],
2773    "create a temporary directory",
2774    "\
2775 This command creates a temporary directory.  The
2776 C<template> parameter should be a full pathname for the
2777 temporary directory name with the final six characters being
2778 \"XXXXXX\".
2779
2780 For example: \"/tmp/myprogXXXXXX\" or \"/Temp/myprogXXXXXX\",
2781 the second one being suitable for Windows filesystems.
2782
2783 The name of the temporary directory that was created
2784 is returned.
2785
2786 The temporary directory is created with mode 0700
2787 and is owned by root.
2788
2789 The caller is responsible for deleting the temporary
2790 directory and its contents after use.
2791
2792 See also: L<mkdtemp(3)>");
2793
2794   ("wc_l", (RInt "lines", [Pathname "path"]), 118, [],
2795    [InitISOFS, Always, TestOutputInt (
2796       [["wc_l"; "/10klines"]], 10000)],
2797    "count lines in a file",
2798    "\
2799 This command counts the lines in a file, using the
2800 C<wc -l> external command.");
2801
2802   ("wc_w", (RInt "words", [Pathname "path"]), 119, [],
2803    [InitISOFS, Always, TestOutputInt (
2804       [["wc_w"; "/10klines"]], 10000)],
2805    "count words in a file",
2806    "\
2807 This command counts the words in a file, using the
2808 C<wc -w> external command.");
2809
2810   ("wc_c", (RInt "chars", [Pathname "path"]), 120, [],
2811    [InitISOFS, Always, TestOutputInt (
2812       [["wc_c"; "/100kallspaces"]], 102400)],
2813    "count characters in a file",
2814    "\
2815 This command counts the characters in a file, using the
2816 C<wc -c> external command.");
2817
2818   ("head", (RStringList "lines", [Pathname "path"]), 121, [ProtocolLimitWarning],
2819    [InitISOFS, Always, TestOutputList (
2820       [["head"; "/10klines"]], ["0abcdefghijklmnopqrstuvwxyz";"1abcdefghijklmnopqrstuvwxyz";"2abcdefghijklmnopqrstuvwxyz";"3abcdefghijklmnopqrstuvwxyz";"4abcdefghijklmnopqrstuvwxyz";"5abcdefghijklmnopqrstuvwxyz";"6abcdefghijklmnopqrstuvwxyz";"7abcdefghijklmnopqrstuvwxyz";"8abcdefghijklmnopqrstuvwxyz";"9abcdefghijklmnopqrstuvwxyz"])],
2821    "return first 10 lines of a file",
2822    "\
2823 This command returns up to the first 10 lines of a file as
2824 a list of strings.");
2825
2826   ("head_n", (RStringList "lines", [Int "nrlines"; Pathname "path"]), 122, [ProtocolLimitWarning],
2827    [InitISOFS, Always, TestOutputList (
2828       [["head_n"; "3"; "/10klines"]], ["0abcdefghijklmnopqrstuvwxyz";"1abcdefghijklmnopqrstuvwxyz";"2abcdefghijklmnopqrstuvwxyz"]);
2829     InitISOFS, Always, TestOutputList (
2830       [["head_n"; "-9997"; "/10klines"]], ["0abcdefghijklmnopqrstuvwxyz";"1abcdefghijklmnopqrstuvwxyz";"2abcdefghijklmnopqrstuvwxyz"]);
2831     InitISOFS, Always, TestOutputList (
2832       [["head_n"; "0"; "/10klines"]], [])],
2833    "return first N lines of a file",
2834    "\
2835 If the parameter C<nrlines> is a positive number, this returns the first
2836 C<nrlines> lines of the file C<path>.
2837
2838 If the parameter C<nrlines> is a negative number, this returns lines
2839 from the file C<path>, excluding the last C<nrlines> lines.
2840
2841 If the parameter C<nrlines> is zero, this returns an empty list.");
2842
2843   ("tail", (RStringList "lines", [Pathname "path"]), 123, [ProtocolLimitWarning],
2844    [InitISOFS, Always, TestOutputList (
2845       [["tail"; "/10klines"]], ["9990abcdefghijklmnopqrstuvwxyz";"9991abcdefghijklmnopqrstuvwxyz";"9992abcdefghijklmnopqrstuvwxyz";"9993abcdefghijklmnopqrstuvwxyz";"9994abcdefghijklmnopqrstuvwxyz";"9995abcdefghijklmnopqrstuvwxyz";"9996abcdefghijklmnopqrstuvwxyz";"9997abcdefghijklmnopqrstuvwxyz";"9998abcdefghijklmnopqrstuvwxyz";"9999abcdefghijklmnopqrstuvwxyz"])],
2846    "return last 10 lines of a file",
2847    "\
2848 This command returns up to the last 10 lines of a file as
2849 a list of strings.");
2850
2851   ("tail_n", (RStringList "lines", [Int "nrlines"; Pathname "path"]), 124, [ProtocolLimitWarning],
2852    [InitISOFS, Always, TestOutputList (
2853       [["tail_n"; "3"; "/10klines"]], ["9997abcdefghijklmnopqrstuvwxyz";"9998abcdefghijklmnopqrstuvwxyz";"9999abcdefghijklmnopqrstuvwxyz"]);
2854     InitISOFS, Always, TestOutputList (
2855       [["tail_n"; "-9998"; "/10klines"]], ["9997abcdefghijklmnopqrstuvwxyz";"9998abcdefghijklmnopqrstuvwxyz";"9999abcdefghijklmnopqrstuvwxyz"]);
2856     InitISOFS, Always, TestOutputList (
2857       [["tail_n"; "0"; "/10klines"]], [])],
2858    "return last N lines of a file",
2859    "\
2860 If the parameter C<nrlines> is a positive number, this returns the last
2861 C<nrlines> lines of the file C<path>.
2862
2863 If the parameter C<nrlines> is a negative number, this returns lines
2864 from the file C<path>, starting with the C<-nrlines>th line.
2865
2866 If the parameter C<nrlines> is zero, this returns an empty list.");
2867
2868   ("df", (RString "output", []), 125, [],
2869    [], (* XXX Tricky to test because it depends on the exact format
2870         * of the 'df' command and other imponderables.
2871         *)
2872    "report file system disk space usage",
2873    "\
2874 This command runs the C<df> command to report disk space used.
2875
2876 This command is mostly useful for interactive sessions.  It
2877 is I<not> intended that you try to parse the output string.
2878 Use C<statvfs> from programs.");
2879
2880   ("df_h", (RString "output", []), 126, [],
2881    [], (* XXX Tricky to test because it depends on the exact format
2882         * of the 'df' command and other imponderables.
2883         *)
2884    "report file system disk space usage (human readable)",
2885    "\
2886 This command runs the C<df -h> command to report disk space used
2887 in human-readable format.
2888
2889 This command is mostly useful for interactive sessions.  It
2890 is I<not> intended that you try to parse the output string.
2891 Use C<statvfs> from programs.");
2892
2893   ("du", (RInt64 "sizekb", [Pathname "path"]), 127, [],
2894    [InitISOFS, Always, TestOutputInt (
2895       [["du"; "/directory"]], 2 (* ISO fs blocksize is 2K *))],
2896    "estimate file space usage",
2897    "\
2898 This command runs the C<du -s> command to estimate file space
2899 usage for C<path>.
2900
2901 C<path> can be a file or a directory.  If C<path> is a directory
2902 then the estimate includes the contents of the directory and all
2903 subdirectories (recursively).
2904
2905 The result is the estimated size in I<kilobytes>
2906 (ie. units of 1024 bytes).");
2907
2908   ("initrd_list", (RStringList "filenames", [Pathname "path"]), 128, [],
2909    [InitISOFS, Always, TestOutputList (
2910       [["initrd_list"; "/initrd"]], ["empty";"known-1";"known-2";"known-3";"known-4"; "known-5"])],
2911    "list files in an initrd",
2912    "\
2913 This command lists out files contained in an initrd.
2914
2915 The files are listed without any initial C</> character.  The
2916 files are listed in the order they appear (not necessarily
2917 alphabetical).  Directory names are listed as separate items.
2918
2919 Old Linux kernels (2.4 and earlier) used a compressed ext2
2920 filesystem as initrd.  We I<only> support the newer initramfs
2921 format (compressed cpio files).");
2922
2923   ("mount_loop", (RErr, [Pathname "file"; Pathname "mountpoint"]), 129, [],
2924    [],
2925    "mount a file using the loop device",
2926    "\
2927 This command lets you mount C<file> (a filesystem image
2928 in a file) on a mount point.  It is entirely equivalent to
2929 the command C<mount -o loop file mountpoint>.");
2930
2931   ("mkswap", (RErr, [Device "device"]), 130, [],
2932    [InitEmpty, Always, TestRun (
2933       [["part_disk"; "/dev/sda"; "mbr"];
2934        ["mkswap"; "/dev/sda1"]])],
2935    "create a swap partition",
2936    "\
2937 Create a swap partition on C<device>.");
2938
2939   ("mkswap_L", (RErr, [String "label"; Device "device"]), 131, [],
2940    [InitEmpty, Always, TestRun (
2941       [["part_disk"; "/dev/sda"; "mbr"];
2942        ["mkswap_L"; "hello"; "/dev/sda1"]])],
2943    "create a swap partition with a label",
2944    "\
2945 Create a swap partition on C<device> with label C<label>.
2946
2947 Note that you cannot attach a swap label to a block device
2948 (eg. C</dev/sda>), just to a partition.  This appears to be
2949 a limitation of the kernel or swap tools.");
2950
2951   ("mkswap_U", (RErr, [String "uuid"; Device "device"]), 132, [Optional "linuxfsuuid"],
2952    (let uuid = uuidgen () in
2953     [InitEmpty, Always, TestRun (
2954        [["part_disk"; "/dev/sda"; "mbr"];
2955         ["mkswap_U"; uuid; "/dev/sda1"]])]),
2956    "create a swap partition with an explicit UUID",
2957    "\
2958 Create a swap partition on C<device> with UUID C<uuid>.");
2959
2960   ("mknod", (RErr, [Int "mode"; Int "devmajor"; Int "devminor"; Pathname "path"]), 133, [Optional "mknod"],
2961    [InitBasicFS, Always, TestOutputStruct (
2962       [["mknod"; "0o10777"; "0"; "0"; "/node"];
2963        (* NB: default umask 022 means 0777 -> 0755 in these tests *)
2964        ["stat"; "/node"]], [CompareWithInt ("mode", 0o10755)]);
2965     InitBasicFS, Always, TestOutputStruct (
2966       [["mknod"; "0o60777"; "66"; "99"; "/node"];
2967        ["stat"; "/node"]], [CompareWithInt ("mode", 0o60755)])],
2968    "make block, character or FIFO devices",
2969    "\
2970 This call creates block or character special devices, or
2971 named pipes (FIFOs).
2972
2973 The C<mode> parameter should be the mode, using the standard
2974 constants.  C<devmajor> and C<devminor> are the
2975 device major and minor numbers, only used when creating block
2976 and character special devices.");
2977
2978   ("mkfifo", (RErr, [Int "mode"; Pathname "path"]), 134, [Optional "mknod"],
2979    [InitBasicFS, Always, TestOutputStruct (
2980       [["mkfifo"; "0o777"; "/node"];
2981        ["stat"; "/node"]], [CompareWithInt ("mode", 0o10755)])],
2982    "make FIFO (named pipe)",
2983    "\
2984 This call creates a FIFO (named pipe) called C<path> with
2985 mode C<mode>.  It is just a convenient wrapper around
2986 C<guestfs_mknod>.");
2987
2988   ("mknod_b", (RErr, [Int "mode"; Int "devmajor"; Int "devminor"; Pathname "path"]), 135, [Optional "mknod"],
2989    [InitBasicFS, Always, TestOutputStruct (
2990       [["mknod_b"; "0o777"; "99"; "66"; "/node"];
2991        ["stat"; "/node"]], [CompareWithInt ("mode", 0o60755)])],
2992    "make block device node",
2993    "\
2994 This call creates a block device node called C<path> with
2995 mode C<mode> and device major/minor C<devmajor> and C<devminor>.
2996 It is just a convenient wrapper around C<guestfs_mknod>.");
2997
2998   ("mknod_c", (RErr, [Int "mode"; Int "devmajor"; Int "devminor"; Pathname "path"]), 136, [Optional "mknod"],
2999    [InitBasicFS, Always, TestOutputStruct (
3000       [["mknod_c"; "0o777"; "99"; "66"; "/node"];
3001        ["stat"; "/node"]], [CompareWithInt ("mode", 0o20755)])],
3002    "make char device node",
3003    "\
3004 This call creates a char device node called C<path> with
3005 mode C<mode> and device major/minor C<devmajor> and C<devminor>.
3006 It is just a convenient wrapper around C<guestfs_mknod>.");
3007
3008   ("umask", (RInt "oldmask", [Int "mask"]), 137, [],
3009    [], (* XXX umask is one of those stateful things that we should
3010         * reset between each test.
3011         *)
3012    "set file mode creation mask (umask)",
3013    "\
3014 This function sets the mask used for creating new files and
3015 device nodes to C<mask & 0777>.
3016
3017 Typical umask values would be C<022> which creates new files
3018 with permissions like \"-rw-r--r--\" or \"-rwxr-xr-x\", and
3019 C<002> which creates new files with permissions like
3020 \"-rw-rw-r--\" or \"-rwxrwxr-x\".
3021
3022 The default umask is C<022>.  This is important because it
3023 means that directories and device nodes will be created with
3024 C<0644> or C<0755> mode even if you specify C<0777>.
3025
3026 See also L<umask(2)>, C<guestfs_mknod>, C<guestfs_mkdir>.
3027
3028 This call returns the previous umask.");
3029
3030   ("readdir", (RStructList ("entries", "dirent"), [Pathname "dir"]), 138, [],
3031    [],
3032    "read directories entries",
3033    "\
3034 This returns the list of directory entries in directory C<dir>.
3035
3036 All entries in the directory are returned, including C<.> and
3037 C<..>.  The entries are I<not> sorted, but returned in the same
3038 order as the underlying filesystem.
3039
3040 Also this call returns basic file type information about each
3041 file.  The C<ftyp> field will contain one of the following characters:
3042
3043 =over 4
3044
3045 =item 'b'
3046
3047 Block special
3048
3049 =item 'c'
3050
3051 Char special
3052
3053 =item 'd'
3054
3055 Directory
3056
3057 =item 'f'
3058
3059 FIFO (named pipe)
3060
3061 =item 'l'
3062
3063 Symbolic link
3064
3065 =item 'r'
3066
3067 Regular file
3068
3069 =item 's'
3070
3071 Socket
3072
3073 =item 'u'
3074
3075 Unknown file type
3076
3077 =item '?'
3078
3079 The L<readdir(3)> returned a C<d_type> field with an
3080 unexpected value
3081
3082 =back
3083
3084 This function is primarily intended for use by programs.  To
3085 get a simple list of names, use C<guestfs_ls>.  To get a printable
3086 directory for human consumption, use C<guestfs_ll>.");
3087
3088   ("sfdiskM", (RErr, [Device "device"; StringList "lines"]), 139, [DangerWillRobinson],
3089    [],
3090    "create partitions on a block device",
3091    "\
3092 This is a simplified interface to the C<guestfs_sfdisk>
3093 command, where partition sizes are specified in megabytes
3094 only (rounded to the nearest cylinder) and you don't need
3095 to specify the cyls, heads and sectors parameters which
3096 were rarely if ever used anyway.
3097
3098 See also: C<guestfs_sfdisk>, the L<sfdisk(8)> manpage
3099 and C<guestfs_part_disk>");
3100
3101   ("zfile", (RString "description", [String "meth"; Pathname "path"]), 140, [DeprecatedBy "file"],
3102    [],
3103    "determine file type inside a compressed file",
3104    "\
3105 This command runs C<file> after first decompressing C<path>
3106 using C<method>.
3107
3108 C<method> must be one of C<gzip>, C<compress> or C<bzip2>.
3109
3110 Since 1.0.63, use C<guestfs_file> instead which can now
3111 process compressed files.");
3112
3113   ("getxattrs", (RStructList ("xattrs", "xattr"), [Pathname "path"]), 141, [Optional "linuxxattrs"],
3114    [],
3115    "list extended attributes of a file or directory",
3116    "\
3117 This call lists the extended attributes of the file or directory
3118 C<path>.
3119
3120 At the system call level, this is a combination of the
3121 L<listxattr(2)> and L<getxattr(2)> calls.
3122
3123 See also: C<guestfs_lgetxattrs>, L<attr(5)>.");
3124
3125   ("lgetxattrs", (RStructList ("xattrs", "xattr"), [Pathname "path"]), 142, [Optional "linuxxattrs"],
3126    [],
3127    "list extended attributes of a file or directory",
3128    "\
3129 This is the same as C<guestfs_getxattrs>, but if C<path>
3130 is a symbolic link, then it returns the extended attributes
3131 of the link itself.");
3132
3133   ("setxattr", (RErr, [String "xattr";
3134                        String "val"; Int "vallen"; (* will be BufferIn *)
3135                        Pathname "path"]), 143, [Optional "linuxxattrs"],
3136    [],
3137    "set extended attribute of a file or directory",
3138    "\
3139 This call sets the extended attribute named C<xattr>
3140 of the file C<path> to the value C<val> (of length C<vallen>).
3141 The value is arbitrary 8 bit data.
3142
3143 See also: C<guestfs_lsetxattr>, L<attr(5)>.");
3144
3145   ("lsetxattr", (RErr, [String "xattr";
3146                         String "val"; Int "vallen"; (* will be BufferIn *)
3147                         Pathname "path"]), 144, [Optional "linuxxattrs"],
3148    [],
3149    "set extended attribute of a file or directory",
3150    "\
3151 This is the same as C<guestfs_setxattr>, but if C<path>
3152 is a symbolic link, then it sets an extended attribute
3153 of the link itself.");
3154
3155   ("removexattr", (RErr, [String "xattr"; Pathname "path"]), 145, [Optional "linuxxattrs"],
3156    [],
3157    "remove extended attribute of a file or directory",
3158    "\
3159 This call removes the extended attribute named C<xattr>
3160 of the file C<path>.
3161
3162 See also: C<guestfs_lremovexattr>, L<attr(5)>.");
3163
3164   ("lremovexattr", (RErr, [String "xattr"; Pathname "path"]), 146, [Optional "linuxxattrs"],
3165    [],
3166    "remove extended attribute of a file or directory",
3167    "\
3168 This is the same as C<guestfs_removexattr>, but if C<path>
3169 is a symbolic link, then it removes an extended attribute
3170 of the link itself.");
3171
3172   ("mountpoints", (RHashtable "mps", []), 147, [],
3173    [],
3174    "show mountpoints",
3175    "\
3176 This call is similar to C<guestfs_mounts>.  That call returns
3177 a list of devices.  This one returns a hash table (map) of
3178 device name to directory where the device is mounted.");
3179
3180   ("mkmountpoint", (RErr, [String "exemptpath"]), 148, [],
3181    (* This is a special case: while you would expect a parameter
3182     * of type "Pathname", that doesn't work, because it implies
3183     * NEED_ROOT in the generated calling code in stubs.c, and
3184     * this function cannot use NEED_ROOT.
3185     *)
3186    [],
3187    "create a mountpoint",
3188    "\
3189 C<guestfs_mkmountpoint> and C<guestfs_rmmountpoint> are
3190 specialized calls that can be used to create extra mountpoints
3191 before mounting the first filesystem.
3192
3193 These calls are I<only> necessary in some very limited circumstances,
3194 mainly the case where you want to mount a mix of unrelated and/or
3195 read-only filesystems together.
3196
3197 For example, live CDs often contain a \"Russian doll\" nest of
3198 filesystems, an ISO outer layer, with a squashfs image inside, with
3199 an ext2/3 image inside that.  You can unpack this as follows
3200 in guestfish:
3201
3202  add-ro Fedora-11-i686-Live.iso
3203  run
3204  mkmountpoint /cd
3205  mkmountpoint /squash
3206  mkmountpoint /ext3
3207  mount /dev/sda /cd
3208  mount-loop /cd/LiveOS/squashfs.img /squash
3209  mount-loop /squash/LiveOS/ext3fs.img /ext3
3210
3211 The inner filesystem is now unpacked under the /ext3 mountpoint.");
3212
3213   ("rmmountpoint", (RErr, [String "exemptpath"]), 149, [],
3214    [],
3215    "remove a mountpoint",
3216    "\
3217 This calls removes a mountpoint that was previously created
3218 with C<guestfs_mkmountpoint>.  See C<guestfs_mkmountpoint>
3219 for full details.");
3220
3221   ("read_file", (RBufferOut "content", [Pathname "path"]), 150, [ProtocolLimitWarning],
3222    [InitISOFS, Always, TestOutputBuffer (
3223       [["read_file"; "/known-4"]], "abc\ndef\nghi")],
3224    "read a file",
3225    "\
3226 This calls returns the contents of the file C<path> as a
3227 buffer.
3228
3229 Unlike C<guestfs_cat>, this function can correctly
3230 handle files that contain embedded ASCII NUL characters.
3231 However unlike C<guestfs_download>, this function is limited
3232 in the total size of file that can be handled.");
3233
3234   ("grep", (RStringList "lines", [String "regex"; Pathname "path"]), 151, [ProtocolLimitWarning],
3235    [InitISOFS, Always, TestOutputList (
3236       [["grep"; "abc"; "/test-grep.txt"]], ["abc"; "abc123"]);
3237     InitISOFS, Always, TestOutputList (
3238       [["grep"; "nomatch"; "/test-grep.txt"]], [])],
3239    "return lines matching a pattern",
3240    "\
3241 This calls the external C<grep> program and returns the
3242 matching lines.");
3243
3244   ("egrep", (RStringList "lines", [String "regex"; Pathname "path"]), 152, [ProtocolLimitWarning],
3245    [InitISOFS, Always, TestOutputList (
3246       [["egrep"; "abc"; "/test-grep.txt"]], ["abc"; "abc123"])],
3247    "return lines matching a pattern",
3248    "\
3249 This calls the external C<egrep> program and returns the
3250 matching lines.");
3251
3252   ("fgrep", (RStringList "lines", [String "pattern"; Pathname "path"]), 153, [ProtocolLimitWarning],
3253    [InitISOFS, Always, TestOutputList (
3254       [["fgrep"; "abc"; "/test-grep.txt"]], ["abc"; "abc123"])],
3255    "return lines matching a pattern",
3256    "\
3257 This calls the external C<fgrep> program and returns the
3258 matching lines.");
3259
3260   ("grepi", (RStringList "lines", [String "regex"; Pathname "path"]), 154, [ProtocolLimitWarning],
3261    [InitISOFS, Always, TestOutputList (
3262       [["grepi"; "abc"; "/test-grep.txt"]], ["abc"; "abc123"; "ABC"])],
3263    "return lines matching a pattern",
3264    "\
3265 This calls the external C<grep -i> program and returns the
3266 matching lines.");
3267
3268   ("egrepi", (RStringList "lines", [String "regex"; Pathname "path"]), 155, [ProtocolLimitWarning],
3269    [InitISOFS, Always, TestOutputList (
3270       [["egrepi"; "abc"; "/test-grep.txt"]], ["abc"; "abc123"; "ABC"])],
3271    "return lines matching a pattern",
3272    "\
3273 This calls the external C<egrep -i> program and returns the
3274 matching lines.");
3275
3276   ("fgrepi", (RStringList "lines", [String "pattern"; Pathname "path"]), 156, [ProtocolLimitWarning],
3277    [InitISOFS, Always, TestOutputList (
3278       [["fgrepi"; "abc"; "/test-grep.txt"]], ["abc"; "abc123"; "ABC"])],
3279    "return lines matching a pattern",
3280    "\
3281 This calls the external C<fgrep -i> program and returns the
3282 matching lines.");
3283
3284   ("zgrep", (RStringList "lines", [String "regex"; Pathname "path"]), 157, [ProtocolLimitWarning],
3285    [InitISOFS, Always, TestOutputList (
3286       [["zgrep"; "abc"; "/test-grep.txt.gz"]], ["abc"; "abc123"])],
3287    "return lines matching a pattern",
3288    "\
3289 This calls the external C<zgrep> program and returns the
3290 matching lines.");
3291
3292   ("zegrep", (RStringList "lines", [String "regex"; Pathname "path"]), 158, [ProtocolLimitWarning],
3293    [InitISOFS, Always, TestOutputList (
3294       [["zegrep"; "abc"; "/test-grep.txt.gz"]], ["abc"; "abc123"])],
3295    "return lines matching a pattern",
3296    "\
3297 This calls the external C<zegrep> program and returns the
3298 matching lines.");
3299
3300   ("zfgrep", (RStringList "lines", [String "pattern"; Pathname "path"]), 159, [ProtocolLimitWarning],
3301    [InitISOFS, Always, TestOutputList (
3302       [["zfgrep"; "abc"; "/test-grep.txt.gz"]], ["abc"; "abc123"])],
3303    "return lines matching a pattern",
3304    "\
3305 This calls the external C<zfgrep> program and returns the
3306 matching lines.");
3307
3308   ("zgrepi", (RStringList "lines", [String "regex"; Pathname "path"]), 160, [ProtocolLimitWarning],
3309    [InitISOFS, Always, TestOutputList (
3310       [["zgrepi"; "abc"; "/test-grep.txt.gz"]], ["abc"; "abc123"; "ABC"])],
3311    "return lines matching a pattern",
3312    "\
3313 This calls the external C<zgrep -i> program and returns the
3314 matching lines.");
3315
3316   ("zegrepi", (RStringList "lines", [String "regex"; Pathname "path"]), 161, [ProtocolLimitWarning],
3317    [InitISOFS, Always, TestOutputList (
3318       [["zegrepi"; "abc"; "/test-grep.txt.gz"]], ["abc"; "abc123"; "ABC"])],
3319    "return lines matching a pattern",
3320    "\
3321 This calls the external C<zegrep -i> program and returns the
3322 matching lines.");
3323
3324   ("zfgrepi", (RStringList "lines", [String "pattern"; Pathname "path"]), 162, [ProtocolLimitWarning],
3325    [InitISOFS, Always, TestOutputList (
3326       [["zfgrepi"; "abc"; "/test-grep.txt.gz"]], ["abc"; "abc123"; "ABC"])],
3327    "return lines matching a pattern",
3328    "\
3329 This calls the external C<zfgrep -i> program and returns the
3330 matching lines.");
3331
3332   ("realpath", (RString "rpath", [Pathname "path"]), 163, [Optional "realpath"],
3333    [InitISOFS, Always, TestOutput (
3334       [["realpath"; "/../directory"]], "/directory")],
3335    "canonicalized absolute pathname",
3336    "\
3337 Return the canonicalized absolute pathname of C<path>.  The
3338 returned path has no C<.>, C<..> or symbolic link path elements.");
3339
3340   ("ln", (RErr, [String "target"; Pathname "linkname"]), 164, [],
3341    [InitBasicFS, Always, TestOutputStruct (
3342       [["touch"; "/a"];
3343        ["ln"; "/a"; "/b"];
3344        ["stat"; "/b"]], [CompareWithInt ("nlink", 2)])],
3345    "create a hard link",
3346    "\
3347 This command creates a hard link using the C<ln> command.");
3348
3349   ("ln_f", (RErr, [String "target"; Pathname "linkname"]), 165, [],
3350    [InitBasicFS, Always, TestOutputStruct (
3351       [["touch"; "/a"];
3352        ["touch"; "/b"];
3353        ["ln_f"; "/a"; "/b"];
3354        ["stat"; "/b"]], [CompareWithInt ("nlink", 2)])],
3355    "create a hard link",
3356    "\
3357 This command creates a hard link using the C<ln -f> command.
3358 The C<-f> option removes the link (C<linkname>) if it exists already.");
3359
3360   ("ln_s", (RErr, [String "target"; Pathname "linkname"]), 166, [],
3361    [InitBasicFS, Always, TestOutputStruct (
3362       [["touch"; "/a"];
3363        ["ln_s"; "a"; "/b"];
3364        ["lstat"; "/b"]], [CompareWithInt ("mode", 0o120777)])],
3365    "create a symbolic link",
3366    "\
3367 This command creates a symbolic link using the C<ln -s> command.");
3368
3369   ("ln_sf", (RErr, [String "target"; Pathname "linkname"]), 167, [],
3370    [InitBasicFS, Always, TestOutput (
3371       [["mkdir_p"; "/a/b"];
3372        ["touch"; "/a/b/c"];
3373        ["ln_sf"; "../d"; "/a/b/c"];
3374        ["readlink"; "/a/b/c"]], "../d")],
3375    "create a symbolic link",
3376    "\
3377 This command creates a symbolic link using the C<ln -sf> command,
3378 The C<-f> option removes the link (C<linkname>) if it exists already.");
3379
3380   ("readlink", (RString "link", [Pathname "path"]), 168, [],
3381    [] (* XXX tested above *),
3382    "read the target of a symbolic link",
3383    "\
3384 This command reads the target of a symbolic link.");
3385
3386   ("fallocate", (RErr, [Pathname "path"; Int "len"]), 169, [],
3387    [InitBasicFS, Always, TestOutputStruct (
3388       [["fallocate"; "/a"; "1000000"];
3389        ["stat"; "/a"]], [CompareWithInt ("size", 1_000_000)])],
3390    "preallocate a file in the guest filesystem",
3391    "\
3392 This command preallocates a file (containing zero bytes) named
3393 C<path> of size C<len> bytes.  If the file exists already, it
3394 is overwritten.
3395
3396 Do not confuse this with the guestfish-specific
3397 C<alloc> command which allocates a file in the host and
3398 attaches it as a device.");
3399
3400   ("swapon_device", (RErr, [Device "device"]), 170, [],
3401    [InitPartition, Always, TestRun (
3402       [["mkswap"; "/dev/sda1"];
3403        ["swapon_device"; "/dev/sda1"];
3404        ["swapoff_device"; "/dev/sda1"]])],
3405    "enable swap on device",
3406    "\
3407 This command enables the libguestfs appliance to use the
3408 swap device or partition named C<device>.  The increased
3409 memory is made available for all commands, for example
3410 those run using C<guestfs_command> or C<guestfs_sh>.
3411
3412 Note that you should not swap to existing guest swap
3413 partitions unless you know what you are doing.  They may
3414 contain hibernation information, or other information that
3415 the guest doesn't want you to trash.  You also risk leaking
3416 information about the host to the guest this way.  Instead,
3417 attach a new host device to the guest and swap on that.");
3418
3419   ("swapoff_device", (RErr, [Device "device"]), 171, [],
3420    [], (* XXX tested by swapon_device *)
3421    "disable swap on device",
3422    "\
3423 This command disables the libguestfs appliance swap
3424 device or partition named C<device>.
3425 See C<guestfs_swapon_device>.");
3426
3427   ("swapon_file", (RErr, [Pathname "file"]), 172, [],
3428    [InitBasicFS, Always, TestRun (
3429       [["fallocate"; "/swap"; "8388608"];
3430        ["mkswap_file"; "/swap"];
3431        ["swapon_file"; "/swap"];
3432        ["swapoff_file"; "/swap"]])],
3433    "enable swap on file",
3434    "\
3435 This command enables swap to a file.
3436 See C<guestfs_swapon_device> for other notes.");
3437
3438   ("swapoff_file", (RErr, [Pathname "file"]), 173, [],
3439    [], (* XXX tested by swapon_file *)
3440    "disable swap on file",
3441    "\
3442 This command disables the libguestfs appliance swap on file.");
3443
3444   ("swapon_label", (RErr, [String "label"]), 174, [],
3445    [InitEmpty, Always, TestRun (
3446       [["part_disk"; "/dev/sdb"; "mbr"];
3447        ["mkswap_L"; "swapit"; "/dev/sdb1"];
3448        ["swapon_label"; "swapit"];
3449        ["swapoff_label"; "swapit"];
3450        ["zero"; "/dev/sdb"];
3451        ["blockdev_rereadpt"; "/dev/sdb"]])],
3452    "enable swap on labeled swap partition",
3453    "\
3454 This command enables swap to a labeled swap partition.
3455 See C<guestfs_swapon_device> for other notes.");
3456
3457   ("swapoff_label", (RErr, [String "label"]), 175, [],
3458    [], (* XXX tested by swapon_label *)
3459    "disable swap on labeled swap partition",
3460    "\
3461 This command disables the libguestfs appliance swap on
3462 labeled swap partition.");
3463
3464   ("swapon_uuid", (RErr, [String "uuid"]), 176, [Optional "linuxfsuuid"],
3465    (let uuid = uuidgen () in
3466     [InitEmpty, Always, TestRun (
3467        [["mkswap_U"; uuid; "/dev/sdb"];
3468         ["swapon_uuid"; uuid];
3469         ["swapoff_uuid"; uuid]])]),
3470    "enable swap on swap partition by UUID",
3471    "\
3472 This command enables swap to a swap partition with the given UUID.
3473 See C<guestfs_swapon_device> for other notes.");
3474
3475   ("swapoff_uuid", (RErr, [String "uuid"]), 177, [Optional "linuxfsuuid"],
3476    [], (* XXX tested by swapon_uuid *)
3477    "disable swap on swap partition by UUID",
3478    "\
3479 This command disables the libguestfs appliance swap partition
3480 with the given UUID.");
3481
3482   ("mkswap_file", (RErr, [Pathname "path"]), 178, [],
3483    [InitBasicFS, Always, TestRun (
3484       [["fallocate"; "/swap"; "8388608"];
3485        ["mkswap_file"; "/swap"]])],
3486    "create a swap file",
3487    "\
3488 Create a swap file.
3489
3490 This command just writes a swap file signature to an existing
3491 file.  To create the file itself, use something like C<guestfs_fallocate>.");
3492
3493   ("inotify_init", (RErr, [Int "maxevents"]), 179, [Optional "inotify"],
3494    [InitISOFS, Always, TestRun (
3495       [["inotify_init"; "0"]])],
3496    "create an inotify handle",
3497    "\
3498 This command creates a new inotify handle.
3499 The inotify subsystem can be used to notify events which happen to
3500 objects in the guest filesystem.
3501
3502 C<maxevents> is the maximum number of events which will be
3503 queued up between calls to C<guestfs_inotify_read> or
3504 C<guestfs_inotify_files>.
3505 If this is passed as C<0>, then the kernel (or previously set)
3506 default is used.  For Linux 2.6.29 the default was 16384 events.
3507 Beyond this limit, the kernel throws away events, but records
3508 the fact that it threw them away by setting a flag
3509 C<IN_Q_OVERFLOW> in the returned structure list (see
3510 C<guestfs_inotify_read>).
3511
3512 Before any events are generated, you have to add some
3513 watches to the internal watch list.  See:
3514 C<guestfs_inotify_add_watch>,
3515 C<guestfs_inotify_rm_watch> and
3516 C<guestfs_inotify_watch_all>.
3517
3518 Queued up events should be read periodically by calling
3519 C<guestfs_inotify_read>
3520 (or C<guestfs_inotify_files> which is just a helpful
3521 wrapper around C<guestfs_inotify_read>).  If you don't
3522 read the events out often enough then you risk the internal
3523 queue overflowing.
3524
3525 The handle should be closed after use by calling
3526 C<guestfs_inotify_close>.  This also removes any
3527 watches automatically.
3528
3529 See also L<inotify(7)> for an overview of the inotify interface
3530 as exposed by the Linux kernel, which is roughly what we expose
3531 via libguestfs.  Note that there is one global inotify handle
3532 per libguestfs instance.");
3533
3534   ("inotify_add_watch", (RInt64 "wd", [Pathname "path"; Int "mask"]), 180, [Optional "inotify"],
3535    [InitBasicFS, Always, TestOutputList (
3536       [["inotify_init"; "0"];
3537        ["inotify_add_watch"; "/"; "1073741823"];
3538        ["touch"; "/a"];
3539        ["touch"; "/b"];
3540        ["inotify_files"]], ["a"; "b"])],
3541    "add an inotify watch",
3542    "\
3543 Watch C<path> for the events listed in C<mask>.
3544
3545 Note that if C<path> is a directory then events within that
3546 directory are watched, but this does I<not> happen recursively
3547 (in subdirectories).
3548
3549 Note for non-C or non-Linux callers: the inotify events are
3550 defined by the Linux kernel ABI and are listed in
3551 C</usr/include/sys/inotify.h>.");
3552
3553   ("inotify_rm_watch", (RErr, [Int(*XXX64*) "wd"]), 181, [Optional "inotify"],
3554    [],
3555    "remove an inotify watch",
3556    "\
3557 Remove a previously defined inotify watch.
3558 See C<guestfs_inotify_add_watch>.");
3559
3560   ("inotify_read", (RStructList ("events", "inotify_event"), []), 182, [Optional "inotify"],
3561    [],
3562    "return list of inotify events",
3563    "\
3564 Return the complete queue of events that have happened
3565 since the previous read call.
3566
3567 If no events have happened, this returns an empty list.
3568
3569 I<Note>: In order to make sure that all events have been
3570 read, you must call this function repeatedly until it
3571 returns an empty list.  The reason is that the call will
3572 read events up to the maximum appliance-to-host message
3573 size and leave remaining events in the queue.");
3574
3575   ("inotify_files", (RStringList "paths", []), 183, [Optional "inotify"],
3576    [],
3577    "return list of watched files that had events",
3578    "\
3579 This function is a helpful wrapper around C<guestfs_inotify_read>
3580 which just returns a list of pathnames of objects that were
3581 touched.  The returned pathnames are sorted and deduplicated.");
3582
3583   ("inotify_close", (RErr, []), 184, [Optional "inotify"],
3584    [],
3585    "close the inotify handle",
3586    "\
3587 This closes the inotify handle which was previously
3588 opened by inotify_init.  It removes all watches, throws
3589 away any pending events, and deallocates all resources.");
3590
3591   ("setcon", (RErr, [String "context"]), 185, [Optional "selinux"],
3592    [],
3593    "set SELinux security context",
3594    "\
3595 This sets the SELinux security context of the daemon
3596 to the string C<context>.
3597
3598 See the documentation about SELINUX in L<guestfs(3)>.");
3599
3600   ("getcon", (RString "context", []), 186, [Optional "selinux"],
3601    [],
3602    "get SELinux security context",
3603    "\
3604 This gets the SELinux security context of the daemon.
3605
3606 See the documentation about SELINUX in L<guestfs(3)>,
3607 and C<guestfs_setcon>");
3608
3609   ("mkfs_b", (RErr, [String "fstype"; Int "blocksize"; Device "device"]), 187, [],
3610    [InitEmpty, Always, TestOutput (
3611       [["part_disk"; "/dev/sda"; "mbr"];
3612        ["mkfs_b"; "ext2"; "4096"; "/dev/sda1"];
3613        ["mount_options"; ""; "/dev/sda1"; "/"];
3614        ["write_file"; "/new"; "new file contents"; "0"];
3615        ["cat"; "/new"]], "new file contents")],
3616    "make a filesystem with block size",
3617    "\
3618 This call is similar to C<guestfs_mkfs>, but it allows you to
3619 control the block size of the resulting filesystem.  Supported
3620 block sizes depend on the filesystem type, but typically they
3621 are C<1024>, C<2048> or C<4096> only.");
3622
3623   ("mke2journal", (RErr, [Int "blocksize"; Device "device"]), 188, [],
3624    [InitEmpty, Always, TestOutput (
3625       [["sfdiskM"; "/dev/sda"; ",100 ,"];
3626        ["mke2journal"; "4096"; "/dev/sda1"];
3627        ["mke2fs_J"; "ext2"; "4096"; "/dev/sda2"; "/dev/sda1"];
3628        ["mount_options"; ""; "/dev/sda2"; "/"];
3629        ["write_file"; "/new"; "new file contents"; "0"];
3630        ["cat"; "/new"]], "new file contents")],
3631    "make ext2/3/4 external journal",
3632    "\
3633 This creates an ext2 external journal on C<device>.  It is equivalent
3634 to the command:
3635
3636  mke2fs -O journal_dev -b blocksize device");
3637
3638   ("mke2journal_L", (RErr, [Int "blocksize"; String "label"; Device "device"]), 189, [],
3639    [InitEmpty, Always, TestOutput (
3640       [["sfdiskM"; "/dev/sda"; ",100 ,"];
3641        ["mke2journal_L"; "4096"; "JOURNAL"; "/dev/sda1"];
3642        ["mke2fs_JL"; "ext2"; "4096"; "/dev/sda2"; "JOURNAL"];
3643        ["mount_options"; ""; "/dev/sda2"; "/"];
3644        ["write_file"; "/new"; "new file contents"; "0"];
3645        ["cat"; "/new"]], "new file contents")],
3646    "make ext2/3/4 external journal with label",
3647    "\
3648 This creates an ext2 external journal on C<device> with label C<label>.");
3649
3650   ("mke2journal_U", (RErr, [Int "blocksize"; String "uuid"; Device "device"]), 190, [Optional "linuxfsuuid"],
3651    (let uuid = uuidgen () in
3652     [InitEmpty, Always, TestOutput (
3653        [["sfdiskM"; "/dev/sda"; ",100 ,"];
3654         ["mke2journal_U"; "4096"; uuid; "/dev/sda1"];
3655         ["mke2fs_JU"; "ext2"; "4096"; "/dev/sda2"; uuid];
3656         ["mount_options"; ""; "/dev/sda2"; "/"];
3657         ["write_file"; "/new"; "new file contents"; "0"];
3658         ["cat"; "/new"]], "new file contents")]),
3659    "make ext2/3/4 external journal with UUID",
3660    "\
3661 This creates an ext2 external journal on C<device> with UUID C<uuid>.");
3662
3663   ("mke2fs_J", (RErr, [String "fstype"; Int "blocksize"; Device "device"; Device "journal"]), 191, [],
3664    [],
3665    "make ext2/3/4 filesystem with external journal",
3666    "\
3667 This creates an ext2/3/4 filesystem on C<device> with
3668 an external journal on C<journal>.  It is equivalent
3669 to the command:
3670
3671  mke2fs -t fstype -b blocksize -J device=<journal> <device>
3672
3673 See also C<guestfs_mke2journal>.");
3674
3675   ("mke2fs_JL", (RErr, [String "fstype"; Int "blocksize"; Device "device"; String "label"]), 192, [],
3676    [],
3677    "make ext2/3/4 filesystem with external journal",
3678    "\
3679 This creates an ext2/3/4 filesystem on C<device> with
3680 an external journal on the journal labeled C<label>.
3681
3682 See also C<guestfs_mke2journal_L>.");
3683
3684   ("mke2fs_JU", (RErr, [String "fstype"; Int "blocksize"; Device "device"; String "uuid"]), 193, [Optional "linuxfsuuid"],
3685    [],
3686    "make ext2/3/4 filesystem with external journal",
3687    "\
3688 This creates an ext2/3/4 filesystem on C<device> with
3689 an external journal on the journal with UUID C<uuid>.
3690
3691 See also C<guestfs_mke2journal_U>.");
3692
3693   ("modprobe", (RErr, [String "modulename"]), 194, [Optional "linuxmodules"],
3694    [InitNone, Always, TestRun [["modprobe"; "fat"]]],
3695    "load a kernel module",
3696    "\
3697 This loads a kernel module in the appliance.
3698
3699 The kernel module must have been whitelisted when libguestfs
3700 was built (see C<appliance/kmod.whitelist.in> in the source).");
3701
3702   ("echo_daemon", (RString "output", [StringList "words"]), 195, [],
3703    [InitNone, Always, TestOutput (
3704       [["echo_daemon"; "This is a test"]], "This is a test"
3705     )],
3706    "echo arguments back to the client",
3707    "\
3708 This command concatenate the list of C<words> passed with single spaces between
3709 them and returns the resulting string.
3710
3711 You can use this command to test the connection through to the daemon.
3712
3713 See also C<guestfs_ping_daemon>.");
3714
3715   ("find0", (RErr, [Pathname "directory"; FileOut "files"]), 196, [],
3716    [], (* There is a regression test for this. *)
3717    "find all files and directories, returning NUL-separated list",
3718    "\
3719 This command lists out all files and directories, recursively,
3720 starting at C<directory>, placing the resulting list in the
3721 external file called C<files>.
3722
3723 This command works the same way as C<guestfs_find> with the
3724 following exceptions:
3725
3726 =over 4
3727
3728 =item *
3729
3730 The resulting list is written to an external file.
3731
3732 =item *
3733
3734 Items (filenames) in the result are separated
3735 by C<\\0> characters.  See L<find(1)> option I<-print0>.
3736
3737 =item *
3738
3739 This command is not limited in the number of names that it
3740 can return.
3741
3742 =item *
3743
3744 The result list is not sorted.
3745
3746 =back");
3747
3748   ("case_sensitive_path", (RString "rpath", [Pathname "path"]), 197, [],
3749    [InitISOFS, Always, TestOutput (
3750       [["case_sensitive_path"; "/DIRECTORY"]], "/directory");
3751     InitISOFS, Always, TestOutput (
3752       [["case_sensitive_path"; "/DIRECTORY/"]], "/directory");
3753     InitISOFS, Always, TestOutput (
3754       [["case_sensitive_path"; "/Known-1"]], "/known-1");
3755     InitISOFS, Always, TestLastFail (
3756       [["case_sensitive_path"; "/Known-1/"]]);
3757     InitBasicFS, Always, TestOutput (
3758       [["mkdir"; "/a"];
3759        ["mkdir"; "/a/bbb"];
3760        ["touch"; "/a/bbb/c"];
3761        ["case_sensitive_path"; "/A/bbB/C"]], "/a/bbb/c");
3762     InitBasicFS, Always, TestOutput (
3763       [["mkdir"; "/a"];
3764        ["mkdir"; "/a/bbb"];
3765        ["touch"; "/a/bbb/c"];
3766        ["case_sensitive_path"; "/A////bbB/C"]], "/a/bbb/c");
3767     InitBasicFS, Always, TestLastFail (
3768       [["mkdir"; "/a"];
3769        ["mkdir"; "/a/bbb"];
3770        ["touch"; "/a/bbb/c"];
3771        ["case_sensitive_path"; "/A/bbb/../bbb/C"]])],
3772    "return true path on case-insensitive filesystem",
3773    "\
3774 This can be used to resolve case insensitive paths on
3775 a filesystem which is case sensitive.  The use case is
3776 to resolve paths which you have read from Windows configuration
3777 files or the Windows Registry, to the true path.
3778
3779 The command handles a peculiarity of the Linux ntfs-3g
3780 filesystem driver (and probably others), which is that although
3781 the underlying filesystem is case-insensitive, the driver
3782 exports the filesystem to Linux as case-sensitive.
3783
3784 One consequence of this is that special directories such
3785 as C<c:\\windows> may appear as C</WINDOWS> or C</windows>
3786 (or other things) depending on the precise details of how
3787 they were created.  In Windows itself this would not be
3788 a problem.
3789
3790 Bug or feature?  You decide:
3791 L<http://www.tuxera.com/community/ntfs-3g-faq/#posixfilenames1>
3792
3793 This function resolves the true case of each element in the
3794 path and returns the case-sensitive path.
3795
3796 Thus C<guestfs_case_sensitive_path> (\"/Windows/System32\")
3797 might return C<\"/WINDOWS/system32\"> (the exact return value
3798 would depend on details of how the directories were originally
3799 created under Windows).
3800
3801 I<Note>:
3802 This function does not handle drive names, backslashes etc.
3803
3804 See also C<guestfs_realpath>.");
3805
3806   ("vfs_type", (RString "fstype", [Device "device"]), 198, [],
3807    [InitBasicFS, Always, TestOutput (
3808       [["vfs_type"; "/dev/sda1"]], "ext2")],
3809    "get the Linux VFS type corresponding to a mounted device",
3810    "\
3811 This command gets the block device type corresponding to
3812 a mounted device called C<device>.
3813
3814 Usually the result is the name of the Linux VFS module that
3815 is used to mount this device (probably determined automatically
3816 if you used the C<guestfs_mount> call).");
3817
3818   ("truncate", (RErr, [Pathname "path"]), 199, [],
3819    [InitBasicFS, Always, TestOutputStruct (
3820       [["write_file"; "/test"; "some stuff so size is not zero"; "0"];
3821        ["truncate"; "/test"];
3822        ["stat"; "/test"]], [CompareWithInt ("size", 0)])],
3823    "truncate a file to zero size",
3824    "\
3825 This command truncates C<path> to a zero-length file.  The
3826 file must exist already.");
3827
3828   ("truncate_size", (RErr, [Pathname "path"; Int64 "size"]), 200, [],
3829    [InitBasicFS, Always, TestOutputStruct (
3830       [["touch"; "/test"];
3831        ["truncate_size"; "/test"; "1000"];
3832        ["stat"; "/test"]], [CompareWithInt ("size", 1000)])],
3833    "truncate a file to a particular size",
3834    "\
3835 This command truncates C<path> to size C<size> bytes.  The file
3836 must exist already.  If the file is smaller than C<size> then
3837 the file is extended to the required size with null bytes.");
3838
3839   ("utimens", (RErr, [Pathname "path"; Int64 "atsecs"; Int64 "atnsecs"; Int64 "mtsecs"; Int64 "mtnsecs"]), 201, [],
3840    [InitBasicFS, Always, TestOutputStruct (
3841       [["touch"; "/test"];
3842        ["utimens"; "/test"; "12345"; "67890"; "9876"; "5432"];
3843        ["stat"; "/test"]], [CompareWithInt ("mtime", 9876)])],
3844    "set timestamp of a file with nanosecond precision",
3845    "\
3846 This command sets the timestamps of a file with nanosecond
3847 precision.
3848
3849 C<atsecs, atnsecs> are the last access time (atime) in secs and
3850 nanoseconds from the epoch.
3851
3852 C<mtsecs, mtnsecs> are the last modification time (mtime) in
3853 secs and nanoseconds from the epoch.
3854
3855 If the C<*nsecs> field contains the special value C<-1> then
3856 the corresponding timestamp is set to the current time.  (The
3857 C<*secs> field is ignored in this case).
3858
3859 If the C<*nsecs> field contains the special value C<-2> then
3860 the corresponding timestamp is left unchanged.  (The
3861 C<*secs> field is ignored in this case).");
3862
3863   ("mkdir_mode", (RErr, [Pathname "path"; Int "mode"]), 202, [],
3864    [InitBasicFS, Always, TestOutputStruct (
3865       [["mkdir_mode"; "/test"; "0o111"];
3866        ["stat"; "/test"]], [CompareWithInt ("mode", 0o40111)])],
3867    "create a directory with a particular mode",
3868    "\
3869 This command creates a directory, setting the initial permissions
3870 of the directory to C<mode>.  See also C<guestfs_mkdir>.");
3871
3872   ("lchown", (RErr, [Int "owner"; Int "group"; Pathname "path"]), 203, [],
3873    [], (* XXX *)
3874    "change file owner and group",
3875    "\
3876 Change the file owner to C<owner> and group to C<group>.
3877 This is like C<guestfs_chown> but if C<path> is a symlink then
3878 the link itself is changed, not the target.
3879
3880 Only numeric uid and gid are supported.  If you want to use
3881 names, you will need to locate and parse the password file
3882 yourself (Augeas support makes this relatively easy).");
3883
3884   ("lstatlist", (RStructList ("statbufs", "stat"), [Pathname "path"; StringList "names"]), 204, [],
3885    [], (* XXX *)
3886    "lstat on multiple files",
3887    "\
3888 This call allows you to perform the C<guestfs_lstat> operation
3889 on multiple files, where all files are in the directory C<path>.
3890 C<names> is the list of files from this directory.
3891
3892 On return you get a list of stat structs, with a one-to-one
3893 correspondence to the C<names> list.  If any name did not exist
3894 or could not be lstat'd, then the C<ino> field of that structure
3895 is set to C<-1>.
3896
3897 This call is intended for programs that want to efficiently
3898 list a directory contents without making many round-trips.
3899 See also C<guestfs_lxattrlist> for a similarly efficient call
3900 for getting extended attributes.  Very long directory listings
3901 might cause the protocol message size to be exceeded, causing
3902 this call to fail.  The caller must split up such requests
3903 into smaller groups of names.");
3904
3905   ("lxattrlist", (RStructList ("xattrs", "xattr"), [Pathname "path"; StringList "names"]), 205, [Optional "linuxxattrs"],
3906    [], (* XXX *)
3907    "lgetxattr on multiple files",
3908    "\
3909 This call allows you to get the extended attributes
3910 of multiple files, where all files are in the directory C<path>.
3911 C<names> is the list of files from this directory.
3912
3913 On return you get a flat list of xattr structs which must be
3914 interpreted sequentially.  The first xattr struct always has a zero-length
3915 C<attrname>.  C<attrval> in this struct is zero-length
3916 to indicate there was an error doing C<lgetxattr> for this
3917 file, I<or> is a C string which is a decimal number
3918 (the number of following attributes for this file, which could
3919 be C<\"0\">).  Then after the first xattr struct are the
3920 zero or more attributes for the first named file.
3921 This repeats for the second and subsequent files.
3922
3923 This call is intended for programs that want to efficiently
3924 list a directory contents without making many round-trips.
3925 See also C<guestfs_lstatlist> for a similarly efficient call
3926 for getting standard stats.  Very long directory listings
3927 might cause the protocol message size to be exceeded, causing
3928 this call to fail.  The caller must split up such requests
3929 into smaller groups of names.");
3930
3931   ("readlinklist", (RStringList "links", [Pathname "path"; StringList "names"]), 206, [],
3932    [], (* XXX *)
3933    "readlink on multiple files",
3934    "\
3935 This call allows you to do a C<readlink> operation
3936 on multiple files, where all files are in the directory C<path>.
3937 C<names> is the list of files from this directory.
3938
3939 On return you get a list of strings, with a one-to-one
3940 correspondence to the C<names> list.  Each string is the
3941 value of the symbol link.
3942
3943 If the C<readlink(2)> operation fails on any name, then
3944 the corresponding result string is the empty string C<\"\">.
3945 However the whole operation is completed even if there
3946 were C<readlink(2)> errors, and so you can call this
3947 function with names where you don't know if they are
3948 symbolic links already (albeit slightly less efficient).
3949
3950 This call is intended for programs that want to efficiently
3951 list a directory contents without making many round-trips.
3952 Very long directory listings might cause the protocol
3953 message size to be exceeded, causing
3954 this call to fail.  The caller must split up such requests
3955 into smaller groups of names.");
3956
3957   ("pread", (RBufferOut "content", [Pathname "path"; Int "count"; Int64 "offset"]), 207, [ProtocolLimitWarning],
3958    [InitISOFS, Always, TestOutputBuffer (
3959       [["pread"; "/known-4"; "1"; "3"]], "\n");
3960     InitISOFS, Always, TestOutputBuffer (
3961       [["pread"; "/empty"; "0"; "100"]], "")],
3962    "read part of a file",
3963    "\
3964 This command lets you read part of a file.  It reads C<count>
3965 bytes of the file, starting at C<offset>, from file C<path>.
3966
3967 This may read fewer bytes than requested.  For further details
3968 see the L<pread(2)> system call.");
3969
3970   ("part_init", (RErr, [Device "device"; String "parttype"]), 208, [],
3971    [InitEmpty, Always, TestRun (
3972       [["part_init"; "/dev/sda"; "gpt"]])],
3973    "create an empty partition table",
3974    "\
3975 This creates an empty partition table on C<device> of one of the
3976 partition types listed below.  Usually C<parttype> should be
3977 either C<msdos> or C<gpt> (for large disks).
3978
3979 Initially there are no partitions.  Following this, you should
3980 call C<guestfs_part_add> for each partition required.
3981
3982 Possible values for C<parttype> are:
3983
3984 =over 4
3985
3986 =item B<efi> | B<gpt>
3987
3988 Intel EFI / GPT partition table.
3989
3990 This is recommended for >= 2 TB partitions that will be accessed
3991 from Linux and Intel-based Mac OS X.  It also has limited backwards
3992 compatibility with the C<mbr> format.
3993
3994 =item B<mbr> | B<msdos>
3995
3996 The standard PC \"Master Boot Record\" (MBR) format used
3997 by MS-DOS and Windows.  This partition type will B<only> work
3998 for device sizes up to 2 TB.  For large disks we recommend
3999 using C<gpt>.
4000
4001 =back
4002
4003 Other partition table types that may work but are not
4004 supported include:
4005
4006 =over 4
4007
4008 =item B<aix>
4009
4010 AIX disk labels.
4011
4012 =item B<amiga> | B<rdb>
4013
4014 Amiga \"Rigid Disk Block\" format.
4015
4016 =item B<bsd>
4017
4018 BSD disk labels.
4019
4020 =item B<dasd>
4021
4022 DASD, used on IBM mainframes.
4023
4024 =item B<dvh>
4025
4026 MIPS/SGI volumes.
4027
4028 =item B<mac>
4029
4030 Old Mac partition format.  Modern Macs use C<gpt>.
4031
4032 =item B<pc98>
4033
4034 NEC PC-98 format, common in Japan apparently.
4035
4036 =item B<sun>
4037
4038 Sun disk labels.
4039
4040 =back");
4041
4042   ("part_add", (RErr, [Device "device"; String "prlogex"; Int64 "startsect"; Int64 "endsect"]), 209, [],
4043    [InitEmpty, Always, TestRun (
4044       [["part_init"; "/dev/sda"; "mbr"];
4045        ["part_add"; "/dev/sda"; "primary"; "1"; "-1"]]);
4046     InitEmpty, Always, TestRun (
4047       [["part_init"; "/dev/sda"; "gpt"];
4048        ["part_add"; "/dev/sda"; "primary"; "34"; "127"];
4049        ["part_add"; "/dev/sda"; "primary"; "128"; "-34"]]);
4050     InitEmpty, Always, TestRun (
4051       [["part_init"; "/dev/sda"; "mbr"];
4052        ["part_add"; "/dev/sda"; "primary"; "32"; "127"];
4053        ["part_add"; "/dev/sda"; "primary"; "128"; "255"];
4054        ["part_add"; "/dev/sda"; "primary"; "256"; "511"];
4055        ["part_add"; "/dev/sda"; "primary"; "512"; "-1"]])],
4056    "add a partition to the device",
4057    "\
4058 This command adds a partition to C<device>.  If there is no partition
4059 table on the device, call C<guestfs_part_init> first.
4060
4061 The C<prlogex> parameter is the type of partition.  Normally you
4062 should pass C<p> or C<primary> here, but MBR partition tables also
4063 support C<l> (or C<logical>) and C<e> (or C<extended>) partition
4064 types.
4065
4066 C<startsect> and C<endsect> are the start and end of the partition
4067 in I<sectors>.  C<endsect> may be negative, which means it counts
4068 backwards from the end of the disk (C<-1> is the last sector).
4069
4070 Creating a partition which covers the whole disk is not so easy.
4071 Use C<guestfs_part_disk> to do that.");
4072
4073   ("part_disk", (RErr, [Device "device"; String "parttype"]), 210, [DangerWillRobinson],
4074    [InitEmpty, Always, TestRun (
4075       [["part_disk"; "/dev/sda"; "mbr"]]);
4076     InitEmpty, Always, TestRun (
4077       [["part_disk"; "/dev/sda"; "gpt"]])],
4078    "partition whole disk with a single primary partition",
4079    "\
4080 This command is simply a combination of C<guestfs_part_init>
4081 followed by C<guestfs_part_add> to create a single primary partition
4082 covering the whole disk.
4083
4084 C<parttype> is the partition table type, usually C<mbr> or C<gpt>,
4085 but other possible values are described in C<guestfs_part_init>.");
4086
4087   ("part_set_bootable", (RErr, [Device "device"; Int "partnum"; Bool "bootable"]), 211, [],
4088    [InitEmpty, Always, TestRun (
4089       [["part_disk"; "/dev/sda"; "mbr"];
4090        ["part_set_bootable"; "/dev/sda"; "1"; "true"]])],
4091    "make a partition bootable",
4092    "\
4093 This sets the bootable flag on partition numbered C<partnum> on
4094 device C<device>.  Note that partitions are numbered from 1.
4095
4096 The bootable flag is used by some PC BIOSes to determine which
4097 partition to boot from.  It is by no means universally recognized,
4098 and in any case if your operating system installed a boot
4099 sector on the device itself, then that takes precedence.");
4100
4101   ("part_set_name", (RErr, [Device "device"; Int "partnum"; String "name"]), 212, [],
4102    [InitEmpty, Always, TestRun (
4103       [["part_disk"; "/dev/sda"; "gpt"];
4104        ["part_set_name"; "/dev/sda"; "1"; "thepartname"]])],
4105    "set partition name",
4106    "\
4107 This sets the partition name on partition numbered C<partnum> on
4108 device C<device>.  Note that partitions are numbered from 1.
4109
4110 The partition name can only be set on certain types of partition
4111 table.  This works on C<gpt> but not on C<mbr> partitions.");
4112
4113   ("part_list", (RStructList ("partitions", "partition"), [Device "device"]), 213, [],
4114    [], (* XXX Add a regression test for this. *)
4115    "list partitions on a device",
4116    "\
4117 This command parses the partition table on C<device> and
4118 returns the list of partitions found.
4119
4120 The fields in the returned structure are:
4121
4122 =over 4
4123
4124 =item B<part_num>
4125
4126 Partition number, counting from 1.
4127
4128 =item B<part_start>
4129
4130 Start of the partition I<in bytes>.  To get sectors you have to
4131 divide by the device's sector size, see C<guestfs_blockdev_getss>.
4132
4133 =item B<part_end>
4134
4135 End of the partition in bytes.
4136
4137 =item B<part_size>
4138
4139 Size of the partition in bytes.
4140
4141 =back");
4142
4143   ("part_get_parttype", (RString "parttype", [Device "device"]), 214, [],
4144    [InitEmpty, Always, TestOutput (
4145       [["part_disk"; "/dev/sda"; "gpt"];
4146        ["part_get_parttype"; "/dev/sda"]], "gpt")],
4147    "get the partition table type",
4148    "\
4149 This command examines the partition table on C<device> and
4150 returns the partition table type (format) being used.
4151
4152 Common return values include: C<msdos> (a DOS/Windows style MBR
4153 partition table), C<gpt> (a GPT/EFI-style partition table).  Other
4154 values are possible, although unusual.  See C<guestfs_part_init>
4155 for a full list.");
4156
4157   ("fill", (RErr, [Int "c"; Int "len"; Pathname "path"]), 215, [],
4158    [InitBasicFS, Always, TestOutputBuffer (
4159       [["fill"; "0x63"; "10"; "/test"];
4160        ["read_file"; "/test"]], "cccccccccc")],
4161    "fill a file with octets",
4162    "\
4163 This command creates a new file called C<path>.  The initial
4164 content of the file is C<len> octets of C<c>, where C<c>
4165 must be a number in the range C<[0..255]>.
4166
4167 To fill a file with zero bytes (sparsely), it is
4168 much more efficient to use C<guestfs_truncate_size>.");
4169
4170   ("available", (RErr, [StringList "groups"]), 216, [],
4171    [InitNone, Always, TestRun [["available"; ""]]],
4172    "test availability of some parts of the API",
4173    "\
4174 This command is used to check the availability of some
4175 groups of functionality in the appliance, which not all builds of
4176 the libguestfs appliance will be able to provide.
4177
4178 The libguestfs groups, and the functions that those
4179 groups correspond to, are listed in L<guestfs(3)/AVAILABILITY>.
4180
4181 The argument C<groups> is a list of group names, eg:
4182 C<[\"inotify\", \"augeas\"]> would check for the availability of
4183 the Linux inotify functions and Augeas (configuration file
4184 editing) functions.
4185
4186 The command returns no error if I<all> requested groups are available.
4187
4188 It fails with an error if one or more of the requested
4189 groups is unavailable in the appliance.
4190
4191 If an unknown group name is included in the
4192 list of groups then an error is always returned.
4193
4194 I<Notes:>
4195
4196 =over 4
4197
4198 =item *
4199
4200 You must call C<guestfs_launch> before calling this function.
4201
4202 The reason is because we don't know what groups are
4203 supported by the appliance/daemon until it is running and can
4204 be queried.
4205
4206 =item *
4207
4208 If a group of functions is available, this does not necessarily
4209 mean that they will work.  You still have to check for errors
4210 when calling individual API functions even if they are
4211 available.
4212
4213 =item *
4214
4215 It is usually the job of distro packagers to build
4216 complete functionality into the libguestfs appliance.
4217 Upstream libguestfs, if built from source with all
4218 requirements satisfied, will support everything.
4219
4220 =item *
4221
4222 This call was added in version C<1.0.80>.  In previous
4223 versions of libguestfs all you could do would be to speculatively
4224 execute a command to find out if the daemon implemented it.
4225 See also C<guestfs_version>.
4226
4227 =back");
4228
4229   ("dd", (RErr, [Dev_or_Path "src"; Dev_or_Path "dest"]), 217, [],
4230    [InitBasicFS, Always, TestOutputBuffer (
4231       [["write_file"; "/src"; "hello, world"; "0"];
4232        ["dd"; "/src"; "/dest"];
4233        ["read_file"; "/dest"]], "hello, world")],
4234    "copy from source to destination using dd",
4235    "\
4236 This command copies from one source device or file C<src>
4237 to another destination device or file C<dest>.  Normally you
4238 would use this to copy to or from a device or partition, for
4239 example to duplicate a filesystem.
4240
4241 If the destination is a device, it must be as large or larger
4242 than the source file or device, otherwise the copy will fail.
4243 This command cannot do partial copies (see C<guestfs_copy_size>).");
4244
4245   ("filesize", (RInt64 "size", [Pathname "file"]), 218, [],
4246    [InitBasicFS, Always, TestOutputInt (
4247       [["write_file"; "/file"; "hello, world"; "0"];
4248        ["filesize"; "/file"]], 12)],
4249    "return the size of the file in bytes",
4250    "\
4251 This command returns the size of C<file> in bytes.
4252
4253 To get other stats about a file, use C<guestfs_stat>, C<guestfs_lstat>,
4254 C<guestfs_is_dir>, C<guestfs_is_file> etc.
4255 To get the size of block devices, use C<guestfs_blockdev_getsize64>.");
4256
4257   ("lvrename", (RErr, [String "logvol"; String "newlogvol"]), 219, [],
4258    [InitBasicFSonLVM, Always, TestOutputList (
4259       [["lvrename"; "/dev/VG/LV"; "/dev/VG/LV2"];
4260        ["lvs"]], ["/dev/VG/LV2"])],
4261    "rename an LVM logical volume",
4262    "\
4263 Rename a logical volume C<logvol> with the new name C<newlogvol>.");
4264
4265   ("vgrename", (RErr, [String "volgroup"; String "newvolgroup"]), 220, [],
4266    [InitBasicFSonLVM, Always, TestOutputList (
4267       [["umount"; "/"];
4268        ["vg_activate"; "false"; "VG"];
4269        ["vgrename"; "VG"; "VG2"];
4270        ["vg_activate"; "true"; "VG2"];
4271        ["mount_options"; ""; "/dev/VG2/LV"; "/"];
4272        ["vgs"]], ["VG2"])],
4273    "rename an LVM volume group",
4274    "\
4275 Rename a volume group C<volgroup> with the new name C<newvolgroup>.");
4276
4277   ("initrd_cat", (RBufferOut "content", [Pathname "initrdpath"; String "filename"]), 221, [ProtocolLimitWarning],
4278    [InitISOFS, Always, TestOutputBuffer (
4279       [["initrd_cat"; "/initrd"; "known-4"]], "abc\ndef\nghi")],
4280    "list the contents of a single file in an initrd",
4281    "\
4282 This command unpacks the file C<filename> from the initrd file
4283 called C<initrdpath>.  The filename must be given I<without> the
4284 initial C</> character.
4285
4286 For example, in guestfish you could use the following command
4287 to examine the boot script (usually called C</init>)
4288 contained in a Linux initrd or initramfs image:
4289
4290  initrd-cat /boot/initrd-<version>.img init
4291
4292 See also C<guestfs_initrd_list>.");
4293
4294   ("pvuuid", (RString "uuid", [Device "device"]), 222, [],
4295    [],
4296    "get the UUID of a physical volume",
4297    "\
4298 This command returns the UUID of the LVM PV C<device>.");
4299
4300   ("vguuid", (RString "uuid", [String "vgname"]), 223, [],
4301    [],
4302    "get the UUID of a volume group",
4303    "\
4304 This command returns the UUID of the LVM VG named C<vgname>.");
4305
4306   ("lvuuid", (RString "uuid", [Device "device"]), 224, [],
4307    [],
4308    "get the UUID of a logical volume",
4309    "\
4310 This command returns the UUID of the LVM LV C<device>.");
4311
4312   ("vgpvuuids", (RStringList "uuids", [String "vgname"]), 225, [],
4313    [],
4314    "get the PV UUIDs containing the volume group",
4315    "\
4316 Given a VG called C<vgname>, this returns the UUIDs of all
4317 the physical volumes that this volume group resides on.
4318
4319 You can use this along with C<guestfs_pvs> and C<guestfs_pvuuid>
4320 calls to associate physical volumes and volume groups.
4321
4322 See also C<guestfs_vglvuuids>.");
4323
4324   ("vglvuuids", (RStringList "uuids", [String "vgname"]), 226, [],
4325    [],
4326    "get the LV UUIDs of all LVs in the volume group",
4327    "\
4328 Given a VG called C<vgname>, this returns the UUIDs of all
4329 the logical volumes created in this volume group.
4330
4331 You can use this along with C<guestfs_lvs> and C<guestfs_lvuuid>
4332 calls to associate logical volumes and volume groups.
4333
4334 See also C<guestfs_vgpvuuids>.");
4335
4336   ("copy_size", (RErr, [Dev_or_Path "src"; Dev_or_Path "dest"; Int64 "size"]), 227, [],
4337    [InitBasicFS, Always, TestOutputBuffer (
4338       [["write_file"; "/src"; "hello, world"; "0"];
4339        ["copy_size"; "/src"; "/dest"; "5"];
4340        ["read_file"; "/dest"]], "hello")],
4341    "copy size bytes from source to destination using dd",
4342    "\
4343 This command copies exactly C<size> bytes from one source device
4344 or file C<src> to another destination device or file C<dest>.
4345
4346 Note this will fail if the source is too short or if the destination
4347 is not large enough.");
4348
4349 ]
4350
4351 let all_functions = non_daemon_functions @ daemon_functions
4352
4353 (* In some places we want the functions to be displayed sorted
4354  * alphabetically, so this is useful:
4355  *)
4356 let all_functions_sorted =
4357   List.sort (fun (n1,_,_,_,_,_,_) (n2,_,_,_,_,_,_) ->
4358                compare n1 n2) all_functions
4359
4360 (* Field types for structures. *)
4361 type field =
4362   | FChar                       (* C 'char' (really, a 7 bit byte). *)
4363   | FString                     (* nul-terminated ASCII string, NOT NULL. *)
4364   | FBuffer                     (* opaque buffer of bytes, (char *, int) pair *)
4365   | FUInt32
4366   | FInt32
4367   | FUInt64
4368   | FInt64
4369   | FBytes                      (* Any int measure that counts bytes. *)
4370   | FUUID                       (* 32 bytes long, NOT nul-terminated. *)
4371   | FOptPercent                 (* [0..100], or -1 meaning "not present". *)
4372
4373 (* Because we generate extra parsing code for LVM command line tools,
4374  * we have to pull out the LVM columns separately here.
4375  *)
4376 let lvm_pv_cols = [
4377   "pv_name", FString;
4378   "pv_uuid", FUUID;
4379   "pv_fmt", FString;
4380   "pv_size", FBytes;
4381   "dev_size", FBytes;
4382   "pv_free", FBytes;
4383   "pv_used", FBytes;
4384   "pv_attr", FString (* XXX *);
4385   "pv_pe_count", FInt64;
4386   "pv_pe_alloc_count", FInt64;
4387   "pv_tags", FString;
4388   "pe_start", FBytes;
4389   "pv_mda_count", FInt64;
4390   "pv_mda_free", FBytes;
4391   (* Not in Fedora 10:
4392      "pv_mda_size", FBytes;
4393   *)
4394 ]
4395 let lvm_vg_cols = [
4396   "vg_name", FString;
4397   "vg_uuid", FUUID;
4398   "vg_fmt", FString;
4399   "vg_attr", FString (* XXX *);
4400   "vg_size", FBytes;
4401   "vg_free", FBytes;
4402   "vg_sysid", FString;
4403   "vg_extent_size", FBytes;
4404   "vg_extent_count", FInt64;
4405   "vg_free_count", FInt64;
4406   "max_lv", FInt64;
4407   "max_pv", FInt64;
4408   "pv_count", FInt64;
4409   "lv_count", FInt64;
4410   "snap_count", FInt64;
4411   "vg_seqno", FInt64;
4412   "vg_tags", FString;
4413   "vg_mda_count", FInt64;
4414   "vg_mda_free", FBytes;
4415   (* Not in Fedora 10:
4416      "vg_mda_size", FBytes;
4417   *)
4418 ]
4419 let lvm_lv_cols = [
4420   "lv_name", FString;
4421   "lv_uuid", FUUID;
4422   "lv_attr", FString (* XXX *);
4423   "lv_major", FInt64;
4424   "lv_minor", FInt64;
4425   "lv_kernel_major", FInt64;
4426   "lv_kernel_minor", FInt64;
4427   "lv_size", FBytes;
4428   "seg_count", FInt64;
4429   "origin", FString;
4430   "snap_percent", FOptPercent;
4431   "copy_percent", FOptPercent;
4432   "move_pv", FString;
4433   "lv_tags", FString;
4434   "mirror_log", FString;
4435   "modules", FString;
4436 ]
4437
4438 (* Names and fields in all structures (in RStruct and RStructList)
4439  * that we support.
4440  *)
4441 let structs = [
4442   (* The old RIntBool return type, only ever used for aug_defnode.  Do
4443    * not use this struct in any new code.
4444    *)
4445   "int_bool", [
4446     "i", FInt32;                (* for historical compatibility *)
4447     "b", FInt32;                (* for historical compatibility *)
4448   ];
4449
4450   (* LVM PVs, VGs, LVs. *)
4451   "lvm_pv", lvm_pv_cols;
4452   "lvm_vg", lvm_vg_cols;
4453   "lvm_lv", lvm_lv_cols;
4454
4455   (* Column names and types from stat structures.
4456    * NB. Can't use things like 'st_atime' because glibc header files
4457    * define some of these as macros.  Ugh.
4458    *)
4459   "stat", [
4460     "dev", FInt64;
4461     "ino", FInt64;
4462     "mode", FInt64;
4463     "nlink", FInt64;
4464     "uid", FInt64;
4465     "gid", FInt64;
4466     "rdev", FInt64;
4467     "size", FInt64;
4468     "blksize", FInt64;
4469     "blocks", FInt64;
4470     "atime", FInt64;
4471     "mtime", FInt64;
4472     "ctime", FInt64;
4473   ];
4474   "statvfs", [
4475     "bsize", FInt64;
4476     "frsize", FInt64;
4477     "blocks", FInt64;
4478     "bfree", FInt64;
4479     "bavail", FInt64;
4480     "files", FInt64;
4481     "ffree", FInt64;
4482     "favail", FInt64;
4483     "fsid", FInt64;
4484     "flag", FInt64;
4485     "namemax", FInt64;
4486   ];
4487
4488   (* Column names in dirent structure. *)
4489   "dirent", [
4490     "ino", FInt64;
4491     (* 'b' 'c' 'd' 'f' (FIFO) 'l' 'r' (regular file) 's' 'u' '?' *)
4492     "ftyp", FChar;
4493     "name", FString;
4494   ];
4495
4496   (* Version numbers. *)
4497   "version", [
4498     "major", FInt64;
4499     "minor", FInt64;
4500     "release", FInt64;
4501     "extra", FString;
4502   ];
4503
4504   (* Extended attribute. *)
4505   "xattr", [
4506     "attrname", FString;
4507     "attrval", FBuffer;
4508   ];
4509
4510   (* Inotify events. *)
4511   "inotify_event", [
4512     "in_wd", FInt64;
4513     "in_mask", FUInt32;
4514     "in_cookie", FUInt32;
4515     "in_name", FString;
4516   ];
4517
4518   (* Partition table entry. *)
4519   "partition", [
4520     "part_num", FInt32;
4521     "part_start", FBytes;
4522     "part_end", FBytes;
4523     "part_size", FBytes;
4524   ];
4525 ] (* end of structs *)
4526
4527 (* Ugh, Java has to be different ..
4528  * These names are also used by the Haskell bindings.
4529  *)
4530 let java_structs = [
4531   "int_bool", "IntBool";
4532   "lvm_pv", "PV";
4533   "lvm_vg", "VG";
4534   "lvm_lv", "LV";
4535   "stat", "Stat";
4536   "statvfs", "StatVFS";
4537   "dirent", "Dirent";
4538   "version", "Version";
4539   "xattr", "XAttr";
4540   "inotify_event", "INotifyEvent";
4541   "partition", "Partition";
4542 ]
4543
4544 (* What structs are actually returned. *)
4545 type rstructs_used_t = RStructOnly | RStructListOnly | RStructAndList
4546
4547 (* Returns a list of RStruct/RStructList structs that are returned
4548  * by any function.  Each element of returned list is a pair:
4549  *
4550  * (structname, RStructOnly)
4551  *    == there exists function which returns RStruct (_, structname)
4552  * (structname, RStructListOnly)
4553  *    == there exists function which returns RStructList (_, structname)
4554  * (structname, RStructAndList)
4555  *    == there are functions returning both RStruct (_, structname)
4556  *                                      and RStructList (_, structname)
4557  *)
4558 let rstructs_used_by functions =
4559   (* ||| is a "logical OR" for rstructs_used_t *)
4560   let (|||) a b =
4561     match a, b with
4562     | RStructAndList, _
4563     | _, RStructAndList -> RStructAndList
4564     | RStructOnly, RStructListOnly
4565     | RStructListOnly, RStructOnly -> RStructAndList
4566     | RStructOnly, RStructOnly -> RStructOnly
4567     | RStructListOnly, RStructListOnly -> RStructListOnly
4568   in
4569
4570   let h = Hashtbl.create 13 in
4571
4572   (* if elem->oldv exists, update entry using ||| operator,
4573    * else just add elem->newv to the hash
4574    *)
4575   let update elem newv =
4576     try  let oldv = Hashtbl.find h elem in
4577          Hashtbl.replace h elem (newv ||| oldv)
4578     with Not_found -> Hashtbl.add h elem newv
4579   in
4580
4581   List.iter (
4582     fun (_, style, _, _, _, _, _) ->
4583       match fst style with
4584       | RStruct (_, structname) -> update structname RStructOnly
4585       | RStructList (_, structname) -> update structname RStructListOnly
4586       | _ -> ()
4587   ) functions;
4588
4589   (* return key->values as a list of (key,value) *)
4590   Hashtbl.fold (fun key value xs -> (key, value) :: xs) h []
4591
4592 (* Used for testing language bindings. *)
4593 type callt =
4594   | CallString of string
4595   | CallOptString of string option
4596   | CallStringList of string list
4597   | CallInt of int
4598   | CallInt64 of int64
4599   | CallBool of bool
4600
4601 (* Used to memoize the result of pod2text. *)
4602 let pod2text_memo_filename = "src/.pod2text.data"
4603 let pod2text_memo : ((int * string * string), string list) Hashtbl.t =
4604   try
4605     let chan = open_in pod2text_memo_filename in
4606     let v = input_value chan in
4607     close_in chan;
4608     v
4609   with
4610     _ -> Hashtbl.create 13
4611 let pod2text_memo_updated () =
4612   let chan = open_out pod2text_memo_filename in
4613   output_value chan pod2text_memo;
4614   close_out chan
4615
4616 (* Useful functions.
4617  * Note we don't want to use any external OCaml libraries which
4618  * makes this a bit harder than it should be.
4619  *)
4620 module StringMap = Map.Make (String)
4621
4622 let failwithf fs = ksprintf failwith fs
4623
4624 let unique = let i = ref 0 in fun () -> incr i; !i
4625
4626 let replace_char s c1 c2 =
4627   let s2 = String.copy s in
4628   let r = ref false in
4629   for i = 0 to String.length s2 - 1 do
4630     if String.unsafe_get s2 i = c1 then (
4631       String.unsafe_set s2 i c2;
4632       r := true
4633     )
4634   done;
4635   if not !r then s else s2
4636
4637 let isspace c =
4638   c = ' '
4639   (* || c = '\f' *) || c = '\n' || c = '\r' || c = '\t' (* || c = '\v' *)
4640
4641 let triml ?(test = isspace) str =
4642   let i = ref 0 in
4643   let n = ref (String.length str) in
4644   while !n > 0 && test str.[!i]; do
4645     decr n;
4646     incr i
4647   done;
4648   if !i = 0 then str
4649   else String.sub str !i !n
4650
4651 let trimr ?(test = isspace) str =
4652   let n = ref (String.length str) in
4653   while !n > 0 && test str.[!n-1]; do
4654     decr n
4655   done;
4656   if !n = String.length str then str
4657   else String.sub str 0 !n
4658
4659 let trim ?(test = isspace) str =
4660   trimr ~test (triml ~test str)
4661
4662 let rec find s sub =
4663   let len = String.length s in
4664   let sublen = String.length sub in
4665   let rec loop i =
4666     if i <= len-sublen then (
4667       let rec loop2 j =
4668         if j < sublen then (
4669           if s.[i+j] = sub.[j] then loop2 (j+1)
4670           else -1
4671         ) else
4672           i (* found *)
4673       in
4674       let r = loop2 0 in
4675       if r = -1 then loop (i+1) else r
4676     ) else
4677       -1 (* not found *)
4678   in
4679   loop 0
4680
4681 let rec replace_str s s1 s2 =
4682   let len = String.length s in
4683   let sublen = String.length s1 in
4684   let i = find s s1 in
4685   if i = -1 then s
4686   else (
4687     let s' = String.sub s 0 i in
4688     let s'' = String.sub s (i+sublen) (len-i-sublen) in
4689     s' ^ s2 ^ replace_str s'' s1 s2
4690   )
4691
4692 let rec string_split sep str =
4693   let len = String.length str in
4694   let seplen = String.length sep in
4695   let i = find str sep in
4696   if i = -1 then [str]
4697   else (
4698     let s' = String.sub str 0 i in
4699     let s'' = String.sub str (i+seplen) (len-i-seplen) in
4700     s' :: string_split sep s''
4701   )
4702
4703 let files_equal n1 n2 =
4704   let cmd = sprintf "cmp -s %s %s" (Filename.quote n1) (Filename.quote n2) in
4705   match Sys.command cmd with
4706   | 0 -> true
4707   | 1 -> false
4708   | i -> failwithf "%s: failed with error code %d" cmd i
4709
4710 let rec filter_map f = function
4711   | [] -> []
4712   | x :: xs ->
4713       match f x with
4714       | Some y -> y :: filter_map f xs
4715       | None -> filter_map f xs
4716
4717 let rec find_map f = function
4718   | [] -> raise Not_found
4719   | x :: xs ->
4720       match f x with
4721       | Some y -> y
4722       | None -> find_map f xs
4723
4724 let iteri f xs =
4725   let rec loop i = function
4726     | [] -> ()
4727     | x :: xs -> f i x; loop (i+1) xs
4728   in
4729   loop 0 xs
4730
4731 let mapi f xs =
4732   let rec loop i = function
4733     | [] -> []
4734     | x :: xs -> let r = f i x in r :: loop (i+1) xs
4735   in
4736   loop 0 xs
4737
4738 let count_chars c str =
4739   let count = ref 0 in
4740   for i = 0 to String.length str - 1 do
4741     if c = String.unsafe_get str i then incr count
4742   done;
4743   !count
4744
4745 let name_of_argt = function
4746   | Pathname n | Device n | Dev_or_Path n | String n | OptString n
4747   | StringList n | DeviceList n | Bool n | Int n | Int64 n
4748   | FileIn n | FileOut n -> n
4749
4750 let java_name_of_struct typ =
4751   try List.assoc typ java_structs
4752   with Not_found ->
4753     failwithf
4754       "java_name_of_struct: no java_structs entry corresponding to %s" typ
4755
4756 let cols_of_struct typ =
4757   try List.assoc typ structs
4758   with Not_found ->
4759     failwithf "cols_of_struct: unknown struct %s" typ
4760
4761 let seq_of_test = function
4762   | TestRun s | TestOutput (s, _) | TestOutputList (s, _)
4763   | TestOutputListOfDevices (s, _)
4764   | TestOutputInt (s, _) | TestOutputIntOp (s, _, _)
4765   | TestOutputTrue s | TestOutputFalse s
4766   | TestOutputLength (s, _) | TestOutputBuffer (s, _)
4767   | TestOutputStruct (s, _)
4768   | TestLastFail s -> s
4769
4770 (* Handling for function flags. *)
4771 let protocol_limit_warning =
4772   "Because of the message protocol, there is a transfer limit
4773 of somewhere between 2MB and 4MB.  See L<guestfs(3)/PROTOCOL LIMITS>."
4774
4775 let danger_will_robinson =
4776   "B<This command is dangerous.  Without careful use you
4777 can easily destroy all your data>."
4778
4779 let deprecation_notice flags =
4780   try
4781     let alt =
4782       find_map (function DeprecatedBy str -> Some str | _ -> None) flags in
4783     let txt =
4784       sprintf "This function is deprecated.
4785 In new code, use the C<%s> call instead.
4786
4787 Deprecated functions will not be removed from the API, but the
4788 fact that they are deprecated indicates that there are problems
4789 with correct use of these functions." alt in
4790     Some txt
4791   with
4792     Not_found -> None
4793
4794 (* Create list of optional groups. *)
4795 let optgroups =
4796   let h = Hashtbl.create 13 in
4797   List.iter (
4798     fun (name, _, _, flags, _, _, _) ->
4799       List.iter (
4800         function
4801         | Optional group ->
4802             let names = try Hashtbl.find h group with Not_found -> [] in
4803             Hashtbl.replace h group (name :: names)
4804         | _ -> ()
4805       ) flags
4806   ) daemon_functions;
4807   let groups = Hashtbl.fold (fun k _ ks -> k :: ks) h [] in
4808   let groups =
4809     List.map (
4810       fun group -> group, List.sort compare (Hashtbl.find h group)
4811     ) groups in
4812   List.sort (fun x y -> compare (fst x) (fst y)) groups
4813
4814 (* Check function names etc. for consistency. *)
4815 let check_functions () =
4816   let contains_uppercase str =
4817     let len = String.length str in
4818     let rec loop i =
4819       if i >= len then false
4820       else (
4821         let c = str.[i] in
4822         if c >= 'A' && c <= 'Z' then true
4823         else loop (i+1)
4824       )
4825     in
4826     loop 0
4827   in
4828
4829   (* Check function names. *)
4830   List.iter (
4831     fun (name, _, _, _, _, _, _) ->
4832       if String.length name >= 7 && String.sub name 0 7 = "guestfs" then
4833         failwithf "function name %s does not need 'guestfs' prefix" name;
4834       if name = "" then
4835         failwithf "function name is empty";
4836       if name.[0] < 'a' || name.[0] > 'z' then
4837         failwithf "function name %s must start with lowercase a-z" name;
4838       if String.contains name '-' then
4839         failwithf "function name %s should not contain '-', use '_' instead."
4840           name
4841   ) all_functions;
4842
4843   (* Check function parameter/return names. *)
4844   List.iter (
4845     fun (name, style, _, _, _, _, _) ->
4846       let check_arg_ret_name n =
4847         if contains_uppercase n then
4848           failwithf "%s param/ret %s should not contain uppercase chars"
4849             name n;
4850         if String.contains n '-' || String.contains n '_' then
4851           failwithf "%s param/ret %s should not contain '-' or '_'"
4852             name n;
4853         if n = "value" then
4854           failwithf "%s has a param/ret called 'value', which causes conflicts in the OCaml bindings, use something like 'val' or a more descriptive name" name;
4855         if n = "int" || n = "char" || n = "short" || n = "long" then
4856           failwithf "%s has a param/ret which conflicts with a C type (eg. 'int', 'char' etc.)" name;
4857         if n = "i" || n = "n" then
4858           failwithf "%s has a param/ret called 'i' or 'n', which will cause some conflicts in the generated code" name;
4859         if n = "argv" || n = "args" then
4860           failwithf "%s has a param/ret called 'argv' or 'args', which will cause some conflicts in the generated code" name;
4861
4862         (* List Haskell, OCaml and C keywords here.
4863          * http://www.haskell.org/haskellwiki/Keywords
4864          * http://caml.inria.fr/pub/docs/manual-ocaml/lex.html#operator-char
4865          * http://en.wikipedia.org/wiki/C_syntax#Reserved_keywords
4866          * Formatted via: cat c haskell ocaml|sort -u|grep -vE '_|^val$' \
4867          *   |perl -pe 's/(.+)/"$1";/'|fmt -70
4868          * Omitting _-containing words, since they're handled above.
4869          * Omitting the OCaml reserved word, "val", is ok,
4870          * and saves us from renaming several parameters.
4871          *)
4872         let reserved = [
4873           "and"; "as"; "asr"; "assert"; "auto"; "begin"; "break"; "case";
4874           "char"; "class"; "const"; "constraint"; "continue"; "data";
4875           "default"; "deriving"; "do"; "done"; "double"; "downto"; "else";
4876           "end"; "enum"; "exception"; "extern"; "external"; "false"; "float";
4877           "for"; "forall"; "foreign"; "fun"; "function"; "functor"; "goto";
4878           "hiding"; "if"; "import"; "in"; "include"; "infix"; "infixl";
4879           "infixr"; "inherit"; "initializer"; "inline"; "instance"; "int";
4880           "interface";
4881           "land"; "lazy"; "let"; "long"; "lor"; "lsl"; "lsr"; "lxor";
4882           "match"; "mdo"; "method"; "mod"; "module"; "mutable"; "new";
4883           "newtype"; "object"; "of"; "open"; "or"; "private"; "qualified";
4884           "rec"; "register"; "restrict"; "return"; "short"; "sig"; "signed";
4885           "sizeof"; "static"; "struct"; "switch"; "then"; "to"; "true"; "try";
4886           "type"; "typedef"; "union"; "unsigned"; "virtual"; "void";
4887           "volatile"; "when"; "where"; "while";
4888           ] in
4889         if List.mem n reserved then
4890           failwithf "%s has param/ret using reserved word %s" name n;
4891       in
4892
4893       (match fst style with
4894        | RErr -> ()
4895        | RInt n | RInt64 n | RBool n
4896        | RConstString n | RConstOptString n | RString n
4897        | RStringList n | RStruct (n, _) | RStructList (n, _)
4898        | RHashtable n | RBufferOut n ->
4899            check_arg_ret_name n
4900       );
4901       List.iter (fun arg -> check_arg_ret_name (name_of_argt arg)) (snd style)
4902   ) all_functions;
4903
4904   (* Check short descriptions. *)
4905   List.iter (
4906     fun (name, _, _, _, _, shortdesc, _) ->
4907       if shortdesc.[0] <> Char.lowercase shortdesc.[0] then
4908         failwithf "short description of %s should begin with lowercase." name;
4909       let c = shortdesc.[String.length shortdesc-1] in
4910       if c = '\n' || c = '.' then
4911         failwithf "short description of %s should not end with . or \\n." name
4912   ) all_functions;
4913
4914   (* Check long dscriptions. *)
4915   List.iter (
4916     fun (name, _, _, _, _, _, longdesc) ->
4917       if longdesc.[String.length longdesc-1] = '\n' then
4918         failwithf "long description of %s should not end with \\n." name
4919   ) all_functions;
4920
4921   (* Check proc_nrs. *)
4922   List.iter (
4923     fun (name, _, proc_nr, _, _, _, _) ->
4924       if proc_nr <= 0 then
4925         failwithf "daemon function %s should have proc_nr > 0" name
4926   ) daemon_functions;
4927
4928   List.iter (
4929     fun (name, _, proc_nr, _, _, _, _) ->
4930       if proc_nr <> -1 then
4931         failwithf "non-daemon function %s should have proc_nr -1" name
4932   ) non_daemon_functions;
4933
4934   let proc_nrs =
4935     List.map (fun (name, _, proc_nr, _, _, _, _) -> name, proc_nr)
4936       daemon_functions in
4937   let proc_nrs =
4938     List.sort (fun (_,nr1) (_,nr2) -> compare nr1 nr2) proc_nrs in
4939   let rec loop = function
4940     | [] -> ()
4941     | [_] -> ()
4942     | (name1,nr1) :: ((name2,nr2) :: _ as rest) when nr1 < nr2 ->
4943         loop rest
4944     | (name1,nr1) :: (name2,nr2) :: _ ->
4945         failwithf "%s and %s have conflicting procedure numbers (%d, %d)"
4946           name1 name2 nr1 nr2
4947   in
4948   loop proc_nrs;
4949
4950   (* Check tests. *)
4951   List.iter (
4952     function
4953       (* Ignore functions that have no tests.  We generate a
4954        * warning when the user does 'make check' instead.
4955        *)
4956     | name, _, _, _, [], _, _ -> ()
4957     | name, _, _, _, tests, _, _ ->
4958         let funcs =
4959           List.map (
4960             fun (_, _, test) ->
4961               match seq_of_test test with
4962               | [] ->
4963                   failwithf "%s has a test containing an empty sequence" name
4964               | cmds -> List.map List.hd cmds
4965           ) tests in
4966         let funcs = List.flatten funcs in
4967
4968         let tested = List.mem name funcs in
4969
4970         if not tested then
4971           failwithf "function %s has tests but does not test itself" name
4972   ) all_functions
4973
4974 (* 'pr' prints to the current output file. *)
4975 let chan = ref Pervasives.stdout
4976 let lines = ref 0
4977 let pr fs =
4978   ksprintf
4979     (fun str ->
4980        let i = count_chars '\n' str in
4981        lines := !lines + i;
4982        output_string !chan str
4983     ) fs
4984
4985 let copyright_years =
4986   let this_year = 1900 + (localtime (time ())).tm_year in
4987   if this_year > 2009 then sprintf "2009-%04d" this_year else "2009"
4988
4989 (* Generate a header block in a number of standard styles. *)
4990 type comment_style =
4991     CStyle | CPlusPlusStyle | HashStyle | OCamlStyle | HaskellStyle
4992 type license = GPLv2plus | LGPLv2plus
4993
4994 let generate_header ?(extra_inputs = []) comment license =
4995   let inputs = "src/generator.ml" :: extra_inputs in
4996   let c = match comment with
4997     | CStyle ->         pr "/* "; " *"
4998     | CPlusPlusStyle -> pr "// "; "//"
4999     | HashStyle ->      pr "# ";  "#"
5000     | OCamlStyle ->     pr "(* "; " *"
5001     | HaskellStyle ->   pr "{- "; "  " in
5002   pr "libguestfs generated file\n";
5003   pr "%s WARNING: THIS FILE IS GENERATED FROM:\n" c;
5004   List.iter (pr "%s   %s\n" c) inputs;
5005   pr "%s ANY CHANGES YOU MAKE TO THIS FILE WILL BE LOST.\n" c;
5006   pr "%s\n" c;
5007   pr "%s Copyright (C) %s Red Hat Inc.\n" c copyright_years;
5008   pr "%s\n" c;
5009   (match license with
5010    | GPLv2plus ->
5011        pr "%s This program is free software; you can redistribute it and/or modify\n" c;
5012        pr "%s it under the terms of the GNU General Public License as published by\n" c;
5013        pr "%s the Free Software Foundation; either version 2 of the License, or\n" c;
5014        pr "%s (at your option) any later version.\n" c;
5015        pr "%s\n" c;
5016        pr "%s This program is distributed in the hope that it will be useful,\n" c;
5017        pr "%s but WITHOUT ANY WARRANTY; without even the implied warranty of\n" c;
5018        pr "%s MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\n" c;
5019        pr "%s GNU General Public License for more details.\n" c;
5020        pr "%s\n" c;
5021        pr "%s You should have received a copy of the GNU General Public License along\n" c;
5022        pr "%s with this program; if not, write to the Free Software Foundation, Inc.,\n" c;
5023        pr "%s 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.\n" c;
5024
5025    | LGPLv2plus ->
5026        pr "%s This library is free software; you can redistribute it and/or\n" c;
5027        pr "%s modify it under the terms of the GNU Lesser General Public\n" c;
5028        pr "%s License as published by the Free Software Foundation; either\n" c;
5029        pr "%s version 2 of the License, or (at your option) any later version.\n" c;
5030        pr "%s\n" c;
5031        pr "%s This library is distributed in the hope that it will be useful,\n" c;
5032        pr "%s but WITHOUT ANY WARRANTY; without even the implied warranty of\n" c;
5033        pr "%s MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n" c;
5034        pr "%s Lesser General Public License for more details.\n" c;
5035        pr "%s\n" c;
5036        pr "%s You should have received a copy of the GNU Lesser General Public\n" c;
5037        pr "%s License along with this library; if not, write to the Free Software\n" c;
5038        pr "%s Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA\n" c;
5039   );
5040   (match comment with
5041    | CStyle -> pr " */\n"
5042    | CPlusPlusStyle
5043    | HashStyle -> ()
5044    | OCamlStyle -> pr " *)\n"
5045    | HaskellStyle -> pr "-}\n"
5046   );
5047   pr "\n"
5048
5049 (* Start of main code generation functions below this line. *)
5050
5051 (* Generate the pod documentation for the C API. *)
5052 let rec generate_actions_pod () =
5053   List.iter (
5054     fun (shortname, style, _, flags, _, _, longdesc) ->
5055       if not (List.mem NotInDocs flags) then (
5056         let name = "guestfs_" ^ shortname in
5057         pr "=head2 %s\n\n" name;
5058         pr " ";
5059         generate_prototype ~extern:false ~handle:"handle" name style;
5060         pr "\n\n";
5061         pr "%s\n\n" longdesc;
5062         (match fst style with
5063          | RErr ->
5064              pr "This function returns 0 on success or -1 on error.\n\n"
5065          | RInt _ ->
5066              pr "On error this function returns -1.\n\n"
5067          | RInt64 _ ->
5068              pr "On error this function returns -1.\n\n"
5069          | RBool _ ->
5070              pr "This function returns a C truth value on success or -1 on error.\n\n"
5071          | RConstString _ ->
5072              pr "This function returns a string, or NULL on error.
5073 The string is owned by the guest handle and must I<not> be freed.\n\n"
5074          | RConstOptString _ ->
5075              pr "This function returns a string which may be NULL.
5076 There is way to return an error from this function.
5077 The string is owned by the guest handle and must I<not> be freed.\n\n"
5078          | RString _ ->
5079              pr "This function returns a string, or NULL on error.
5080 I<The caller must free the returned string after use>.\n\n"
5081          | RStringList _ ->
5082              pr "This function returns a NULL-terminated array of strings
5083 (like L<environ(3)>), or NULL if there was an error.
5084 I<The caller must free the strings and the array after use>.\n\n"
5085          | RStruct (_, typ) ->
5086              pr "This function returns a C<struct guestfs_%s *>,
5087 or NULL if there was an error.
5088 I<The caller must call C<guestfs_free_%s> after use>.\n\n" typ typ
5089          | RStructList (_, typ) ->
5090              pr "This function returns a C<struct guestfs_%s_list *>
5091 (see E<lt>guestfs-structs.hE<gt>),
5092 or NULL if there was an error.
5093 I<The caller must call C<guestfs_free_%s_list> after use>.\n\n" typ typ
5094          | RHashtable _ ->
5095              pr "This function returns a NULL-terminated array of
5096 strings, or NULL if there was an error.
5097 The array of strings will always have length C<2n+1>, where
5098 C<n> keys and values alternate, followed by the trailing NULL entry.
5099 I<The caller must free the strings and the array after use>.\n\n"
5100          | RBufferOut _ ->
5101              pr "This function returns a buffer, or NULL on error.
5102 The size of the returned buffer is written to C<*size_r>.
5103 I<The caller must free the returned buffer after use>.\n\n"
5104         );
5105         if List.mem ProtocolLimitWarning flags then
5106           pr "%s\n\n" protocol_limit_warning;
5107         if List.mem DangerWillRobinson flags then
5108           pr "%s\n\n" danger_will_robinson;
5109         match deprecation_notice flags with
5110         | None -> ()
5111         | Some txt -> pr "%s\n\n" txt
5112       )
5113   ) all_functions_sorted
5114
5115 and generate_structs_pod () =
5116   (* Structs documentation. *)
5117   List.iter (
5118     fun (typ, cols) ->
5119       pr "=head2 guestfs_%s\n" typ;
5120       pr "\n";
5121       pr " struct guestfs_%s {\n" typ;
5122       List.iter (
5123         function
5124         | name, FChar -> pr "   char %s;\n" name
5125         | name, FUInt32 -> pr "   uint32_t %s;\n" name
5126         | name, FInt32 -> pr "   int32_t %s;\n" name
5127         | name, (FUInt64|FBytes) -> pr "   uint64_t %s;\n" name
5128         | name, FInt64 -> pr "   int64_t %s;\n" name
5129         | name, FString -> pr "   char *%s;\n" name
5130         | name, FBuffer ->
5131             pr "   /* The next two fields describe a byte array. */\n";
5132             pr "   uint32_t %s_len;\n" name;
5133             pr "   char *%s;\n" name
5134         | name, FUUID ->
5135             pr "   /* The next field is NOT nul-terminated, be careful when printing it: */\n";
5136             pr "   char %s[32];\n" name
5137         | name, FOptPercent ->
5138             pr "   /* The next field is [0..100] or -1 meaning 'not present': */\n";
5139             pr "   float %s;\n" name
5140       ) cols;
5141       pr " };\n";
5142       pr " \n";
5143       pr " struct guestfs_%s_list {\n" typ;
5144       pr "   uint32_t len; /* Number of elements in list. */\n";
5145       pr "   struct guestfs_%s *val; /* Elements. */\n" typ;
5146       pr " };\n";
5147       pr " \n";
5148       pr " void guestfs_free_%s (struct guestfs_free_%s *);\n" typ typ;
5149       pr " void guestfs_free_%s_list (struct guestfs_free_%s_list *);\n"
5150         typ typ;
5151       pr "\n"
5152   ) structs
5153
5154 and generate_availability_pod () =
5155   (* Availability documentation. *)
5156   pr "=over 4\n";
5157   pr "\n";
5158   List.iter (
5159     fun (group, functions) ->
5160       pr "=item B<%s>\n" group;
5161       pr "\n";
5162       pr "The following functions:\n";
5163       List.iter (pr "L</guestfs_%s>\n") functions;
5164       pr "\n"
5165   ) optgroups;
5166   pr "=back\n";
5167   pr "\n"
5168
5169 (* Generate the protocol (XDR) file, 'guestfs_protocol.x' and
5170  * indirectly 'guestfs_protocol.h' and 'guestfs_protocol.c'.
5171  *
5172  * We have to use an underscore instead of a dash because otherwise
5173  * rpcgen generates incorrect code.
5174  *
5175  * This header is NOT exported to clients, but see also generate_structs_h.
5176  *)
5177 and generate_xdr () =
5178   generate_header CStyle LGPLv2plus;
5179
5180   (* This has to be defined to get around a limitation in Sun's rpcgen. *)
5181   pr "typedef string str<>;\n";
5182   pr "\n";
5183
5184   (* Internal structures. *)
5185   List.iter (
5186     function
5187     | typ, cols ->
5188         pr "struct guestfs_int_%s {\n" typ;
5189         List.iter (function
5190                    | name, FChar -> pr "  char %s;\n" name
5191                    | name, FString -> pr "  string %s<>;\n" name
5192                    | name, FBuffer -> pr "  opaque %s<>;\n" name
5193                    | name, FUUID -> pr "  opaque %s[32];\n" name
5194                    | name, (FInt32|FUInt32) -> pr "  int %s;\n" name
5195                    | name, (FInt64|FUInt64|FBytes) -> pr "  hyper %s;\n" name
5196                    | name, FOptPercent -> pr "  float %s;\n" name
5197                   ) cols;
5198         pr "};\n";
5199         pr "\n";
5200         pr "typedef struct guestfs_int_%s guestfs_int_%s_list<>;\n" typ typ;
5201         pr "\n";
5202   ) structs;
5203
5204   List.iter (
5205     fun (shortname, style, _, _, _, _, _) ->
5206       let name = "guestfs_" ^ shortname in
5207
5208       (match snd style with
5209        | [] -> ()
5210        | args ->
5211            pr "struct %s_args {\n" name;
5212            List.iter (
5213              function
5214              | Pathname n | Device n | Dev_or_Path n | String n ->
5215                  pr "  string %s<>;\n" n
5216              | OptString n -> pr "  str *%s;\n" n
5217              | StringList n | DeviceList n -> pr "  str %s<>;\n" n
5218              | Bool n -> pr "  bool %s;\n" n
5219              | Int n -> pr "  int %s;\n" n
5220              | Int64 n -> pr "  hyper %s;\n" n
5221              | FileIn _ | FileOut _ -> ()
5222            ) args;
5223            pr "};\n\n"
5224       );
5225       (match fst style with
5226        | RErr -> ()
5227        | RInt n ->
5228            pr "struct %s_ret {\n" name;
5229            pr "  int %s;\n" n;
5230            pr "};\n\n"
5231        | RInt64 n ->
5232            pr "struct %s_ret {\n" name;
5233            pr "  hyper %s;\n" n;
5234            pr "};\n\n"
5235        | RBool n ->
5236            pr "struct %s_ret {\n" name;
5237            pr "  bool %s;\n" n;
5238            pr "};\n\n"
5239        | RConstString _ | RConstOptString _ ->
5240            failwithf "RConstString|RConstOptString cannot be used by daemon functions"
5241        | RString n ->
5242            pr "struct %s_ret {\n" name;
5243            pr "  string %s<>;\n" n;
5244            pr "};\n\n"
5245        | RStringList n ->
5246            pr "struct %s_ret {\n" name;
5247            pr "  str %s<>;\n" n;
5248            pr "};\n\n"
5249        | RStruct (n, typ) ->
5250            pr "struct %s_ret {\n" name;
5251            pr "  guestfs_int_%s %s;\n" typ n;
5252            pr "};\n\n"
5253        | RStructList (n, typ) ->
5254            pr "struct %s_ret {\n" name;
5255            pr "  guestfs_int_%s_list %s;\n" typ n;
5256            pr "};\n\n"
5257        | RHashtable n ->
5258            pr "struct %s_ret {\n" name;
5259            pr "  str %s<>;\n" n;
5260            pr "};\n\n"
5261        | RBufferOut n ->
5262            pr "struct %s_ret {\n" name;
5263            pr "  opaque %s<>;\n" n;
5264            pr "};\n\n"
5265       );
5266   ) daemon_functions;
5267
5268   (* Table of procedure numbers. *)
5269   pr "enum guestfs_procedure {\n";
5270   List.iter (
5271     fun (shortname, _, proc_nr, _, _, _, _) ->
5272       pr "  GUESTFS_PROC_%s = %d,\n" (String.uppercase shortname) proc_nr
5273   ) daemon_functions;
5274   pr "  GUESTFS_PROC_NR_PROCS\n";
5275   pr "};\n";
5276   pr "\n";
5277
5278   (* Having to choose a maximum message size is annoying for several
5279    * reasons (it limits what we can do in the API), but it (a) makes
5280    * the protocol a lot simpler, and (b) provides a bound on the size
5281    * of the daemon which operates in limited memory space.
5282    *)
5283   pr "const GUESTFS_MESSAGE_MAX = %d;\n" (4 * 1024 * 1024);
5284   pr "\n";
5285
5286   (* Message header, etc. *)
5287   pr "\
5288 /* The communication protocol is now documented in the guestfs(3)
5289  * manpage.
5290  */
5291
5292 const GUESTFS_PROGRAM = 0x2000F5F5;
5293 const GUESTFS_PROTOCOL_VERSION = 1;
5294
5295 /* These constants must be larger than any possible message length. */
5296 const GUESTFS_LAUNCH_FLAG = 0xf5f55ff5;
5297 const GUESTFS_CANCEL_FLAG = 0xffffeeee;
5298
5299 enum guestfs_message_direction {
5300   GUESTFS_DIRECTION_CALL = 0,        /* client -> daemon */
5301   GUESTFS_DIRECTION_REPLY = 1        /* daemon -> client */
5302 };
5303
5304 enum guestfs_message_status {
5305   GUESTFS_STATUS_OK = 0,
5306   GUESTFS_STATUS_ERROR = 1
5307 };
5308
5309 const GUESTFS_ERROR_LEN = 256;
5310
5311 struct guestfs_message_error {
5312   string error_message<GUESTFS_ERROR_LEN>;
5313 };
5314
5315 struct guestfs_message_header {
5316   unsigned prog;                     /* GUESTFS_PROGRAM */
5317   unsigned vers;                     /* GUESTFS_PROTOCOL_VERSION */
5318   guestfs_procedure proc;            /* GUESTFS_PROC_x */
5319   guestfs_message_direction direction;
5320   unsigned serial;                   /* message serial number */
5321   guestfs_message_status status;
5322 };
5323
5324 const GUESTFS_MAX_CHUNK_SIZE = 8192;
5325
5326 struct guestfs_chunk {
5327   int cancel;                        /* if non-zero, transfer is cancelled */
5328   /* data size is 0 bytes if the transfer has finished successfully */
5329   opaque data<GUESTFS_MAX_CHUNK_SIZE>;
5330 };
5331 "
5332
5333 (* Generate the guestfs-structs.h file. *)
5334 and generate_structs_h () =
5335   generate_header CStyle LGPLv2plus;
5336
5337   (* This is a public exported header file containing various
5338    * structures.  The structures are carefully written to have
5339    * exactly the same in-memory format as the XDR structures that
5340    * we use on the wire to the daemon.  The reason for creating
5341    * copies of these structures here is just so we don't have to
5342    * export the whole of guestfs_protocol.h (which includes much
5343    * unrelated and XDR-dependent stuff that we don't want to be
5344    * public, or required by clients).
5345    *
5346    * To reiterate, we will pass these structures to and from the
5347    * client with a simple assignment or memcpy, so the format
5348    * must be identical to what rpcgen / the RFC defines.
5349    *)
5350
5351   (* Public structures. *)
5352   List.iter (
5353     fun (typ, cols) ->
5354       pr "struct guestfs_%s {\n" typ;
5355       List.iter (
5356         function
5357         | name, FChar -> pr "  char %s;\n" name
5358         | name, FString -> pr "  char *%s;\n" name
5359         | name, FBuffer ->
5360             pr "  uint32_t %s_len;\n" name;
5361             pr "  char *%s;\n" name
5362         | name, FUUID -> pr "  char %s[32]; /* this is NOT nul-terminated, be careful when printing */\n" name
5363         | name, FUInt32 -> pr "  uint32_t %s;\n" name
5364         | name, FInt32 -> pr "  int32_t %s;\n" name
5365         | name, (FUInt64|FBytes) -> pr "  uint64_t %s;\n" name
5366         | name, FInt64 -> pr "  int64_t %s;\n" name
5367         | name, FOptPercent -> pr "  float %s; /* [0..100] or -1 */\n" name
5368       ) cols;
5369       pr "};\n";
5370       pr "\n";
5371       pr "struct guestfs_%s_list {\n" typ;
5372       pr "  uint32_t len;\n";
5373       pr "  struct guestfs_%s *val;\n" typ;
5374       pr "};\n";
5375       pr "\n";
5376       pr "extern void guestfs_free_%s (struct guestfs_%s *);\n" typ typ;
5377       pr "extern void guestfs_free_%s_list (struct guestfs_%s_list *);\n" typ typ;
5378       pr "\n"
5379   ) structs
5380
5381 (* Generate the guestfs-actions.h file. *)
5382 and generate_actions_h () =
5383   generate_header CStyle LGPLv2plus;
5384   List.iter (
5385     fun (shortname, style, _, _, _, _, _) ->
5386       let name = "guestfs_" ^ shortname in
5387       generate_prototype ~single_line:true ~newline:true ~handle:"handle"
5388         name style
5389   ) all_functions
5390
5391 (* Generate the guestfs-internal-actions.h file. *)
5392 and generate_internal_actions_h () =
5393   generate_header CStyle LGPLv2plus;
5394   List.iter (
5395     fun (shortname, style, _, _, _, _, _) ->
5396       let name = "guestfs__" ^ shortname in
5397       generate_prototype ~single_line:true ~newline:true ~handle:"handle"
5398         name style
5399   ) non_daemon_functions
5400
5401 (* Generate the client-side dispatch stubs. *)
5402 and generate_client_actions () =
5403   generate_header CStyle LGPLv2plus;
5404
5405   pr "\
5406 #include <stdio.h>
5407 #include <stdlib.h>
5408 #include <stdint.h>
5409 #include <string.h>
5410 #include <inttypes.h>
5411
5412 #include \"guestfs.h\"
5413 #include \"guestfs-internal.h\"
5414 #include \"guestfs-internal-actions.h\"
5415 #include \"guestfs_protocol.h\"
5416
5417 #define error guestfs_error
5418 //#define perrorf guestfs_perrorf
5419 #define safe_malloc guestfs_safe_malloc
5420 #define safe_realloc guestfs_safe_realloc
5421 //#define safe_strdup guestfs_safe_strdup
5422 #define safe_memdup guestfs_safe_memdup
5423
5424 /* Check the return message from a call for validity. */
5425 static int
5426 check_reply_header (guestfs_h *g,
5427                     const struct guestfs_message_header *hdr,
5428                     unsigned int proc_nr, unsigned int serial)
5429 {
5430   if (hdr->prog != GUESTFS_PROGRAM) {
5431     error (g, \"wrong program (%%d/%%d)\", hdr->prog, GUESTFS_PROGRAM);
5432     return -1;
5433   }
5434   if (hdr->vers != GUESTFS_PROTOCOL_VERSION) {
5435     error (g, \"wrong protocol version (%%d/%%d)\",
5436            hdr->vers, GUESTFS_PROTOCOL_VERSION);
5437     return -1;
5438   }
5439   if (hdr->direction != GUESTFS_DIRECTION_REPLY) {
5440     error (g, \"unexpected message direction (%%d/%%d)\",
5441            hdr->direction, GUESTFS_DIRECTION_REPLY);
5442     return -1;
5443   }
5444   if (hdr->proc != proc_nr) {
5445     error (g, \"unexpected procedure number (%%d/%%d)\", hdr->proc, proc_nr);
5446     return -1;
5447   }
5448   if (hdr->serial != serial) {
5449     error (g, \"unexpected serial (%%d/%%d)\", hdr->serial, serial);
5450     return -1;
5451   }
5452
5453   return 0;
5454 }
5455
5456 /* Check we are in the right state to run a high-level action. */
5457 static int
5458 check_state (guestfs_h *g, const char *caller)
5459 {
5460   if (!guestfs__is_ready (g)) {
5461     if (guestfs__is_config (g) || guestfs__is_launching (g))
5462       error (g, \"%%s: call launch before using this function\\n(in guestfish, don't forget to use the 'run' command)\",
5463         caller);
5464     else
5465       error (g, \"%%s called from the wrong state, %%d != READY\",
5466         caller, guestfs__get_state (g));
5467     return -1;
5468   }
5469   return 0;
5470 }
5471
5472 ";
5473
5474   (* Generate code to generate guestfish call traces. *)
5475   let trace_call shortname style =
5476     pr "  if (guestfs__get_trace (g)) {\n";
5477
5478     let needs_i =
5479       List.exists (function
5480                    | StringList _ | DeviceList _ -> true
5481                    | _ -> false) (snd style) in
5482     if needs_i then (
5483       pr "    int i;\n";
5484       pr "\n"
5485     );
5486
5487     pr "    printf (\"%s\");\n" shortname;
5488     List.iter (
5489       function
5490       | String n                        (* strings *)
5491       | Device n
5492       | Pathname n
5493       | Dev_or_Path n
5494       | FileIn n
5495       | FileOut n ->
5496           (* guestfish doesn't support string escaping, so neither do we *)
5497           pr "    printf (\" \\\"%%s\\\"\", %s);\n" n
5498       | OptString n ->                  (* string option *)
5499           pr "    if (%s) printf (\" \\\"%%s\\\"\", %s);\n" n n;
5500           pr "    else printf (\" null\");\n"
5501       | StringList n
5502       | DeviceList n ->                 (* string list *)
5503           pr "    putchar (' ');\n";
5504           pr "    putchar ('\"');\n";
5505           pr "    for (i = 0; %s[i]; ++i) {\n" n;
5506           pr "      if (i > 0) putchar (' ');\n";
5507           pr "      fputs (%s[i], stdout);\n" n;
5508           pr "    }\n";
5509           pr "    putchar ('\"');\n";
5510       | Bool n ->                       (* boolean *)
5511           pr "    fputs (%s ? \" true\" : \" false\", stdout);\n" n
5512       | Int n ->                        (* int *)
5513           pr "    printf (\" %%d\", %s);\n" n
5514       | Int64 n ->
5515           pr "    printf (\" %%\" PRIi64, %s);\n" n
5516     ) (snd style);
5517     pr "    putchar ('\\n');\n";
5518     pr "  }\n";
5519     pr "\n";
5520   in
5521
5522   (* For non-daemon functions, generate a wrapper around each function. *)
5523   List.iter (
5524     fun (shortname, style, _, _, _, _, _) ->
5525       let name = "guestfs_" ^ shortname in
5526
5527       generate_prototype ~extern:false ~semicolon:false ~newline:true
5528         ~handle:"g" name style;
5529       pr "{\n";
5530       trace_call shortname style;
5531       pr "  return guestfs__%s " shortname;
5532       generate_c_call_args ~handle:"g" style;
5533       pr ";\n";
5534       pr "}\n";
5535       pr "\n"
5536   ) non_daemon_functions;
5537
5538   (* Client-side stubs for each function. *)
5539   List.iter (
5540     fun (shortname, style, _, _, _, _, _) ->
5541       let name = "guestfs_" ^ shortname in
5542
5543       (* Generate the action stub. *)
5544       generate_prototype ~extern:false ~semicolon:false ~newline:true
5545         ~handle:"g" name style;
5546
5547       let error_code =
5548         match fst style with
5549         | RErr | RInt _ | RInt64 _ | RBool _ -> "-1"
5550         | RConstString _ | RConstOptString _ ->
5551             failwithf "RConstString|RConstOptString cannot be used by daemon functions"
5552         | RString _ | RStringList _
5553         | RStruct _ | RStructList _
5554         | RHashtable _ | RBufferOut _ ->
5555             "NULL" in
5556
5557       pr "{\n";
5558
5559       (match snd style with
5560        | [] -> ()
5561        | _ -> pr "  struct %s_args args;\n" name
5562       );
5563
5564       pr "  guestfs_message_header hdr;\n";
5565       pr "  guestfs_message_error err;\n";
5566       let has_ret =
5567         match fst style with
5568         | RErr -> false
5569         | RConstString _ | RConstOptString _ ->
5570             failwithf "RConstString|RConstOptString cannot be used by daemon functions"
5571         | RInt _ | RInt64 _
5572         | RBool _ | RString _ | RStringList _
5573         | RStruct _ | RStructList _
5574         | RHashtable _ | RBufferOut _ ->
5575             pr "  struct %s_ret ret;\n" name;
5576             true in
5577
5578       pr "  int serial;\n";
5579       pr "  int r;\n";
5580       pr "\n";
5581       trace_call shortname style;
5582       pr "  if (check_state (g, \"%s\") == -1) return %s;\n" name error_code;
5583       pr "  guestfs___set_busy (g);\n";
5584       pr "\n";
5585
5586       (* Send the main header and arguments. *)
5587       (match snd style with
5588        | [] ->
5589            pr "  serial = guestfs___send (g, GUESTFS_PROC_%s, NULL, NULL);\n"
5590              (String.uppercase shortname)
5591        | args ->
5592            List.iter (
5593              function
5594              | Pathname n | Device n | Dev_or_Path n | String n ->
5595                  pr "  args.%s = (char *) %s;\n" n n
5596              | OptString n ->
5597                  pr "  args.%s = %s ? (char **) &%s : NULL;\n" n n n
5598              | StringList n | DeviceList n ->
5599                  pr "  args.%s.%s_val = (char **) %s;\n" n n n;
5600                  pr "  for (args.%s.%s_len = 0; %s[args.%s.%s_len]; args.%s.%s_len++) ;\n" n n n n n n n;
5601              | Bool n ->
5602                  pr "  args.%s = %s;\n" n n
5603              | Int n ->
5604                  pr "  args.%s = %s;\n" n n
5605              | Int64 n ->
5606                  pr "  args.%s = %s;\n" n n
5607              | FileIn _ | FileOut _ -> ()
5608            ) args;
5609            pr "  serial = guestfs___send (g, GUESTFS_PROC_%s,\n"
5610              (String.uppercase shortname);
5611            pr "        (xdrproc_t) xdr_%s_args, (char *) &args);\n"
5612              name;
5613       );
5614       pr "  if (serial == -1) {\n";
5615       pr "    guestfs___end_busy (g);\n";
5616       pr "    return %s;\n" error_code;
5617       pr "  }\n";
5618       pr "\n";
5619
5620       (* Send any additional files (FileIn) requested. *)
5621       let need_read_reply_label = ref false in
5622       List.iter (
5623         function
5624         | FileIn n ->
5625             pr "  r = guestfs___send_file (g, %s);\n" n;
5626             pr "  if (r == -1) {\n";
5627             pr "    guestfs___end_busy (g);\n";
5628             pr "    return %s;\n" error_code;
5629             pr "  }\n";
5630             pr "  if (r == -2) /* daemon cancelled */\n";
5631             pr "    goto read_reply;\n";
5632             need_read_reply_label := true;
5633             pr "\n";
5634         | _ -> ()
5635       ) (snd style);
5636
5637       (* Wait for the reply from the remote end. *)
5638       if !need_read_reply_label then pr " read_reply:\n";
5639       pr "  memset (&hdr, 0, sizeof hdr);\n";
5640       pr "  memset (&err, 0, sizeof err);\n";
5641       if has_ret then pr "  memset (&ret, 0, sizeof ret);\n";
5642       pr "\n";
5643       pr "  r = guestfs___recv (g, \"%s\", &hdr, &err,\n        " shortname;
5644       if not has_ret then
5645         pr "NULL, NULL"
5646       else
5647         pr "(xdrproc_t) xdr_guestfs_%s_ret, (char *) &ret" shortname;
5648       pr ");\n";
5649
5650       pr "  if (r == -1) {\n";
5651       pr "    guestfs___end_busy (g);\n";
5652       pr "    return %s;\n" error_code;
5653       pr "  }\n";
5654       pr "\n";
5655
5656       pr "  if (check_reply_header (g, &hdr, GUESTFS_PROC_%s, serial) == -1) {\n"
5657         (String.uppercase shortname);
5658       pr "    guestfs___end_busy (g);\n";
5659       pr "    return %s;\n" error_code;
5660       pr "  }\n";
5661       pr "\n";
5662
5663       pr "  if (hdr.status == GUESTFS_STATUS_ERROR) {\n";
5664       pr "    error (g, \"%%s: %%s\", \"%s\", err.error_message);\n" shortname;
5665       pr "    free (err.error_message);\n";
5666       pr "    guestfs___end_busy (g);\n";
5667       pr "    return %s;\n" error_code;
5668       pr "  }\n";
5669       pr "\n";
5670
5671       (* Expecting to receive further files (FileOut)? *)
5672       List.iter (
5673         function
5674         | FileOut n ->
5675             pr "  if (guestfs___recv_file (g, %s) == -1) {\n" n;
5676             pr "    guestfs___end_busy (g);\n";
5677             pr "    return %s;\n" error_code;
5678             pr "  }\n";
5679             pr "\n";
5680         | _ -> ()
5681       ) (snd style);
5682
5683       pr "  guestfs___end_busy (g);\n";
5684
5685       (match fst style with
5686        | RErr -> pr "  return 0;\n"
5687        | RInt n | RInt64 n | RBool n ->
5688            pr "  return ret.%s;\n" n
5689        | RConstString _ | RConstOptString _ ->
5690            failwithf "RConstString|RConstOptString cannot be used by daemon functions"
5691        | RString n ->
5692            pr "  return ret.%s; /* caller will free */\n" n
5693        | RStringList n | RHashtable n ->
5694            pr "  /* caller will free this, but we need to add a NULL entry */\n";
5695            pr "  ret.%s.%s_val =\n" n n;
5696            pr "    safe_realloc (g, ret.%s.%s_val,\n" n n;
5697            pr "                  sizeof (char *) * (ret.%s.%s_len + 1));\n"
5698              n n;
5699            pr "  ret.%s.%s_val[ret.%s.%s_len] = NULL;\n" n n n n;
5700            pr "  return ret.%s.%s_val;\n" n n
5701        | RStruct (n, _) ->
5702            pr "  /* caller will free this */\n";
5703            pr "  return safe_memdup (g, &ret.%s, sizeof (ret.%s));\n" n n
5704        | RStructList (n, _) ->
5705            pr "  /* caller will free this */\n";
5706            pr "  return safe_memdup (g, &ret.%s, sizeof (ret.%s));\n" n n
5707        | RBufferOut n ->
5708            pr "  /* RBufferOut is tricky: If the buffer is zero-length, then\n";
5709            pr "   * _val might be NULL here.  To make the API saner for\n";
5710            pr "   * callers, we turn this case into a unique pointer (using\n";
5711            pr "   * malloc(1)).\n";
5712            pr "   */\n";
5713            pr "  if (ret.%s.%s_len > 0) {\n" n n;
5714            pr "    *size_r = ret.%s.%s_len;\n" n n;
5715            pr "    return ret.%s.%s_val; /* caller will free */\n" n n;
5716            pr "  } else {\n";
5717            pr "    free (ret.%s.%s_val);\n" n n;
5718            pr "    char *p = safe_malloc (g, 1);\n";
5719            pr "    *size_r = ret.%s.%s_len;\n" n n;
5720            pr "    return p;\n";
5721            pr "  }\n";
5722       );
5723
5724       pr "}\n\n"
5725   ) daemon_functions;
5726
5727   (* Functions to free structures. *)
5728   pr "/* Structure-freeing functions.  These rely on the fact that the\n";
5729   pr " * structure format is identical to the XDR format.  See note in\n";
5730   pr " * generator.ml.\n";
5731   pr " */\n";
5732   pr "\n";
5733
5734   List.iter (
5735     fun (typ, _) ->
5736       pr "void\n";
5737       pr "guestfs_free_%s (struct guestfs_%s *x)\n" typ typ;
5738       pr "{\n";
5739       pr "  xdr_free ((xdrproc_t) xdr_guestfs_int_%s, (char *) x);\n" typ;
5740       pr "  free (x);\n";
5741       pr "}\n";
5742       pr "\n";
5743
5744       pr "void\n";
5745       pr "guestfs_free_%s_list (struct guestfs_%s_list *x)\n" typ typ;
5746       pr "{\n";
5747       pr "  xdr_free ((xdrproc_t) xdr_guestfs_int_%s_list, (char *) x);\n" typ;
5748       pr "  free (x);\n";
5749       pr "}\n";
5750       pr "\n";
5751
5752   ) structs;
5753
5754 (* Generate daemon/actions.h. *)
5755 and generate_daemon_actions_h () =
5756   generate_header CStyle GPLv2plus;
5757
5758   pr "#include \"../src/guestfs_protocol.h\"\n";
5759   pr "\n";
5760
5761   List.iter (
5762     fun (name, style, _, _, _, _, _) ->
5763       generate_prototype
5764         ~single_line:true ~newline:true ~in_daemon:true ~prefix:"do_"
5765         name style;
5766   ) daemon_functions
5767
5768 (* Generate the linker script which controls the visibility of
5769  * symbols in the public ABI and ensures no other symbols get
5770  * exported accidentally.
5771  *)
5772 and generate_linker_script () =
5773   generate_header HashStyle GPLv2plus;
5774
5775   let globals = [
5776     "guestfs_create";
5777     "guestfs_close";
5778     "guestfs_get_error_handler";
5779     "guestfs_get_out_of_memory_handler";
5780     "guestfs_last_error";
5781     "guestfs_set_error_handler";
5782     "guestfs_set_launch_done_callback";
5783     "guestfs_set_log_message_callback";
5784     "guestfs_set_out_of_memory_handler";
5785     "guestfs_set_subprocess_quit_callback";
5786
5787     (* Unofficial parts of the API: the bindings code use these
5788      * functions, so it is useful to export them.
5789      *)
5790     "guestfs_safe_calloc";
5791     "guestfs_safe_malloc";
5792   ] in
5793   let functions =
5794     List.map (fun (name, _, _, _, _, _, _) -> "guestfs_" ^ name)
5795       all_functions in
5796   let structs =
5797     List.concat (
5798       List.map (fun (typ, _) ->
5799                   ["guestfs_free_" ^ typ; "guestfs_free_" ^ typ ^ "_list"])
5800         structs
5801     ) in
5802   let globals = List.sort compare (globals @ functions @ structs) in
5803
5804   pr "{\n";
5805   pr "    global:\n";
5806   List.iter (pr "        %s;\n") globals;
5807   pr "\n";
5808
5809   pr "    local:\n";
5810   pr "        *;\n";
5811   pr "};\n"
5812
5813 (* Generate the server-side stubs. *)
5814 and generate_daemon_actions () =
5815   generate_header CStyle GPLv2plus;
5816
5817   pr "#include <config.h>\n";
5818   pr "\n";
5819   pr "#include <stdio.h>\n";
5820   pr "#include <stdlib.h>\n";
5821   pr "#include <string.h>\n";
5822   pr "#include <inttypes.h>\n";
5823   pr "#include <rpc/types.h>\n";
5824   pr "#include <rpc/xdr.h>\n";
5825   pr "\n";
5826   pr "#include \"daemon.h\"\n";
5827   pr "#include \"c-ctype.h\"\n";
5828   pr "#include \"../src/guestfs_protocol.h\"\n";
5829   pr "#include \"actions.h\"\n";
5830   pr "\n";
5831
5832   List.iter (
5833     fun (name, style, _, _, _, _, _) ->
5834       (* Generate server-side stubs. *)
5835       pr "static void %s_stub (XDR *xdr_in)\n" name;
5836       pr "{\n";
5837       let error_code =
5838         match fst style with
5839         | RErr | RInt _ -> pr "  int r;\n"; "-1"
5840         | RInt64 _ -> pr "  int64_t r;\n"; "-1"
5841         | RBool _ -> pr "  int r;\n"; "-1"
5842         | RConstString _ | RConstOptString _ ->
5843             failwithf "RConstString|RConstOptString cannot be used by daemon functions"
5844         | RString _ -> pr "  char *r;\n"; "NULL"
5845         | RStringList _ | RHashtable _ -> pr "  char **r;\n"; "NULL"
5846         | RStruct (_, typ) -> pr "  guestfs_int_%s *r;\n" typ; "NULL"
5847         | RStructList (_, typ) -> pr "  guestfs_int_%s_list *r;\n" typ; "NULL"
5848         | RBufferOut _ ->
5849             pr "  size_t size = 1;\n";
5850             pr "  char *r;\n";
5851             "NULL" in
5852
5853       (match snd style with
5854        | [] -> ()
5855        | args ->
5856            pr "  struct guestfs_%s_args args;\n" name;
5857            List.iter (
5858              function
5859              | Device n | Dev_or_Path n
5860              | Pathname n
5861              | String n -> ()
5862              | OptString n -> pr "  char *%s;\n" n
5863              | StringList n | DeviceList n -> pr "  char **%s;\n" n
5864              | Bool n -> pr "  int %s;\n" n
5865              | Int n -> pr "  int %s;\n" n
5866              | Int64 n -> pr "  int64_t %s;\n" n
5867              | FileIn _ | FileOut _ -> ()
5868            ) args
5869       );
5870       pr "\n";
5871
5872       (match snd style with
5873        | [] -> ()
5874        | args ->
5875            pr "  memset (&args, 0, sizeof args);\n";
5876            pr "\n";
5877            pr "  if (!xdr_guestfs_%s_args (xdr_in, &args)) {\n" name;
5878            pr "    reply_with_error (\"daemon failed to decode procedure arguments\");\n";
5879            pr "    return;\n";
5880            pr "  }\n";
5881            let pr_args n =
5882              pr "  char *%s = args.%s;\n" n n
5883            in
5884            let pr_list_handling_code n =
5885              pr "  %s = realloc (args.%s.%s_val,\n" n n n;
5886              pr "                sizeof (char *) * (args.%s.%s_len+1));\n" n n;
5887              pr "  if (%s == NULL) {\n" n;
5888              pr "    reply_with_perror (\"realloc\");\n";
5889              pr "    goto done;\n";
5890              pr "  }\n";
5891              pr "  %s[args.%s.%s_len] = NULL;\n" n n n;
5892              pr "  args.%s.%s_val = %s;\n" n n n;
5893            in
5894            List.iter (
5895              function
5896              | Pathname n ->
5897                  pr_args n;
5898                  pr "  ABS_PATH (%s, goto done);\n" n;
5899              | Device n ->
5900                  pr_args n;
5901                  pr "  RESOLVE_DEVICE (%s, goto done);\n" n;
5902              | Dev_or_Path n ->
5903                  pr_args n;
5904                  pr "  REQUIRE_ROOT_OR_RESOLVE_DEVICE (%s, goto done);\n" n;
5905              | String n -> pr_args n
5906              | OptString n -> pr "  %s = args.%s ? *args.%s : NULL;\n" n n n
5907              | StringList n ->
5908                  pr_list_handling_code n;
5909              | DeviceList n ->
5910                  pr_list_handling_code n;
5911                  pr "  /* Ensure that each is a device,\n";
5912                  pr "   * and perform device name translation. */\n";
5913                  pr "  { int pvi; for (pvi = 0; physvols[pvi] != NULL; ++pvi)\n";
5914                  pr "    RESOLVE_DEVICE (physvols[pvi], goto done);\n";
5915                  pr "  }\n";
5916              | Bool n -> pr "  %s = args.%s;\n" n n
5917              | Int n -> pr "  %s = args.%s;\n" n n
5918              | Int64 n -> pr "  %s = args.%s;\n" n n
5919              | FileIn _ | FileOut _ -> ()
5920            ) args;
5921            pr "\n"
5922       );
5923
5924
5925       (* this is used at least for do_equal *)
5926       if List.exists (function Pathname _ -> true | _ -> false) (snd style) then (
5927         (* Emit NEED_ROOT just once, even when there are two or
5928            more Pathname args *)
5929         pr "  NEED_ROOT (goto done);\n";
5930       );
5931
5932       (* Don't want to call the impl with any FileIn or FileOut
5933        * parameters, since these go "outside" the RPC protocol.
5934        *)
5935       let args' =
5936         List.filter (function FileIn _ | FileOut _ -> false | _ -> true)
5937           (snd style) in
5938       pr "  r = do_%s " name;
5939       generate_c_call_args (fst style, args');
5940       pr ";\n";
5941
5942       (match fst style with
5943        | RErr | RInt _ | RInt64 _ | RBool _
5944        | RConstString _ | RConstOptString _
5945        | RString _ | RStringList _ | RHashtable _
5946        | RStruct (_, _) | RStructList (_, _) ->
5947            pr "  if (r == %s)\n" error_code;
5948            pr "    /* do_%s has already called reply_with_error */\n" name;
5949            pr "    goto done;\n";
5950            pr "\n"
5951        | RBufferOut _ ->
5952            pr "  /* size == 0 && r == NULL could be a non-error case (just\n";
5953            pr "   * an ordinary zero-length buffer), so be careful ...\n";
5954            pr "   */\n";
5955            pr "  if (size == 1 && r == %s)\n" error_code;
5956            pr "    /* do_%s has already called reply_with_error */\n" name;
5957            pr "    goto done;\n";
5958            pr "\n"
5959       );
5960
5961       (* If there are any FileOut parameters, then the impl must
5962        * send its own reply.
5963        *)
5964       let no_reply =
5965         List.exists (function FileOut _ -> true | _ -> false) (snd style) in
5966       if no_reply then
5967         pr "  /* do_%s has already sent a reply */\n" name
5968       else (
5969         match fst style with
5970         | RErr -> pr "  reply (NULL, NULL);\n"
5971         | RInt n | RInt64 n | RBool n ->
5972             pr "  struct guestfs_%s_ret ret;\n" name;
5973             pr "  ret.%s = r;\n" n;
5974             pr "  reply ((xdrproc_t) &xdr_guestfs_%s_ret, (char *) &ret);\n"
5975               name
5976         | RConstString _ | RConstOptString _ ->
5977             failwithf "RConstString|RConstOptString cannot be used by daemon functions"
5978         | RString n ->
5979             pr "  struct guestfs_%s_ret ret;\n" name;
5980             pr "  ret.%s = r;\n" n;
5981             pr "  reply ((xdrproc_t) &xdr_guestfs_%s_ret, (char *) &ret);\n"
5982               name;
5983             pr "  free (r);\n"
5984         | RStringList n | RHashtable n ->
5985             pr "  struct guestfs_%s_ret ret;\n" name;
5986             pr "  ret.%s.%s_len = count_strings (r);\n" n n;
5987             pr "  ret.%s.%s_val = r;\n" n n;
5988             pr "  reply ((xdrproc_t) &xdr_guestfs_%s_ret, (char *) &ret);\n"
5989               name;
5990             pr "  free_strings (r);\n"
5991         | RStruct (n, _) ->
5992             pr "  struct guestfs_%s_ret ret;\n" name;
5993             pr "  ret.%s = *r;\n" n;
5994             pr "  reply ((xdrproc_t) xdr_guestfs_%s_ret, (char *) &ret);\n"
5995               name;
5996             pr "  xdr_free ((xdrproc_t) xdr_guestfs_%s_ret, (char *) &ret);\n"
5997               name
5998         | RStructList (n, _) ->
5999             pr "  struct guestfs_%s_ret ret;\n" name;
6000             pr "  ret.%s = *r;\n" n;
6001             pr "  reply ((xdrproc_t) xdr_guestfs_%s_ret, (char *) &ret);\n"
6002               name;
6003             pr "  xdr_free ((xdrproc_t) xdr_guestfs_%s_ret, (char *) &ret);\n"
6004               name
6005         | RBufferOut n ->
6006             pr "  struct guestfs_%s_ret ret;\n" name;
6007             pr "  ret.%s.%s_val = r;\n" n n;
6008             pr "  ret.%s.%s_len = size;\n" n n;
6009             pr "  reply ((xdrproc_t) &xdr_guestfs_%s_ret, (char *) &ret);\n"
6010               name;
6011             pr "  free (r);\n"
6012       );
6013
6014       (* Free the args. *)
6015       (match snd style with
6016        | [] ->
6017            pr "done: ;\n";
6018        | _ ->
6019            pr "done:\n";
6020            pr "  xdr_free ((xdrproc_t) xdr_guestfs_%s_args, (char *) &args);\n"
6021              name
6022       );
6023
6024       pr "}\n\n";
6025   ) daemon_functions;
6026
6027   (* Dispatch function. *)
6028   pr "void dispatch_incoming_message (XDR *xdr_in)\n";
6029   pr "{\n";
6030   pr "  switch (proc_nr) {\n";
6031
6032   List.iter (
6033     fun (name, style, _, _, _, _, _) ->
6034       pr "    case GUESTFS_PROC_%s:\n" (String.uppercase name);
6035       pr "      %s_stub (xdr_in);\n" name;
6036       pr "      break;\n"
6037   ) daemon_functions;
6038
6039   pr "    default:\n";
6040   pr "      reply_with_error (\"dispatch_incoming_message: unknown procedure number %%d, set LIBGUESTFS_PATH to point to the matching libguestfs appliance directory\", proc_nr);\n";
6041   pr "  }\n";
6042   pr "}\n";
6043   pr "\n";
6044
6045   (* LVM columns and tokenization functions. *)
6046   (* XXX This generates crap code.  We should rethink how we
6047    * do this parsing.
6048    *)
6049   List.iter (
6050     function
6051     | typ, cols ->
6052         pr "static const char *lvm_%s_cols = \"%s\";\n"
6053           typ (String.concat "," (List.map fst cols));
6054         pr "\n";
6055
6056         pr "static int lvm_tokenize_%s (char *str, guestfs_int_lvm_%s *r)\n" typ typ;
6057         pr "{\n";
6058         pr "  char *tok, *p, *next;\n";
6059         pr "  int i, j;\n";
6060         pr "\n";
6061         (*
6062           pr "  fprintf (stderr, \"%%s: <<%%s>>\\n\", __func__, str);\n";
6063           pr "\n";
6064         *)
6065         pr "  if (!str) {\n";
6066         pr "    fprintf (stderr, \"%%s: failed: passed a NULL string\\n\", __func__);\n";
6067         pr "    return -1;\n";
6068         pr "  }\n";
6069         pr "  if (!*str || c_isspace (*str)) {\n";
6070         pr "    fprintf (stderr, \"%%s: failed: passed a empty string or one beginning with whitespace\\n\", __func__);\n";
6071         pr "    return -1;\n";
6072         pr "  }\n";
6073         pr "  tok = str;\n";
6074         List.iter (
6075           fun (name, coltype) ->
6076             pr "  if (!tok) {\n";
6077             pr "    fprintf (stderr, \"%%s: failed: string finished early, around token %%s\\n\", __func__, \"%s\");\n" name;
6078             pr "    return -1;\n";
6079             pr "  }\n";
6080             pr "  p = strchrnul (tok, ',');\n";
6081             pr "  if (*p) next = p+1; else next = NULL;\n";
6082             pr "  *p = '\\0';\n";
6083             (match coltype with
6084              | FString ->
6085                  pr "  r->%s = strdup (tok);\n" name;
6086                  pr "  if (r->%s == NULL) {\n" name;
6087                  pr "    perror (\"strdup\");\n";
6088                  pr "    return -1;\n";
6089                  pr "  }\n"
6090              | FUUID ->
6091                  pr "  for (i = j = 0; i < 32; ++j) {\n";
6092                  pr "    if (tok[j] == '\\0') {\n";
6093                  pr "      fprintf (stderr, \"%%s: failed to parse UUID from '%%s'\\n\", __func__, tok);\n";
6094                  pr "      return -1;\n";
6095                  pr "    } else if (tok[j] != '-')\n";
6096                  pr "      r->%s[i++] = tok[j];\n" name;
6097                  pr "  }\n";
6098              | FBytes ->
6099                  pr "  if (sscanf (tok, \"%%\"SCNu64, &r->%s) != 1) {\n" name;
6100                  pr "    fprintf (stderr, \"%%s: failed to parse size '%%s' from token %%s\\n\", __func__, tok, \"%s\");\n" name;
6101                  pr "    return -1;\n";
6102                  pr "  }\n";
6103              | FInt64 ->
6104                  pr "  if (sscanf (tok, \"%%\"SCNi64, &r->%s) != 1) {\n" name;
6105                  pr "    fprintf (stderr, \"%%s: failed to parse int '%%s' from token %%s\\n\", __func__, tok, \"%s\");\n" name;
6106                  pr "    return -1;\n";
6107                  pr "  }\n";
6108              | FOptPercent ->
6109                  pr "  if (tok[0] == '\\0')\n";
6110                  pr "    r->%s = -1;\n" name;
6111                  pr "  else if (sscanf (tok, \"%%f\", &r->%s) != 1) {\n" name;
6112                  pr "    fprintf (stderr, \"%%s: failed to parse float '%%s' from token %%s\\n\", __func__, tok, \"%s\");\n" name;
6113                  pr "    return -1;\n";
6114                  pr "  }\n";
6115              | FBuffer | FInt32 | FUInt32 | FUInt64 | FChar ->
6116                  assert false (* can never be an LVM column *)
6117             );
6118             pr "  tok = next;\n";
6119         ) cols;
6120
6121         pr "  if (tok != NULL) {\n";
6122         pr "    fprintf (stderr, \"%%s: failed: extra tokens at end of string\\n\", __func__);\n";
6123         pr "    return -1;\n";
6124         pr "  }\n";
6125         pr "  return 0;\n";
6126         pr "}\n";
6127         pr "\n";
6128
6129         pr "guestfs_int_lvm_%s_list *\n" typ;
6130         pr "parse_command_line_%ss (void)\n" typ;
6131         pr "{\n";
6132         pr "  char *out, *err;\n";
6133         pr "  char *p, *pend;\n";
6134         pr "  int r, i;\n";
6135         pr "  guestfs_int_lvm_%s_list *ret;\n" typ;
6136         pr "  void *newp;\n";
6137         pr "\n";
6138         pr "  ret = malloc (sizeof *ret);\n";
6139         pr "  if (!ret) {\n";
6140         pr "    reply_with_perror (\"malloc\");\n";
6141         pr "    return NULL;\n";
6142         pr "  }\n";
6143         pr "\n";
6144         pr "  ret->guestfs_int_lvm_%s_list_len = 0;\n" typ;
6145         pr "  ret->guestfs_int_lvm_%s_list_val = NULL;\n" typ;
6146         pr "\n";
6147         pr "  r = command (&out, &err,\n";
6148         pr "           \"lvm\", \"%ss\",\n" typ;
6149         pr "           \"-o\", lvm_%s_cols, \"--unbuffered\", \"--noheadings\",\n" typ;
6150         pr "           \"--nosuffix\", \"--separator\", \",\", \"--units\", \"b\", NULL);\n";
6151         pr "  if (r == -1) {\n";
6152         pr "    reply_with_error (\"%%s\", err);\n";
6153         pr "    free (out);\n";
6154         pr "    free (err);\n";
6155         pr "    free (ret);\n";
6156         pr "    return NULL;\n";
6157         pr "  }\n";
6158         pr "\n";
6159         pr "  free (err);\n";
6160         pr "\n";
6161         pr "  /* Tokenize each line of the output. */\n";
6162         pr "  p = out;\n";
6163         pr "  i = 0;\n";
6164         pr "  while (p) {\n";
6165         pr "    pend = strchr (p, '\\n');       /* Get the next line of output. */\n";
6166         pr "    if (pend) {\n";
6167         pr "      *pend = '\\0';\n";
6168         pr "      pend++;\n";
6169         pr "    }\n";
6170         pr "\n";
6171         pr "    while (*p && c_isspace (*p))    /* Skip any leading whitespace. */\n";
6172         pr "      p++;\n";
6173         pr "\n";
6174         pr "    if (!*p) {                      /* Empty line?  Skip it. */\n";
6175         pr "      p = pend;\n";
6176         pr "      continue;\n";
6177         pr "    }\n";
6178         pr "\n";
6179         pr "    /* Allocate some space to store this next entry. */\n";
6180         pr "    newp = realloc (ret->guestfs_int_lvm_%s_list_val,\n" typ;
6181         pr "                sizeof (guestfs_int_lvm_%s) * (i+1));\n" typ;
6182         pr "    if (newp == NULL) {\n";
6183         pr "      reply_with_perror (\"realloc\");\n";
6184         pr "      free (ret->guestfs_int_lvm_%s_list_val);\n" typ;
6185         pr "      free (ret);\n";
6186         pr "      free (out);\n";
6187         pr "      return NULL;\n";
6188         pr "    }\n";
6189         pr "    ret->guestfs_int_lvm_%s_list_val = newp;\n" typ;
6190         pr "\n";
6191         pr "    /* Tokenize the next entry. */\n";
6192         pr "    r = lvm_tokenize_%s (p, &ret->guestfs_int_lvm_%s_list_val[i]);\n" typ typ;
6193         pr "    if (r == -1) {\n";
6194         pr "      reply_with_error (\"failed to parse output of '%ss' command\");\n" typ;
6195         pr "      free (ret->guestfs_int_lvm_%s_list_val);\n" typ;
6196         pr "      free (ret);\n";
6197         pr "      free (out);\n";
6198         pr "      return NULL;\n";
6199         pr "    }\n";
6200         pr "\n";
6201         pr "    ++i;\n";
6202         pr "    p = pend;\n";
6203         pr "  }\n";
6204         pr "\n";
6205         pr "  ret->guestfs_int_lvm_%s_list_len = i;\n" typ;
6206         pr "\n";
6207         pr "  free (out);\n";
6208         pr "  return ret;\n";
6209         pr "}\n"
6210
6211   ) ["pv", lvm_pv_cols; "vg", lvm_vg_cols; "lv", lvm_lv_cols]
6212
6213 (* Generate a list of function names, for debugging in the daemon.. *)
6214 and generate_daemon_names () =
6215   generate_header CStyle GPLv2plus;
6216
6217   pr "#include <config.h>\n";
6218   pr "\n";
6219   pr "#include \"daemon.h\"\n";
6220   pr "\n";
6221
6222   pr "/* This array is indexed by proc_nr.  See guestfs_protocol.x. */\n";
6223   pr "const char *function_names[] = {\n";
6224   List.iter (
6225     fun (name, _, proc_nr, _, _, _, _) -> pr "  [%d] = \"%s\",\n" proc_nr name
6226   ) daemon_functions;
6227   pr "};\n";
6228
6229 (* Generate the optional groups for the daemon to implement
6230  * guestfs_available.
6231  *)
6232 and generate_daemon_optgroups_c () =
6233   generate_header CStyle GPLv2plus;
6234
6235   pr "#include <config.h>\n";
6236   pr "\n";
6237   pr "#include \"daemon.h\"\n";
6238   pr "#include \"optgroups.h\"\n";
6239   pr "\n";
6240
6241   pr "struct optgroup optgroups[] = {\n";
6242   List.iter (
6243     fun (group, _) ->
6244       pr "  { \"%s\", optgroup_%s_available },\n" group group
6245   ) optgroups;
6246   pr "  { NULL, NULL }\n";
6247   pr "};\n"
6248
6249 and generate_daemon_optgroups_h () =
6250   generate_header CStyle GPLv2plus;
6251
6252   List.iter (
6253     fun (group, _) ->
6254       pr "extern int optgroup_%s_available (void);\n" group
6255   ) optgroups
6256
6257 (* Generate the tests. *)
6258 and generate_tests () =
6259   generate_header CStyle GPLv2plus;
6260
6261   pr "\
6262 #include <stdio.h>
6263 #include <stdlib.h>
6264 #include <string.h>
6265 #include <unistd.h>
6266 #include <sys/types.h>
6267 #include <fcntl.h>
6268
6269 #include \"guestfs.h\"
6270 #include \"guestfs-internal.h\"
6271
6272 static guestfs_h *g;
6273 static int suppress_error = 0;
6274
6275 static void print_error (guestfs_h *g, void *data, const char *msg)
6276 {
6277   if (!suppress_error)
6278     fprintf (stderr, \"%%s\\n\", msg);
6279 }
6280
6281 /* FIXME: nearly identical code appears in fish.c */
6282 static void print_strings (char *const *argv)
6283 {
6284   int argc;
6285
6286   for (argc = 0; argv[argc] != NULL; ++argc)
6287     printf (\"\\t%%s\\n\", argv[argc]);
6288 }
6289
6290 /*
6291 static void print_table (char const *const *argv)
6292 {
6293   int i;
6294
6295   for (i = 0; argv[i] != NULL; i += 2)
6296     printf (\"%%s: %%s\\n\", argv[i], argv[i+1]);
6297 }
6298 */
6299
6300 ";
6301
6302   (* Generate a list of commands which are not tested anywhere. *)
6303   pr "static void no_test_warnings (void)\n";
6304   pr "{\n";
6305
6306   let hash : (string, bool) Hashtbl.t = Hashtbl.create 13 in
6307   List.iter (
6308     fun (_, _, _, _, tests, _, _) ->
6309       let tests = filter_map (
6310         function
6311         | (_, (Always|If _|Unless _), test) -> Some test
6312         | (_, Disabled, _) -> None
6313       ) tests in
6314       let seq = List.concat (List.map seq_of_test tests) in
6315       let cmds_tested = List.map List.hd seq in
6316       List.iter (fun cmd -> Hashtbl.replace hash cmd true) cmds_tested
6317   ) all_functions;
6318
6319   List.iter (
6320     fun (name, _, _, _, _, _, _) ->
6321       if not (Hashtbl.mem hash name) then
6322         pr "  fprintf (stderr, \"warning: \\\"guestfs_%s\\\" has no tests\\n\");\n" name
6323   ) all_functions;
6324
6325   pr "}\n";
6326   pr "\n";
6327
6328   (* Generate the actual tests.  Note that we generate the tests
6329    * in reverse order, deliberately, so that (in general) the
6330    * newest tests run first.  This makes it quicker and easier to
6331    * debug them.
6332    *)
6333   let test_names =
6334     List.map (
6335       fun (name, _, _, flags, tests, _, _) ->
6336         mapi (generate_one_test name flags) tests
6337     ) (List.rev all_functions) in
6338   let test_names = List.concat test_names in
6339   let nr_tests = List.length test_names in
6340
6341   pr "\
6342 int main (int argc, char *argv[])
6343 {
6344   char c = 0;
6345   unsigned long int n_failed = 0;
6346   const char *filename;
6347   int fd;
6348   int nr_tests, test_num = 0;
6349
6350   setbuf (stdout, NULL);
6351
6352   no_test_warnings ();
6353
6354   g = guestfs_create ();
6355   if (g == NULL) {
6356     printf (\"guestfs_create FAILED\\n\");
6357     exit (EXIT_FAILURE);
6358   }
6359
6360   guestfs_set_error_handler (g, print_error, NULL);
6361
6362   guestfs_set_path (g, \"../appliance\");
6363
6364   filename = \"test1.img\";
6365   fd = open (filename, O_WRONLY|O_CREAT|O_NOCTTY|O_NONBLOCK|O_TRUNC, 0666);
6366   if (fd == -1) {
6367     perror (filename);
6368     exit (EXIT_FAILURE);
6369   }
6370   if (lseek (fd, %d, SEEK_SET) == -1) {
6371     perror (\"lseek\");
6372     close (fd);
6373     unlink (filename);
6374     exit (EXIT_FAILURE);
6375   }
6376   if (write (fd, &c, 1) == -1) {
6377     perror (\"write\");
6378     close (fd);
6379     unlink (filename);
6380     exit (EXIT_FAILURE);
6381   }
6382   if (close (fd) == -1) {
6383     perror (filename);
6384     unlink (filename);
6385     exit (EXIT_FAILURE);
6386   }
6387   if (guestfs_add_drive (g, filename) == -1) {
6388     printf (\"guestfs_add_drive %%s FAILED\\n\", filename);
6389     exit (EXIT_FAILURE);
6390   }
6391
6392   filename = \"test2.img\";
6393   fd = open (filename, O_WRONLY|O_CREAT|O_NOCTTY|O_NONBLOCK|O_TRUNC, 0666);
6394   if (fd == -1) {
6395     perror (filename);
6396     exit (EXIT_FAILURE);
6397   }
6398   if (lseek (fd, %d, SEEK_SET) == -1) {
6399     perror (\"lseek\");
6400     close (fd);
6401     unlink (filename);
6402     exit (EXIT_FAILURE);
6403   }
6404   if (write (fd, &c, 1) == -1) {
6405     perror (\"write\");
6406     close (fd);
6407     unlink (filename);
6408     exit (EXIT_FAILURE);
6409   }
6410   if (close (fd) == -1) {
6411     perror (filename);
6412     unlink (filename);
6413     exit (EXIT_FAILURE);
6414   }
6415   if (guestfs_add_drive (g, filename) == -1) {
6416     printf (\"guestfs_add_drive %%s FAILED\\n\", filename);
6417     exit (EXIT_FAILURE);
6418   }
6419
6420   filename = \"test3.img\";
6421   fd = open (filename, O_WRONLY|O_CREAT|O_NOCTTY|O_NONBLOCK|O_TRUNC, 0666);
6422   if (fd == -1) {
6423     perror (filename);
6424     exit (EXIT_FAILURE);
6425   }
6426   if (lseek (fd, %d, SEEK_SET) == -1) {
6427     perror (\"lseek\");
6428     close (fd);
6429     unlink (filename);
6430     exit (EXIT_FAILURE);
6431   }
6432   if (write (fd, &c, 1) == -1) {
6433     perror (\"write\");
6434     close (fd);
6435     unlink (filename);
6436     exit (EXIT_FAILURE);
6437   }
6438   if (close (fd) == -1) {
6439     perror (filename);
6440     unlink (filename);
6441     exit (EXIT_FAILURE);
6442   }
6443   if (guestfs_add_drive (g, filename) == -1) {
6444     printf (\"guestfs_add_drive %%s FAILED\\n\", filename);
6445     exit (EXIT_FAILURE);
6446   }
6447
6448   if (guestfs_add_drive_ro (g, \"../images/test.iso\") == -1) {
6449     printf (\"guestfs_add_drive_ro ../images/test.iso FAILED\\n\");
6450     exit (EXIT_FAILURE);
6451   }
6452
6453   /* Set a timeout in case qemu hangs during launch (RHBZ#505329). */
6454   alarm (600);
6455
6456   if (guestfs_launch (g) == -1) {
6457     printf (\"guestfs_launch FAILED\\n\");
6458     exit (EXIT_FAILURE);
6459   }
6460
6461   /* Cancel previous alarm. */
6462   alarm (0);
6463
6464   nr_tests = %d;
6465
6466 " (500 * 1024 * 1024) (50 * 1024 * 1024) (10 * 1024 * 1024) nr_tests;
6467
6468   iteri (
6469     fun i test_name ->
6470       pr "  test_num++;\n";
6471       pr "  printf (\"%%3d/%%3d %s\\n\", test_num, nr_tests);\n" test_name;
6472       pr "  if (%s () == -1) {\n" test_name;
6473       pr "    printf (\"%s FAILED\\n\");\n" test_name;
6474       pr "    n_failed++;\n";
6475       pr "  }\n";
6476   ) test_names;
6477   pr "\n";
6478
6479   pr "  guestfs_close (g);\n";
6480   pr "  unlink (\"test1.img\");\n";
6481   pr "  unlink (\"test2.img\");\n";
6482   pr "  unlink (\"test3.img\");\n";
6483   pr "\n";
6484
6485   pr "  if (n_failed > 0) {\n";
6486   pr "    printf (\"***** %%lu / %%d tests FAILED *****\\n\", n_failed, nr_tests);\n";
6487   pr "    exit (EXIT_FAILURE);\n";
6488   pr "  }\n";
6489   pr "\n";
6490
6491   pr "  exit (EXIT_SUCCESS);\n";
6492   pr "}\n"
6493
6494 and generate_one_test name flags i (init, prereq, test) =
6495   let test_name = sprintf "test_%s_%d" name i in
6496
6497   pr "\
6498 static int %s_skip (void)
6499 {
6500   const char *str;
6501
6502   str = getenv (\"TEST_ONLY\");
6503   if (str)
6504     return strstr (str, \"%s\") == NULL;
6505   str = getenv (\"SKIP_%s\");
6506   if (str && STREQ (str, \"1\")) return 1;
6507   str = getenv (\"SKIP_TEST_%s\");
6508   if (str && STREQ (str, \"1\")) return 1;
6509   return 0;
6510 }
6511
6512 " test_name name (String.uppercase test_name) (String.uppercase name);
6513
6514   (match prereq with
6515    | Disabled | Always -> ()
6516    | If code | Unless code ->
6517        pr "static int %s_prereq (void)\n" test_name;
6518        pr "{\n";
6519        pr "  %s\n" code;
6520        pr "}\n";
6521        pr "\n";
6522   );
6523
6524   pr "\
6525 static int %s (void)
6526 {
6527   if (%s_skip ()) {
6528     printf (\"        %%s skipped (reason: environment variable set)\\n\", \"%s\");
6529     return 0;
6530   }
6531
6532 " test_name test_name test_name;
6533
6534   (* Optional functions should only be tested if the relevant
6535    * support is available in the daemon.
6536    *)
6537   List.iter (
6538     function
6539     | Optional group ->
6540         pr "  {\n";
6541         pr "    const char *groups[] = { \"%s\", NULL };\n" group;
6542         pr "    int r;\n";
6543         pr "    suppress_error = 1;\n";
6544         pr "    r = guestfs_available (g, (char **) groups);\n";
6545         pr "    suppress_error = 0;\n";
6546         pr "    if (r == -1) {\n";
6547         pr "      printf (\"        %%s skipped (reason: group %%s not available in daemon)\\n\", \"%s\", groups[0]);\n" test_name;
6548         pr "      return 0;\n";
6549         pr "    }\n";
6550         pr "  }\n";
6551     | _ -> ()
6552   ) flags;
6553
6554   (match prereq with
6555    | Disabled ->
6556        pr "  printf (\"        %%s skipped (reason: test disabled in generator)\\n\", \"%s\");\n" test_name
6557    | If _ ->
6558        pr "  if (! %s_prereq ()) {\n" test_name;
6559        pr "    printf (\"        %%s skipped (reason: test prerequisite)\\n\", \"%s\");\n" test_name;
6560        pr "    return 0;\n";
6561        pr "  }\n";
6562        pr "\n";
6563        generate_one_test_body name i test_name init test;
6564    | Unless _ ->
6565        pr "  if (%s_prereq ()) {\n" test_name;
6566        pr "    printf (\"        %%s skipped (reason: test prerequisite)\\n\", \"%s\");\n" test_name;
6567        pr "    return 0;\n";
6568        pr "  }\n";
6569        pr "\n";
6570        generate_one_test_body name i test_name init test;
6571    | Always ->
6572        generate_one_test_body name i test_name init test
6573   );
6574
6575   pr "  return 0;\n";
6576   pr "}\n";
6577   pr "\n";
6578   test_name
6579
6580 and generate_one_test_body name i test_name init test =
6581   (match init with
6582    | InitNone (* XXX at some point, InitNone and InitEmpty became
6583                * folded together as the same thing.  Really we should
6584                * make InitNone do nothing at all, but the tests may
6585                * need to be checked to make sure this is OK.
6586                *)
6587    | InitEmpty ->
6588        pr "  /* InitNone|InitEmpty for %s */\n" test_name;
6589        List.iter (generate_test_command_call test_name)
6590          [["blockdev_setrw"; "/dev/sda"];
6591           ["umount_all"];
6592           ["lvm_remove_all"]]
6593    | InitPartition ->
6594        pr "  /* InitPartition for %s: create /dev/sda1 */\n" test_name;
6595        List.iter (generate_test_command_call test_name)
6596          [["blockdev_setrw"; "/dev/sda"];
6597           ["umount_all"];
6598           ["lvm_remove_all"];
6599           ["part_disk"; "/dev/sda"; "mbr"]]
6600    | InitBasicFS ->
6601        pr "  /* InitBasicFS for %s: create ext2 on /dev/sda1 */\n" test_name;
6602        List.iter (generate_test_command_call test_name)
6603          [["blockdev_setrw"; "/dev/sda"];
6604           ["umount_all"];
6605           ["lvm_remove_all"];
6606           ["part_disk"; "/dev/sda"; "mbr"];
6607           ["mkfs"; "ext2"; "/dev/sda1"];
6608           ["mount_options"; ""; "/dev/sda1"; "/"]]
6609    | InitBasicFSonLVM ->
6610        pr "  /* InitBasicFSonLVM for %s: create ext2 on /dev/VG/LV */\n"
6611          test_name;
6612        List.iter (generate_test_command_call test_name)
6613          [["blockdev_setrw"; "/dev/sda"];
6614           ["umount_all"];
6615           ["lvm_remove_all"];
6616           ["part_disk"; "/dev/sda"; "mbr"];
6617           ["pvcreate"; "/dev/sda1"];
6618           ["vgcreate"; "VG"; "/dev/sda1"];
6619           ["lvcreate"; "LV"; "VG"; "8"];
6620           ["mkfs"; "ext2"; "/dev/VG/LV"];
6621           ["mount_options"; ""; "/dev/VG/LV"; "/"]]
6622    | InitISOFS ->
6623        pr "  /* InitISOFS for %s */\n" test_name;
6624        List.iter (generate_test_command_call test_name)
6625          [["blockdev_setrw"; "/dev/sda"];
6626           ["umount_all"];
6627           ["lvm_remove_all"];
6628           ["mount_ro"; "/dev/sdd"; "/"]]
6629   );
6630
6631   let get_seq_last = function
6632     | [] ->
6633         failwithf "%s: you cannot use [] (empty list) when expecting a command"
6634           test_name
6635     | seq ->
6636         let seq = List.rev seq in
6637         List.rev (List.tl seq), List.hd seq
6638   in
6639
6640   match test with
6641   | TestRun seq ->
6642       pr "  /* TestRun for %s (%d) */\n" name i;
6643       List.iter (generate_test_command_call test_name) seq
6644   | TestOutput (seq, expected) ->
6645       pr "  /* TestOutput for %s (%d) */\n" name i;
6646       pr "  const char *expected = \"%s\";\n" (c_quote expected);
6647       let seq, last = get_seq_last seq in
6648       let test () =
6649         pr "    if (STRNEQ (r, expected)) {\n";
6650         pr "      fprintf (stderr, \"%s: expected \\\"%%s\\\" but got \\\"%%s\\\"\\n\", expected, r);\n" test_name;
6651         pr "      return -1;\n";
6652         pr "    }\n"
6653       in
6654       List.iter (generate_test_command_call test_name) seq;
6655       generate_test_command_call ~test test_name last
6656   | TestOutputList (seq, expected) ->
6657       pr "  /* TestOutputList for %s (%d) */\n" name i;
6658       let seq, last = get_seq_last seq in
6659       let test () =
6660         iteri (
6661           fun i str ->
6662             pr "    if (!r[%d]) {\n" i;
6663             pr "      fprintf (stderr, \"%s: short list returned from command\\n\");\n" test_name;
6664             pr "      print_strings (r);\n";
6665             pr "      return -1;\n";
6666             pr "    }\n";
6667             pr "    {\n";
6668             pr "      const char *expected = \"%s\";\n" (c_quote str);
6669             pr "      if (STRNEQ (r[%d], expected)) {\n" i;
6670             pr "        fprintf (stderr, \"%s: expected \\\"%%s\\\" but got \\\"%%s\\\"\\n\", expected, r[%d]);\n" test_name i;
6671             pr "        return -1;\n";
6672             pr "      }\n";
6673             pr "    }\n"
6674         ) expected;
6675         pr "    if (r[%d] != NULL) {\n" (List.length expected);
6676         pr "      fprintf (stderr, \"%s: extra elements returned from command\\n\");\n"
6677           test_name;
6678         pr "      print_strings (r);\n";
6679         pr "      return -1;\n";
6680         pr "    }\n"
6681       in
6682       List.iter (generate_test_command_call test_name) seq;
6683       generate_test_command_call ~test test_name last
6684   | TestOutputListOfDevices (seq, expected) ->
6685       pr "  /* TestOutputListOfDevices for %s (%d) */\n" name i;
6686       let seq, last = get_seq_last seq in
6687       let test () =
6688         iteri (
6689           fun i str ->
6690             pr "    if (!r[%d]) {\n" i;
6691             pr "      fprintf (stderr, \"%s: short list returned from command\\n\");\n" test_name;
6692             pr "      print_strings (r);\n";
6693             pr "      return -1;\n";
6694             pr "    }\n";
6695             pr "    {\n";
6696             pr "      const char *expected = \"%s\";\n" (c_quote str);
6697             pr "      r[%d][5] = 's';\n" i;
6698             pr "      if (STRNEQ (r[%d], expected)) {\n" i;
6699             pr "        fprintf (stderr, \"%s: expected \\\"%%s\\\" but got \\\"%%s\\\"\\n\", expected, r[%d]);\n" test_name i;
6700             pr "        return -1;\n";
6701             pr "      }\n";
6702             pr "    }\n"
6703         ) expected;
6704         pr "    if (r[%d] != NULL) {\n" (List.length expected);
6705         pr "      fprintf (stderr, \"%s: extra elements returned from command\\n\");\n"
6706           test_name;
6707         pr "      print_strings (r);\n";
6708         pr "      return -1;\n";
6709         pr "    }\n"
6710       in
6711       List.iter (generate_test_command_call test_name) seq;
6712       generate_test_command_call ~test test_name last
6713   | TestOutputInt (seq, expected) ->
6714       pr "  /* TestOutputInt for %s (%d) */\n" name i;
6715       let seq, last = get_seq_last seq in
6716       let test () =
6717         pr "    if (r != %d) {\n" expected;
6718         pr "      fprintf (stderr, \"%s: expected %d but got %%d\\n\","
6719           test_name expected;
6720         pr "               (int) r);\n";
6721         pr "      return -1;\n";
6722         pr "    }\n"
6723       in
6724       List.iter (generate_test_command_call test_name) seq;
6725       generate_test_command_call ~test test_name last
6726   | TestOutputIntOp (seq, op, expected) ->
6727       pr "  /* TestOutputIntOp for %s (%d) */\n" name i;
6728       let seq, last = get_seq_last seq in
6729       let test () =
6730         pr "    if (! (r %s %d)) {\n" op expected;
6731         pr "      fprintf (stderr, \"%s: expected %s %d but got %%d\\n\","
6732           test_name op expected;
6733         pr "               (int) r);\n";
6734         pr "      return -1;\n";
6735         pr "    }\n"
6736       in
6737       List.iter (generate_test_command_call test_name) seq;
6738       generate_test_command_call ~test test_name last
6739   | TestOutputTrue seq ->
6740       pr "  /* TestOutputTrue for %s (%d) */\n" name i;
6741       let seq, last = get_seq_last seq in
6742       let test () =
6743         pr "    if (!r) {\n";
6744         pr "      fprintf (stderr, \"%s: expected true, got false\\n\");\n"
6745           test_name;
6746         pr "      return -1;\n";
6747         pr "    }\n"
6748       in
6749       List.iter (generate_test_command_call test_name) seq;
6750       generate_test_command_call ~test test_name last
6751   | TestOutputFalse seq ->
6752       pr "  /* TestOutputFalse for %s (%d) */\n" name i;
6753       let seq, last = get_seq_last seq in
6754       let test () =
6755         pr "    if (r) {\n";
6756         pr "      fprintf (stderr, \"%s: expected false, got true\\n\");\n"
6757           test_name;
6758         pr "      return -1;\n";
6759         pr "    }\n"
6760       in
6761       List.iter (generate_test_command_call test_name) seq;
6762       generate_test_command_call ~test test_name last
6763   | TestOutputLength (seq, expected) ->
6764       pr "  /* TestOutputLength for %s (%d) */\n" name i;
6765       let seq, last = get_seq_last seq in
6766       let test () =
6767         pr "    int j;\n";
6768         pr "    for (j = 0; j < %d; ++j)\n" expected;
6769         pr "      if (r[j] == NULL) {\n";
6770         pr "        fprintf (stderr, \"%s: short list returned\\n\");\n"
6771           test_name;
6772         pr "        print_strings (r);\n";
6773         pr "        return -1;\n";
6774         pr "      }\n";
6775         pr "    if (r[j] != NULL) {\n";
6776         pr "      fprintf (stderr, \"%s: long list returned\\n\");\n"
6777           test_name;
6778         pr "      print_strings (r);\n";
6779         pr "      return -1;\n";
6780         pr "    }\n"
6781       in
6782       List.iter (generate_test_command_call test_name) seq;
6783       generate_test_command_call ~test test_name last
6784   | TestOutputBuffer (seq, expected) ->
6785       pr "  /* TestOutputBuffer for %s (%d) */\n" name i;
6786       pr "  const char *expected = \"%s\";\n" (c_quote expected);
6787       let seq, last = get_seq_last seq in
6788       let len = String.length expected in
6789       let test () =
6790         pr "    if (size != %d) {\n" len;
6791         pr "      fprintf (stderr, \"%s: returned size of buffer wrong, expected %d but got %%zu\\n\", size);\n" test_name len;
6792         pr "      return -1;\n";
6793         pr "    }\n";
6794         pr "    if (STRNEQLEN (r, expected, size)) {\n";
6795         pr "      fprintf (stderr, \"%s: expected \\\"%%s\\\" but got \\\"%%s\\\"\\n\", expected, r);\n" test_name;
6796         pr "      return -1;\n";
6797         pr "    }\n"
6798       in
6799       List.iter (generate_test_command_call test_name) seq;
6800       generate_test_command_call ~test test_name last
6801   | TestOutputStruct (seq, checks) ->
6802       pr "  /* TestOutputStruct for %s (%d) */\n" name i;
6803       let seq, last = get_seq_last seq in
6804       let test () =
6805         List.iter (
6806           function
6807           | CompareWithInt (field, expected) ->
6808               pr "    if (r->%s != %d) {\n" field expected;
6809               pr "      fprintf (stderr, \"%s: %s was %%d, expected %d\\n\",\n"
6810                 test_name field expected;
6811               pr "               (int) r->%s);\n" field;
6812               pr "      return -1;\n";
6813               pr "    }\n"
6814           | CompareWithIntOp (field, op, expected) ->
6815               pr "    if (!(r->%s %s %d)) {\n" field op expected;
6816               pr "      fprintf (stderr, \"%s: %s was %%d, expected %s %d\\n\",\n"
6817                 test_name field op expected;
6818               pr "               (int) r->%s);\n" field;
6819               pr "      return -1;\n";
6820               pr "    }\n"
6821           | CompareWithString (field, expected) ->
6822               pr "    if (STRNEQ (r->%s, \"%s\")) {\n" field expected;
6823               pr "      fprintf (stderr, \"%s: %s was \"%%s\", expected \"%s\"\\n\",\n"
6824                 test_name field expected;
6825               pr "               r->%s);\n" field;
6826               pr "      return -1;\n";
6827               pr "    }\n"
6828           | CompareFieldsIntEq (field1, field2) ->
6829               pr "    if (r->%s != r->%s) {\n" field1 field2;
6830               pr "      fprintf (stderr, \"%s: %s (%%d) <> %s (%%d)\\n\",\n"
6831                 test_name field1 field2;
6832               pr "               (int) r->%s, (int) r->%s);\n" field1 field2;
6833               pr "      return -1;\n";
6834               pr "    }\n"
6835           | CompareFieldsStrEq (field1, field2) ->
6836               pr "    if (STRNEQ (r->%s, r->%s)) {\n" field1 field2;
6837               pr "      fprintf (stderr, \"%s: %s (\"%%s\") <> %s (\"%%s\")\\n\",\n"
6838                 test_name field1 field2;
6839               pr "               r->%s, r->%s);\n" field1 field2;
6840               pr "      return -1;\n";
6841               pr "    }\n"
6842         ) checks
6843       in
6844       List.iter (generate_test_command_call test_name) seq;
6845       generate_test_command_call ~test test_name last
6846   | TestLastFail seq ->
6847       pr "  /* TestLastFail for %s (%d) */\n" name i;
6848       let seq, last = get_seq_last seq in
6849       List.iter (generate_test_command_call test_name) seq;
6850       generate_test_command_call test_name ~expect_error:true last
6851
6852 (* Generate the code to run a command, leaving the result in 'r'.
6853  * If you expect to get an error then you should set expect_error:true.
6854  *)
6855 and generate_test_command_call ?(expect_error = false) ?test test_name cmd =
6856   match cmd with
6857   | [] -> assert false
6858   | name :: args ->
6859       (* Look up the command to find out what args/ret it has. *)
6860       let style =
6861         try
6862           let _, style, _, _, _, _, _ =
6863             List.find (fun (n, _, _, _, _, _, _) -> n = name) all_functions in
6864           style
6865         with Not_found ->
6866           failwithf "%s: in test, command %s was not found" test_name name in
6867
6868       if List.length (snd style) <> List.length args then
6869         failwithf "%s: in test, wrong number of args given to %s"
6870           test_name name;
6871
6872       pr "  {\n";
6873
6874       List.iter (
6875         function
6876         | OptString n, "NULL" -> ()
6877         | Pathname n, arg
6878         | Device n, arg
6879         | Dev_or_Path n, arg
6880         | String n, arg
6881         | OptString n, arg ->
6882             pr "    const char *%s = \"%s\";\n" n (c_quote arg);
6883         | Int _, _
6884         | Int64 _, _
6885         | Bool _, _
6886         | FileIn _, _ | FileOut _, _ -> ()
6887         | StringList n, "" | DeviceList n, "" ->
6888             pr "    const char *const %s[1] = { NULL };\n" n
6889         | StringList n, arg | DeviceList n, arg ->
6890             let strs = string_split " " arg in
6891             iteri (
6892               fun i str ->
6893                 pr "    const char *%s_%d = \"%s\";\n" n i (c_quote str);
6894             ) strs;
6895             pr "    const char *const %s[] = {\n" n;
6896             iteri (
6897               fun i _ -> pr "      %s_%d,\n" n i
6898             ) strs;
6899             pr "      NULL\n";
6900             pr "    };\n";
6901       ) (List.combine (snd style) args);
6902
6903       let error_code =
6904         match fst style with
6905         | RErr | RInt _ | RBool _ -> pr "    int r;\n"; "-1"
6906         | RInt64 _ -> pr "    int64_t r;\n"; "-1"
6907         | RConstString _ | RConstOptString _ ->
6908             pr "    const char *r;\n"; "NULL"
6909         | RString _ -> pr "    char *r;\n"; "NULL"
6910         | RStringList _ | RHashtable _ ->
6911             pr "    char **r;\n";
6912             pr "    int i;\n";
6913             "NULL"
6914         | RStruct (_, typ) ->
6915             pr "    struct guestfs_%s *r;\n" typ; "NULL"
6916         | RStructList (_, typ) ->
6917             pr "    struct guestfs_%s_list *r;\n" typ; "NULL"
6918         | RBufferOut _ ->
6919             pr "    char *r;\n";
6920             pr "    size_t size;\n";
6921             "NULL" in
6922
6923       pr "    suppress_error = %d;\n" (if expect_error then 1 else 0);
6924       pr "    r = guestfs_%s (g" name;
6925
6926       (* Generate the parameters. *)
6927       List.iter (
6928         function
6929         | OptString _, "NULL" -> pr ", NULL"
6930         | Pathname n, _
6931         | Device n, _ | Dev_or_Path n, _
6932         | String n, _
6933         | OptString n, _ ->
6934             pr ", %s" n
6935         | FileIn _, arg | FileOut _, arg ->
6936             pr ", \"%s\"" (c_quote arg)
6937         | StringList n, _ | DeviceList n, _ ->
6938             pr ", (char **) %s" n
6939         | Int _, arg ->
6940             let i =
6941               try int_of_string arg
6942               with Failure "int_of_string" ->
6943                 failwithf "%s: expecting an int, but got '%s'" test_name arg in
6944             pr ", %d" i
6945         | Int64 _, arg ->
6946             let i =
6947               try Int64.of_string arg
6948               with Failure "int_of_string" ->
6949                 failwithf "%s: expecting an int64, but got '%s'" test_name arg in
6950             pr ", %Ld" i
6951         | Bool _, arg ->
6952             let b = bool_of_string arg in pr ", %d" (if b then 1 else 0)
6953       ) (List.combine (snd style) args);
6954
6955       (match fst style with
6956        | RBufferOut _ -> pr ", &size"
6957        | _ -> ()
6958       );
6959
6960       pr ");\n";
6961
6962       if not expect_error then
6963         pr "    if (r == %s)\n" error_code
6964       else
6965         pr "    if (r != %s)\n" error_code;
6966       pr "      return -1;\n";
6967
6968       (* Insert the test code. *)
6969       (match test with
6970        | None -> ()
6971        | Some f -> f ()
6972       );
6973
6974       (match fst style with
6975        | RErr | RInt _ | RInt64 _ | RBool _
6976        | RConstString _ | RConstOptString _ -> ()
6977        | RString _ | RBufferOut _ -> pr "    free (r);\n"
6978        | RStringList _ | RHashtable _ ->
6979            pr "    for (i = 0; r[i] != NULL; ++i)\n";
6980            pr "      free (r[i]);\n";
6981            pr "    free (r);\n"
6982        | RStruct (_, typ) ->
6983            pr "    guestfs_free_%s (r);\n" typ
6984        | RStructList (_, typ) ->
6985            pr "    guestfs_free_%s_list (r);\n" typ
6986       );
6987
6988       pr "  }\n"
6989
6990 and c_quote str =
6991   let str = replace_str str "\r" "\\r" in
6992   let str = replace_str str "\n" "\\n" in
6993   let str = replace_str str "\t" "\\t" in
6994   let str = replace_str str "\000" "\\0" in
6995   str
6996
6997 (* Generate a lot of different functions for guestfish. *)
6998 and generate_fish_cmds () =
6999   generate_header CStyle GPLv2plus;
7000
7001   let all_functions =
7002     List.filter (
7003       fun (_, _, _, flags, _, _, _) -> not (List.mem NotInFish flags)
7004     ) all_functions in
7005   let all_functions_sorted =
7006     List.filter (
7007       fun (_, _, _, flags, _, _, _) -> not (List.mem NotInFish flags)
7008     ) all_functions_sorted in
7009
7010   pr "#include <config.h>\n";
7011   pr "\n";
7012   pr "#include <stdio.h>\n";
7013   pr "#include <stdlib.h>\n";
7014   pr "#include <string.h>\n";
7015   pr "#include <inttypes.h>\n";
7016   pr "\n";
7017   pr "#include <guestfs.h>\n";
7018   pr "#include \"c-ctype.h\"\n";
7019   pr "#include \"full-write.h\"\n";
7020   pr "#include \"xstrtol.h\"\n";
7021   pr "#include \"fish.h\"\n";
7022   pr "\n";
7023
7024   (* list_commands function, which implements guestfish -h *)
7025   pr "void list_commands (void)\n";
7026   pr "{\n";
7027   pr "  printf (\"    %%-16s     %%s\\n\", _(\"Command\"), _(\"Description\"));\n";
7028   pr "  list_builtin_commands ();\n";
7029   List.iter (
7030     fun (name, _, _, flags, _, shortdesc, _) ->
7031       let name = replace_char name '_' '-' in
7032       pr "  printf (\"%%-20s %%s\\n\", \"%s\", _(\"%s\"));\n"
7033         name shortdesc
7034   ) all_functions_sorted;
7035   pr "  printf (\"    %%s\\n\",";
7036   pr "          _(\"Use -h <cmd> / help <cmd> to show detailed help for a command.\"));\n";
7037   pr "}\n";
7038   pr "\n";
7039
7040   (* display_command function, which implements guestfish -h cmd *)
7041   pr "void display_command (const char *cmd)\n";
7042   pr "{\n";
7043   List.iter (
7044     fun (name, style, _, flags, _, shortdesc, longdesc) ->
7045       let name2 = replace_char name '_' '-' in
7046       let alias =
7047         try find_map (function FishAlias n -> Some n | _ -> None) flags
7048         with Not_found -> name in
7049       let longdesc = replace_str longdesc "C<guestfs_" "C<" in
7050       let synopsis =
7051         match snd style with
7052         | [] -> name2
7053         | args ->
7054             sprintf "%s %s"
7055               name2 (String.concat " " (List.map name_of_argt args)) in
7056
7057       let warnings =
7058         if List.mem ProtocolLimitWarning flags then
7059           ("\n\n" ^ protocol_limit_warning)
7060         else "" in
7061
7062       (* For DangerWillRobinson commands, we should probably have
7063        * guestfish prompt before allowing you to use them (especially
7064        * in interactive mode). XXX
7065        *)
7066       let warnings =
7067         warnings ^
7068           if List.mem DangerWillRobinson flags then
7069             ("\n\n" ^ danger_will_robinson)
7070           else "" in
7071
7072       let warnings =
7073         warnings ^
7074           match deprecation_notice flags with
7075           | None -> ""
7076           | Some txt -> "\n\n" ^ txt in
7077
7078       let describe_alias =
7079         if name <> alias then
7080           sprintf "\n\nYou can use '%s' as an alias for this command." alias
7081         else "" in
7082
7083       pr "  if (";
7084       pr "STRCASEEQ (cmd, \"%s\")" name;
7085       if name <> name2 then
7086         pr " || STRCASEEQ (cmd, \"%s\")" name2;
7087       if name <> alias then
7088         pr " || STRCASEEQ (cmd, \"%s\")" alias;
7089       pr ")\n";
7090       pr "    pod2text (\"%s\", _(\"%s\"), %S);\n"
7091         name2 shortdesc
7092         ("=head1 SYNOPSIS\n\n " ^ synopsis ^ "\n\n" ^
7093          "=head1 DESCRIPTION\n\n" ^
7094          longdesc ^ warnings ^ describe_alias);
7095       pr "  else\n"
7096   ) all_functions;
7097   pr "    display_builtin_command (cmd);\n";
7098   pr "}\n";
7099   pr "\n";
7100
7101   let emit_print_list_function typ =
7102     pr "static void print_%s_list (struct guestfs_%s_list *%ss)\n"
7103       typ typ typ;
7104     pr "{\n";
7105     pr "  unsigned int i;\n";
7106     pr "\n";
7107     pr "  for (i = 0; i < %ss->len; ++i) {\n" typ;
7108     pr "    printf (\"[%%d] = {\\n\", i);\n";
7109     pr "    print_%s_indent (&%ss->val[i], \"  \");\n" typ typ;
7110     pr "    printf (\"}\\n\");\n";
7111     pr "  }\n";
7112     pr "}\n";
7113     pr "\n";
7114   in
7115
7116   (* print_* functions *)
7117   List.iter (
7118     fun (typ, cols) ->
7119       let needs_i =
7120         List.exists (function (_, (FUUID|FBuffer)) -> true | _ -> false) cols in
7121
7122       pr "static void print_%s_indent (struct guestfs_%s *%s, const char *indent)\n" typ typ typ;
7123       pr "{\n";
7124       if needs_i then (
7125         pr "  unsigned int i;\n";
7126         pr "\n"
7127       );
7128       List.iter (
7129         function
7130         | name, FString ->
7131             pr "  printf (\"%%s%s: %%s\\n\", indent, %s->%s);\n" name typ name
7132         | name, FUUID ->
7133             pr "  printf (\"%%s%s: \", indent);\n" name;
7134             pr "  for (i = 0; i < 32; ++i)\n";
7135             pr "    printf (\"%%c\", %s->%s[i]);\n" typ name;
7136             pr "  printf (\"\\n\");\n"
7137         | name, FBuffer ->
7138             pr "  printf (\"%%s%s: \", indent);\n" name;
7139             pr "  for (i = 0; i < %s->%s_len; ++i)\n" typ name;
7140             pr "    if (c_isprint (%s->%s[i]))\n" typ name;
7141             pr "      printf (\"%%c\", %s->%s[i]);\n" typ name;
7142             pr "    else\n";
7143             pr "      printf (\"\\\\x%%02x\", %s->%s[i]);\n" typ name;
7144             pr "  printf (\"\\n\");\n"
7145         | name, (FUInt64|FBytes) ->
7146             pr "  printf (\"%%s%s: %%\" PRIu64 \"\\n\", indent, %s->%s);\n"
7147               name typ name
7148         | name, FInt64 ->
7149             pr "  printf (\"%%s%s: %%\" PRIi64 \"\\n\", indent, %s->%s);\n"
7150               name typ name
7151         | name, FUInt32 ->
7152             pr "  printf (\"%%s%s: %%\" PRIu32 \"\\n\", indent, %s->%s);\n"
7153               name typ name
7154         | name, FInt32 ->
7155             pr "  printf (\"%%s%s: %%\" PRIi32 \"\\n\", indent, %s->%s);\n"
7156               name typ name
7157         | name, FChar ->
7158             pr "  printf (\"%%s%s: %%c\\n\", indent, %s->%s);\n"
7159               name typ name
7160         | name, FOptPercent ->
7161             pr "  if (%s->%s >= 0) printf (\"%%s%s: %%g %%%%\\n\", indent, %s->%s);\n"
7162               typ name name typ name;
7163             pr "  else printf (\"%%s%s: \\n\", indent);\n" name
7164       ) cols;
7165       pr "}\n";
7166       pr "\n";
7167   ) structs;
7168
7169   (* Emit a print_TYPE_list function definition only if that function is used. *)
7170   List.iter (
7171     function
7172     | typ, (RStructListOnly | RStructAndList) ->
7173         (* generate the function for typ *)
7174         emit_print_list_function typ
7175     | typ, _ -> () (* empty *)
7176   ) (rstructs_used_by all_functions);
7177
7178   (* Emit a print_TYPE function definition only if that function is used. *)
7179   List.iter (
7180     function
7181     | typ, (RStructOnly | RStructAndList) ->
7182         pr "static void print_%s (struct guestfs_%s *%s)\n" typ typ typ;
7183         pr "{\n";
7184         pr "  print_%s_indent (%s, \"\");\n" typ typ;
7185         pr "}\n";
7186         pr "\n";
7187     | typ, _ -> () (* empty *)
7188   ) (rstructs_used_by all_functions);
7189
7190   (* run_<action> actions *)
7191   List.iter (
7192     fun (name, style, _, flags, _, _, _) ->
7193       pr "static int run_%s (const char *cmd, int argc, char *argv[])\n" name;
7194       pr "{\n";
7195       (match fst style with
7196        | RErr
7197        | RInt _
7198        | RBool _ -> pr "  int r;\n"
7199        | RInt64 _ -> pr "  int64_t r;\n"
7200        | RConstString _ | RConstOptString _ -> pr "  const char *r;\n"
7201        | RString _ -> pr "  char *r;\n"
7202        | RStringList _ | RHashtable _ -> pr "  char **r;\n"
7203        | RStruct (_, typ) -> pr "  struct guestfs_%s *r;\n" typ
7204        | RStructList (_, typ) -> pr "  struct guestfs_%s_list *r;\n" typ
7205        | RBufferOut _ ->
7206            pr "  char *r;\n";
7207            pr "  size_t size;\n";
7208       );
7209       List.iter (
7210         function
7211         | Device n
7212         | String n
7213         | OptString n
7214         | FileIn n
7215         | FileOut n -> pr "  const char *%s;\n" n
7216         | Pathname n
7217         | Dev_or_Path n -> pr "  char *%s;\n" n
7218         | StringList n | DeviceList n -> pr "  char **%s;\n" n
7219         | Bool n -> pr "  int %s;\n" n
7220         | Int n -> pr "  int %s;\n" n
7221         | Int64 n -> pr "  int64_t %s;\n" n
7222       ) (snd style);
7223
7224       (* Check and convert parameters. *)
7225       let argc_expected = List.length (snd style) in
7226       pr "  if (argc != %d) {\n" argc_expected;
7227       pr "    fprintf (stderr, _(\"%%s should have %%d parameter(s)\\n\"), cmd, %d);\n"
7228         argc_expected;
7229       pr "    fprintf (stderr, _(\"type 'help %%s' for help on %%s\\n\"), cmd, cmd);\n";
7230       pr "    return -1;\n";
7231       pr "  }\n";
7232
7233       let parse_integer fn fntyp rtyp range name i =
7234         pr "  {\n";
7235         pr "    strtol_error xerr;\n";
7236         pr "    %s r;\n" fntyp;
7237         pr "\n";
7238         pr "    xerr = %s (argv[%d], NULL, 0, &r, \"\");\n" fn i;
7239         pr "    if (xerr != LONGINT_OK) {\n";
7240         pr "      fprintf (stderr,\n";
7241         pr "               _(\"%%s: %%s: invalid integer parameter (%%s returned %%d)\\n\"),\n";
7242         pr "               cmd, \"%s\", \"%s\", xerr);\n" name fn;
7243         pr "      return -1;\n";
7244         pr "    }\n";
7245         (match range with
7246          | None -> ()
7247          | Some (min, max, comment) ->
7248              pr "    /* %s */\n" comment;
7249              pr "    if (r < %s || r > %s) {\n" min max;
7250              pr "      fprintf (stderr, _(\"%%s: %%s: integer out of range\\n\"), cmd, \"%s\");\n"
7251                name;
7252              pr "      return -1;\n";
7253              pr "    }\n";
7254              pr "    /* The check above should ensure this assignment does not overflow. */\n";
7255         );
7256         pr "    %s = r;\n" name;
7257         pr "  }\n";
7258       in
7259
7260       iteri (
7261         fun i ->
7262           function
7263           | Device name
7264           | String name ->
7265               pr "  %s = argv[%d];\n" name i
7266           | Pathname name
7267           | Dev_or_Path name ->
7268               pr "  %s = resolve_win_path (argv[%d]);\n" name i;
7269               pr "  if (%s == NULL) return -1;\n" name
7270           | OptString name ->
7271               pr "  %s = STRNEQ (argv[%d], \"\") ? argv[%d] : NULL;\n"
7272                 name i i
7273           | FileIn name ->
7274               pr "  %s = STRNEQ (argv[%d], \"-\") ? argv[%d] : \"/dev/stdin\";\n"
7275                 name i i
7276           | FileOut name ->
7277               pr "  %s = STRNEQ (argv[%d], \"-\") ? argv[%d] : \"/dev/stdout\";\n"
7278                 name i i
7279           | StringList name | DeviceList name ->
7280               pr "  %s = parse_string_list (argv[%d]);\n" name i;
7281               pr "  if (%s == NULL) return -1;\n" name;
7282           | Bool name ->
7283               pr "  %s = is_true (argv[%d]) ? 1 : 0;\n" name i
7284           | Int name ->
7285               let range =
7286                 let min = "(-(2LL<<30))"
7287                 and max = "((2LL<<30)-1)"
7288                 and comment =
7289                   "The Int type in the generator is a signed 31 bit int." in
7290                 Some (min, max, comment) in
7291               parse_integer "xstrtoll" "long long" "int" range name i
7292           | Int64 name ->
7293               parse_integer "xstrtoll" "long long" "int64_t" None name i
7294       ) (snd style);
7295
7296       (* Call C API function. *)
7297       let fn =
7298         try find_map (function FishAction n -> Some n | _ -> None) flags
7299         with Not_found -> sprintf "guestfs_%s" name in
7300       pr "  r = %s " fn;
7301       generate_c_call_args ~handle:"g" style;
7302       pr ";\n";
7303
7304       List.iter (
7305         function
7306         | Device name | String name
7307         | OptString name | FileIn name | FileOut name | Bool name
7308         | Int name | Int64 name -> ()
7309         | Pathname name | Dev_or_Path name ->
7310             pr "  free (%s);\n" name
7311         | StringList name | DeviceList name ->
7312             pr "  free_strings (%s);\n" name
7313       ) (snd style);
7314
7315       (* Check return value for errors and display command results. *)
7316       (match fst style with
7317        | RErr -> pr "  return r;\n"
7318        | RInt _ ->
7319            pr "  if (r == -1) return -1;\n";
7320            pr "  printf (\"%%d\\n\", r);\n";
7321            pr "  return 0;\n"
7322        | RInt64 _ ->
7323            pr "  if (r == -1) return -1;\n";
7324            pr "  printf (\"%%\" PRIi64 \"\\n\", r);\n";
7325            pr "  return 0;\n"
7326        | RBool _ ->
7327            pr "  if (r == -1) return -1;\n";
7328            pr "  if (r) printf (\"true\\n\"); else printf (\"false\\n\");\n";
7329            pr "  return 0;\n"
7330        | RConstString _ ->
7331            pr "  if (r == NULL) return -1;\n";
7332            pr "  printf (\"%%s\\n\", r);\n";
7333            pr "  return 0;\n"
7334        | RConstOptString _ ->
7335            pr "  printf (\"%%s\\n\", r ? : \"(null)\");\n";
7336            pr "  return 0;\n"
7337        | RString _ ->
7338            pr "  if (r == NULL) return -1;\n";
7339            pr "  printf (\"%%s\\n\", r);\n";
7340            pr "  free (r);\n";
7341            pr "  return 0;\n"
7342        | RStringList _ ->
7343            pr "  if (r == NULL) return -1;\n";
7344            pr "  print_strings (r);\n";
7345            pr "  free_strings (r);\n";
7346            pr "  return 0;\n"
7347        | RStruct (_, typ) ->
7348            pr "  if (r == NULL) return -1;\n";
7349            pr "  print_%s (r);\n" typ;
7350            pr "  guestfs_free_%s (r);\n" typ;
7351            pr "  return 0;\n"
7352        | RStructList (_, typ) ->
7353            pr "  if (r == NULL) return -1;\n";
7354            pr "  print_%s_list (r);\n" typ;
7355            pr "  guestfs_free_%s_list (r);\n" typ;
7356            pr "  return 0;\n"
7357        | RHashtable _ ->
7358            pr "  if (r == NULL) return -1;\n";
7359            pr "  print_table (r);\n";
7360            pr "  free_strings (r);\n";
7361            pr "  return 0;\n"
7362        | RBufferOut _ ->
7363            pr "  if (r == NULL) return -1;\n";
7364            pr "  if (full_write (1, r, size) != size) {\n";
7365            pr "    perror (\"write\");\n";
7366            pr "    free (r);\n";
7367            pr "    return -1;\n";
7368            pr "  }\n";
7369            pr "  free (r);\n";
7370            pr "  return 0;\n"
7371       );
7372       pr "}\n";
7373       pr "\n"
7374   ) all_functions;
7375
7376   (* run_action function *)
7377   pr "int run_action (const char *cmd, int argc, char *argv[])\n";
7378   pr "{\n";
7379   List.iter (
7380     fun (name, _, _, flags, _, _, _) ->
7381       let name2 = replace_char name '_' '-' in
7382       let alias =
7383         try find_map (function FishAlias n -> Some n | _ -> None) flags
7384         with Not_found -> name in
7385       pr "  if (";
7386       pr "STRCASEEQ (cmd, \"%s\")" name;
7387       if name <> name2 then
7388         pr " || STRCASEEQ (cmd, \"%s\")" name2;
7389       if name <> alias then
7390         pr " || STRCASEEQ (cmd, \"%s\")" alias;
7391       pr ")\n";
7392       pr "    return run_%s (cmd, argc, argv);\n" name;
7393       pr "  else\n";
7394   ) all_functions;
7395   pr "    {\n";
7396   pr "      fprintf (stderr, _(\"%%s: unknown command\\n\"), cmd);\n";
7397   pr "      if (command_num == 1)\n";
7398   pr "        extended_help_message ();\n";
7399   pr "      return -1;\n";
7400   pr "    }\n";
7401   pr "  return 0;\n";
7402   pr "}\n";
7403   pr "\n"
7404
7405 (* Readline completion for guestfish. *)
7406 and generate_fish_completion () =
7407   generate_header CStyle GPLv2plus;
7408
7409   let all_functions =
7410     List.filter (
7411       fun (_, _, _, flags, _, _, _) -> not (List.mem NotInFish flags)
7412     ) all_functions in
7413
7414   pr "\
7415 #include <config.h>
7416
7417 #include <stdio.h>
7418 #include <stdlib.h>
7419 #include <string.h>
7420
7421 #ifdef HAVE_LIBREADLINE
7422 #include <readline/readline.h>
7423 #endif
7424
7425 #include \"fish.h\"
7426
7427 #ifdef HAVE_LIBREADLINE
7428
7429 static const char *const commands[] = {
7430   BUILTIN_COMMANDS_FOR_COMPLETION,
7431 ";
7432
7433   (* Get the commands, including the aliases.  They don't need to be
7434    * sorted - the generator() function just does a dumb linear search.
7435    *)
7436   let commands =
7437     List.map (
7438       fun (name, _, _, flags, _, _, _) ->
7439         let name2 = replace_char name '_' '-' in
7440         let alias =
7441           try find_map (function FishAlias n -> Some n | _ -> None) flags
7442           with Not_found -> name in
7443
7444         if name <> alias then [name2; alias] else [name2]
7445     ) all_functions in
7446   let commands = List.flatten commands in
7447
7448   List.iter (pr "  \"%s\",\n") commands;
7449
7450   pr "  NULL
7451 };
7452
7453 static char *
7454 generator (const char *text, int state)
7455 {
7456   static int index, len;
7457   const char *name;
7458
7459   if (!state) {
7460     index = 0;
7461     len = strlen (text);
7462   }
7463
7464   rl_attempted_completion_over = 1;
7465
7466   while ((name = commands[index]) != NULL) {
7467     index++;
7468     if (STRCASEEQLEN (name, text, len))
7469       return strdup (name);
7470   }
7471
7472   return NULL;
7473 }
7474
7475 #endif /* HAVE_LIBREADLINE */
7476
7477 #ifdef HAVE_RL_COMPLETION_MATCHES
7478 #define RL_COMPLETION_MATCHES rl_completion_matches
7479 #else
7480 #ifdef HAVE_COMPLETION_MATCHES
7481 #define RL_COMPLETION_MATCHES completion_matches
7482 #endif
7483 #endif /* else just fail if we don't have either symbol */
7484
7485 char **
7486 do_completion (const char *text, int start, int end)
7487 {
7488   char **matches = NULL;
7489
7490 #ifdef HAVE_LIBREADLINE
7491   rl_completion_append_character = ' ';
7492
7493   if (start == 0)
7494     matches = RL_COMPLETION_MATCHES (text, generator);
7495   else if (complete_dest_paths)
7496     matches = RL_COMPLETION_MATCHES (text, complete_dest_paths_generator);
7497 #endif
7498
7499   return matches;
7500 }
7501 ";
7502
7503 (* Generate the POD documentation for guestfish. *)
7504 and generate_fish_actions_pod () =
7505   let all_functions_sorted =
7506     List.filter (
7507       fun (_, _, _, flags, _, _, _) ->
7508         not (List.mem NotInFish flags || List.mem NotInDocs flags)
7509     ) all_functions_sorted in
7510
7511   let rex = Str.regexp "C<guestfs_\\([^>]+\\)>" in
7512
7513   List.iter (
7514     fun (name, style, _, flags, _, _, longdesc) ->
7515       let longdesc =
7516         Str.global_substitute rex (
7517           fun s ->
7518             let sub =
7519               try Str.matched_group 1 s
7520               with Not_found ->
7521                 failwithf "error substituting C<guestfs_...> in longdesc of function %s" name in
7522             "C<" ^ replace_char sub '_' '-' ^ ">"
7523         ) longdesc in
7524       let name = replace_char name '_' '-' in
7525       let alias =
7526         try find_map (function FishAlias n -> Some n | _ -> None) flags
7527         with Not_found -> name in
7528
7529       pr "=head2 %s" name;
7530       if name <> alias then
7531         pr " | %s" alias;
7532       pr "\n";
7533       pr "\n";
7534       pr " %s" name;
7535       List.iter (
7536         function
7537         | Pathname n | Device n | Dev_or_Path n | String n -> pr " %s" n
7538         | OptString n -> pr " %s" n
7539         | StringList n | DeviceList n -> pr " '%s ...'" n
7540         | Bool _ -> pr " true|false"
7541         | Int n -> pr " %s" n
7542         | Int64 n -> pr " %s" n
7543         | FileIn n | FileOut n -> pr " (%s|-)" n
7544       ) (snd style);
7545       pr "\n";
7546       pr "\n";
7547       pr "%s\n\n" longdesc;
7548
7549       if List.exists (function FileIn _ | FileOut _ -> true
7550                       | _ -> false) (snd style) then
7551         pr "Use C<-> instead of a filename to read/write from stdin/stdout.\n\n";
7552
7553       if List.mem ProtocolLimitWarning flags then
7554         pr "%s\n\n" protocol_limit_warning;
7555
7556       if List.mem DangerWillRobinson flags then
7557         pr "%s\n\n" danger_will_robinson;
7558
7559       match deprecation_notice flags with
7560       | None -> ()
7561       | Some txt -> pr "%s\n\n" txt
7562   ) all_functions_sorted
7563
7564 (* Generate a C function prototype. *)
7565 and generate_prototype ?(extern = true) ?(static = false) ?(semicolon = true)
7566     ?(single_line = false) ?(newline = false) ?(in_daemon = false)
7567     ?(prefix = "")
7568     ?handle name style =
7569   if extern then pr "extern ";
7570   if static then pr "static ";
7571   (match fst style with
7572    | RErr -> pr "int "
7573    | RInt _ -> pr "int "
7574    | RInt64 _ -> pr "int64_t "
7575    | RBool _ -> pr "int "
7576    | RConstString _ | RConstOptString _ -> pr "const char *"
7577    | RString _ | RBufferOut _ -> pr "char *"
7578    | RStringList _ | RHashtable _ -> pr "char **"
7579    | RStruct (_, typ) ->
7580        if not in_daemon then pr "struct guestfs_%s *" typ
7581        else pr "guestfs_int_%s *" typ
7582    | RStructList (_, typ) ->
7583        if not in_daemon then pr "struct guestfs_%s_list *" typ
7584        else pr "guestfs_int_%s_list *" typ
7585   );
7586   let is_RBufferOut = match fst style with RBufferOut _ -> true | _ -> false in
7587   pr "%s%s (" prefix name;
7588   if handle = None && List.length (snd style) = 0 && not is_RBufferOut then
7589     pr "void"
7590   else (
7591     let comma = ref false in
7592     (match handle with
7593      | None -> ()
7594      | Some handle -> pr "guestfs_h *%s" handle; comma := true
7595     );
7596     let next () =
7597       if !comma then (
7598         if single_line then pr ", " else pr ",\n\t\t"
7599       );
7600       comma := true
7601     in
7602     List.iter (
7603       function
7604       | Pathname n
7605       | Device n | Dev_or_Path n
7606       | String n
7607       | OptString n ->
7608           next ();
7609           pr "const char *%s" n
7610       | StringList n | DeviceList n ->
7611           next ();
7612           pr "char *const *%s" n
7613       | Bool n -> next (); pr "int %s" n
7614       | Int n -> next (); pr "int %s" n
7615       | Int64 n -> next (); pr "int64_t %s" n
7616       | FileIn n
7617       | FileOut n ->
7618           if not in_daemon then (next (); pr "const char *%s" n)
7619     ) (snd style);
7620     if is_RBufferOut then (next (); pr "size_t *size_r");
7621   );
7622   pr ")";
7623   if semicolon then pr ";";
7624   if newline then pr "\n"
7625
7626 (* Generate C call arguments, eg "(handle, foo, bar)" *)
7627 and generate_c_call_args ?handle ?(decl = false) style =
7628   pr "(";
7629   let comma = ref false in
7630   let next () =
7631     if !comma then pr ", ";
7632     comma := true
7633   in
7634   (match handle with
7635    | None -> ()
7636    | Some handle -> pr "%s" handle; comma := true
7637   );
7638   List.iter (
7639     fun arg ->
7640       next ();
7641       pr "%s" (name_of_argt arg)
7642   ) (snd style);
7643   (* For RBufferOut calls, add implicit &size parameter. *)
7644   if not decl then (
7645     match fst style with
7646     | RBufferOut _ ->
7647         next ();
7648         pr "&size"
7649     | _ -> ()
7650   );
7651   pr ")"
7652
7653 (* Generate the OCaml bindings interface. *)
7654 and generate_ocaml_mli () =
7655   generate_header OCamlStyle LGPLv2plus;
7656
7657   pr "\
7658 (** For API documentation you should refer to the C API
7659     in the guestfs(3) manual page.  The OCaml API uses almost
7660     exactly the same calls. *)
7661
7662 type t
7663 (** A [guestfs_h] handle. *)
7664
7665 exception Error of string
7666 (** This exception is raised when there is an error. *)
7667
7668 exception Handle_closed of string
7669 (** This exception is raised if you use a {!Guestfs.t} handle
7670     after calling {!close} on it.  The string is the name of
7671     the function. *)
7672
7673 val create : unit -> t
7674 (** Create a {!Guestfs.t} handle. *)
7675
7676 val close : t -> unit
7677 (** Close the {!Guestfs.t} handle and free up all resources used
7678     by it immediately.
7679
7680     Handles are closed by the garbage collector when they become
7681     unreferenced, but callers can call this in order to provide
7682     predictable cleanup. *)
7683
7684 ";
7685   generate_ocaml_structure_decls ();
7686
7687   (* The actions. *)
7688   List.iter (
7689     fun (name, style, _, _, _, shortdesc, _) ->
7690       generate_ocaml_prototype name style;
7691       pr "(** %s *)\n" shortdesc;
7692       pr "\n"
7693   ) all_functions_sorted
7694
7695 (* Generate the OCaml bindings implementation. *)
7696 and generate_ocaml_ml () =
7697   generate_header OCamlStyle LGPLv2plus;
7698
7699   pr "\
7700 type t
7701
7702 exception Error of string
7703 exception Handle_closed of string
7704
7705 external create : unit -> t = \"ocaml_guestfs_create\"
7706 external close : t -> unit = \"ocaml_guestfs_close\"
7707
7708 (* Give the exceptions names, so they can be raised from the C code. *)
7709 let () =
7710   Callback.register_exception \"ocaml_guestfs_error\" (Error \"\");
7711   Callback.register_exception \"ocaml_guestfs_closed\" (Handle_closed \"\")
7712
7713 ";
7714
7715   generate_ocaml_structure_decls ();
7716
7717   (* The actions. *)
7718   List.iter (
7719     fun (name, style, _, _, _, shortdesc, _) ->
7720       generate_ocaml_prototype ~is_external:true name style;
7721   ) all_functions_sorted
7722
7723 (* Generate the OCaml bindings C implementation. *)
7724 and generate_ocaml_c () =
7725   generate_header CStyle LGPLv2plus;
7726
7727   pr "\
7728 #include <stdio.h>
7729 #include <stdlib.h>
7730 #include <string.h>
7731
7732 #include <caml/config.h>
7733 #include <caml/alloc.h>
7734 #include <caml/callback.h>
7735 #include <caml/fail.h>
7736 #include <caml/memory.h>
7737 #include <caml/mlvalues.h>
7738 #include <caml/signals.h>
7739
7740 #include <guestfs.h>
7741
7742 #include \"guestfs_c.h\"
7743
7744 /* Copy a hashtable of string pairs into an assoc-list.  We return
7745  * the list in reverse order, but hashtables aren't supposed to be
7746  * ordered anyway.
7747  */
7748 static CAMLprim value
7749 copy_table (char * const * argv)
7750 {
7751   CAMLparam0 ();
7752   CAMLlocal5 (rv, pairv, kv, vv, cons);
7753   int i;
7754
7755   rv = Val_int (0);
7756   for (i = 0; argv[i] != NULL; i += 2) {
7757     kv = caml_copy_string (argv[i]);
7758     vv = caml_copy_string (argv[i+1]);
7759     pairv = caml_alloc (2, 0);
7760     Store_field (pairv, 0, kv);
7761     Store_field (pairv, 1, vv);
7762     cons = caml_alloc (2, 0);
7763     Store_field (cons, 1, rv);
7764     rv = cons;
7765     Store_field (cons, 0, pairv);
7766   }
7767
7768   CAMLreturn (rv);
7769 }
7770
7771 ";
7772
7773   (* Struct copy functions. *)
7774
7775   let emit_ocaml_copy_list_function typ =
7776     pr "static CAMLprim value\n";
7777     pr "copy_%s_list (const struct guestfs_%s_list *%ss)\n" typ typ typ;
7778     pr "{\n";
7779     pr "  CAMLparam0 ();\n";
7780     pr "  CAMLlocal2 (rv, v);\n";
7781     pr "  unsigned int i;\n";
7782     pr "\n";
7783     pr "  if (%ss->len == 0)\n" typ;
7784     pr "    CAMLreturn (Atom (0));\n";
7785     pr "  else {\n";
7786     pr "    rv = caml_alloc (%ss->len, 0);\n" typ;
7787     pr "    for (i = 0; i < %ss->len; ++i) {\n" typ;
7788     pr "      v = copy_%s (&%ss->val[i]);\n" typ typ;
7789     pr "      caml_modify (&Field (rv, i), v);\n";
7790     pr "    }\n";
7791     pr "    CAMLreturn (rv);\n";
7792     pr "  }\n";
7793     pr "}\n";
7794     pr "\n";
7795   in
7796
7797   List.iter (
7798     fun (typ, cols) ->
7799       let has_optpercent_col =
7800         List.exists (function (_, FOptPercent) -> true | _ -> false) cols in
7801
7802       pr "static CAMLprim value\n";
7803       pr "copy_%s (const struct guestfs_%s *%s)\n" typ typ typ;
7804       pr "{\n";
7805       pr "  CAMLparam0 ();\n";
7806       if has_optpercent_col then
7807         pr "  CAMLlocal3 (rv, v, v2);\n"
7808       else
7809         pr "  CAMLlocal2 (rv, v);\n";
7810       pr "\n";
7811       pr "  rv = caml_alloc (%d, 0);\n" (List.length cols);
7812       iteri (
7813         fun i col ->
7814           (match col with
7815            | name, FString ->
7816                pr "  v = caml_copy_string (%s->%s);\n" typ name
7817            | name, FBuffer ->
7818                pr "  v = caml_alloc_string (%s->%s_len);\n" typ name;
7819                pr "  memcpy (String_val (v), %s->%s, %s->%s_len);\n"
7820                  typ name typ name
7821            | name, FUUID ->
7822                pr "  v = caml_alloc_string (32);\n";
7823                pr "  memcpy (String_val (v), %s->%s, 32);\n" typ name
7824            | name, (FBytes|FInt64|FUInt64) ->
7825                pr "  v = caml_copy_int64 (%s->%s);\n" typ name
7826            | name, (FInt32|FUInt32) ->
7827                pr "  v = caml_copy_int32 (%s->%s);\n" typ name
7828            | name, FOptPercent ->
7829                pr "  if (%s->%s >= 0) { /* Some %s */\n" typ name name;
7830                pr "    v2 = caml_copy_double (%s->%s);\n" typ name;
7831                pr "    v = caml_alloc (1, 0);\n";
7832                pr "    Store_field (v, 0, v2);\n";
7833                pr "  } else /* None */\n";
7834                pr "    v = Val_int (0);\n";
7835            | name, FChar ->
7836                pr "  v = Val_int (%s->%s);\n" typ name
7837           );
7838           pr "  Store_field (rv, %d, v);\n" i
7839       ) cols;
7840       pr "  CAMLreturn (rv);\n";
7841       pr "}\n";
7842       pr "\n";
7843   ) structs;
7844
7845   (* Emit a copy_TYPE_list function definition only if that function is used. *)
7846   List.iter (
7847     function
7848     | typ, (RStructListOnly | RStructAndList) ->
7849         (* generate the function for typ *)
7850         emit_ocaml_copy_list_function typ
7851     | typ, _ -> () (* empty *)
7852   ) (rstructs_used_by all_functions);
7853
7854   (* The wrappers. *)
7855   List.iter (
7856     fun (name, style, _, _, _, _, _) ->
7857       pr "/* Automatically generated wrapper for function\n";
7858       pr " * ";
7859       generate_ocaml_prototype name style;
7860       pr " */\n";
7861       pr "\n";
7862
7863       let params =
7864         "gv" :: List.map (fun arg -> name_of_argt arg ^ "v") (snd style) in
7865
7866       let needs_extra_vs =
7867         match fst style with RConstOptString _ -> true | _ -> false in
7868
7869       pr "/* Emit prototype to appease gcc's -Wmissing-prototypes. */\n";
7870       pr "CAMLprim value ocaml_guestfs_%s (value %s" name (List.hd params);
7871       List.iter (pr ", value %s") (List.tl params); pr ");\n";
7872       pr "\n";
7873
7874       pr "CAMLprim value\n";
7875       pr "ocaml_guestfs_%s (value %s" name (List.hd params);
7876       List.iter (pr ", value %s") (List.tl params);
7877       pr ")\n";
7878       pr "{\n";
7879
7880       (match params with
7881        | [p1; p2; p3; p4; p5] ->
7882            pr "  CAMLparam5 (%s);\n" (String.concat ", " params)
7883        | p1 :: p2 :: p3 :: p4 :: p5 :: rest ->
7884            pr "  CAMLparam5 (%s);\n" (String.concat ", " [p1; p2; p3; p4; p5]);
7885            pr "  CAMLxparam%d (%s);\n"
7886              (List.length rest) (String.concat ", " rest)
7887        | ps ->
7888            pr "  CAMLparam%d (%s);\n" (List.length ps) (String.concat ", " ps)
7889       );
7890       if not needs_extra_vs then
7891         pr "  CAMLlocal1 (rv);\n"
7892       else
7893         pr "  CAMLlocal3 (rv, v, v2);\n";
7894       pr "\n";
7895
7896       pr "  guestfs_h *g = Guestfs_val (gv);\n";
7897       pr "  if (g == NULL)\n";
7898       pr "    ocaml_guestfs_raise_closed (\"%s\");\n" name;
7899       pr "\n";
7900
7901       List.iter (
7902         function
7903         | Pathname n
7904         | Device n | Dev_or_Path n
7905         | String n
7906         | FileIn n
7907         | FileOut n ->
7908             pr "  const char *%s = String_val (%sv);\n" n n
7909         | OptString n ->
7910             pr "  const char *%s =\n" n;
7911             pr "    %sv != Val_int (0) ? String_val (Field (%sv, 0)) : NULL;\n"
7912               n n
7913         | StringList n | DeviceList n ->
7914             pr "  char **%s = ocaml_guestfs_strings_val (g, %sv);\n" n n
7915         | Bool n ->
7916             pr "  int %s = Bool_val (%sv);\n" n n
7917         | Int n ->
7918             pr "  int %s = Int_val (%sv);\n" n n
7919         | Int64 n ->
7920             pr "  int64_t %s = Int64_val (%sv);\n" n n
7921       ) (snd style);
7922       let error_code =
7923         match fst style with
7924         | RErr -> pr "  int r;\n"; "-1"
7925         | RInt _ -> pr "  int r;\n"; "-1"
7926         | RInt64 _ -> pr "  int64_t r;\n"; "-1"
7927         | RBool _ -> pr "  int r;\n"; "-1"
7928         | RConstString _ | RConstOptString _ ->
7929             pr "  const char *r;\n"; "NULL"
7930         | RString _ -> pr "  char *r;\n"; "NULL"
7931         | RStringList _ ->
7932             pr "  int i;\n";
7933             pr "  char **r;\n";
7934             "NULL"
7935         | RStruct (_, typ) ->
7936             pr "  struct guestfs_%s *r;\n" typ; "NULL"
7937         | RStructList (_, typ) ->
7938             pr "  struct guestfs_%s_list *r;\n" typ; "NULL"
7939         | RHashtable _ ->
7940             pr "  int i;\n";
7941             pr "  char **r;\n";
7942             "NULL"
7943         | RBufferOut _ ->
7944             pr "  char *r;\n";
7945             pr "  size_t size;\n";
7946             "NULL" in
7947       pr "\n";
7948
7949       pr "  caml_enter_blocking_section ();\n";
7950       pr "  r = guestfs_%s " name;
7951       generate_c_call_args ~handle:"g" style;
7952       pr ";\n";
7953       pr "  caml_leave_blocking_section ();\n";
7954
7955       List.iter (
7956         function
7957         | StringList n | DeviceList n ->
7958             pr "  ocaml_guestfs_free_strings (%s);\n" n;
7959         | Pathname _ | Device _ | Dev_or_Path _ | String _ | OptString _
7960         | Bool _ | Int _ | Int64 _
7961         | FileIn _ | FileOut _ -> ()
7962       ) (snd style);
7963
7964       pr "  if (r == %s)\n" error_code;
7965       pr "    ocaml_guestfs_raise_error (g, \"%s\");\n" name;
7966       pr "\n";
7967
7968       (match fst style with
7969        | RErr -> pr "  rv = Val_unit;\n"
7970        | RInt _ -> pr "  rv = Val_int (r);\n"
7971        | RInt64 _ ->
7972            pr "  rv = caml_copy_int64 (r);\n"
7973        | RBool _ -> pr "  rv = Val_bool (r);\n"
7974        | RConstString _ ->
7975            pr "  rv = caml_copy_string (r);\n"
7976        | RConstOptString _ ->
7977            pr "  if (r) { /* Some string */\n";
7978            pr "    v = caml_alloc (1, 0);\n";
7979            pr "    v2 = caml_copy_string (r);\n";
7980            pr "    Store_field (v, 0, v2);\n";
7981            pr "  } else /* None */\n";
7982            pr "    v = Val_int (0);\n";
7983        | RString _ ->
7984            pr "  rv = caml_copy_string (r);\n";
7985            pr "  free (r);\n"
7986        | RStringList _ ->
7987            pr "  rv = caml_copy_string_array ((const char **) r);\n";
7988            pr "  for (i = 0; r[i] != NULL; ++i) free (r[i]);\n";
7989            pr "  free (r);\n"
7990        | RStruct (_, typ) ->
7991            pr "  rv = copy_%s (r);\n" typ;
7992            pr "  guestfs_free_%s (r);\n" typ;
7993        | RStructList (_, typ) ->
7994            pr "  rv = copy_%s_list (r);\n" typ;
7995            pr "  guestfs_free_%s_list (r);\n" typ;
7996        | RHashtable _ ->
7997            pr "  rv = copy_table (r);\n";
7998            pr "  for (i = 0; r[i] != NULL; ++i) free (r[i]);\n";
7999            pr "  free (r);\n";
8000        | RBufferOut _ ->
8001            pr "  rv = caml_alloc_string (size);\n";
8002            pr "  memcpy (String_val (rv), r, size);\n";
8003       );
8004
8005       pr "  CAMLreturn (rv);\n";
8006       pr "}\n";
8007       pr "\n";
8008
8009       if List.length params > 5 then (
8010         pr "/* Emit prototype to appease gcc's -Wmissing-prototypes. */\n";
8011         pr "CAMLprim value ";
8012         pr "ocaml_guestfs_%s_byte (value *argv, int argn);\n" name;
8013         pr "CAMLprim value\n";
8014         pr "ocaml_guestfs_%s_byte (value *argv, int argn)\n" name;
8015         pr "{\n";
8016         pr "  return ocaml_guestfs_%s (argv[0]" name;
8017         iteri (fun i _ -> pr ", argv[%d]" i) (List.tl params);
8018         pr ");\n";
8019         pr "}\n";
8020         pr "\n"
8021       )
8022   ) all_functions_sorted
8023
8024 and generate_ocaml_structure_decls () =
8025   List.iter (
8026     fun (typ, cols) ->
8027       pr "type %s = {\n" typ;
8028       List.iter (
8029         function
8030         | name, FString -> pr "  %s : string;\n" name
8031         | name, FBuffer -> pr "  %s : string;\n" name
8032         | name, FUUID -> pr "  %s : string;\n" name
8033         | name, (FBytes|FInt64|FUInt64) -> pr "  %s : int64;\n" name
8034         | name, (FInt32|FUInt32) -> pr "  %s : int32;\n" name
8035         | name, FChar -> pr "  %s : char;\n" name
8036         | name, FOptPercent -> pr "  %s : float option;\n" name
8037       ) cols;
8038       pr "}\n";
8039       pr "\n"
8040   ) structs
8041
8042 and generate_ocaml_prototype ?(is_external = false) name style =
8043   if is_external then pr "external " else pr "val ";
8044   pr "%s : t -> " name;
8045   List.iter (
8046     function
8047     | Pathname _ | Device _ | Dev_or_Path _ | String _ | FileIn _ | FileOut _ -> pr "string -> "
8048     | OptString _ -> pr "string option -> "
8049     | StringList _ | DeviceList _ -> pr "string array -> "
8050     | Bool _ -> pr "bool -> "
8051     | Int _ -> pr "int -> "
8052     | Int64 _ -> pr "int64 -> "
8053   ) (snd style);
8054   (match fst style with
8055    | RErr -> pr "unit" (* all errors are turned into exceptions *)
8056    | RInt _ -> pr "int"
8057    | RInt64 _ -> pr "int64"
8058    | RBool _ -> pr "bool"
8059    | RConstString _ -> pr "string"
8060    | RConstOptString _ -> pr "string option"
8061    | RString _ | RBufferOut _ -> pr "string"
8062    | RStringList _ -> pr "string array"
8063    | RStruct (_, typ) -> pr "%s" typ
8064    | RStructList (_, typ) -> pr "%s array" typ
8065    | RHashtable _ -> pr "(string * string) list"
8066   );
8067   if is_external then (
8068     pr " = ";
8069     if List.length (snd style) + 1 > 5 then
8070       pr "\"ocaml_guestfs_%s_byte\" " name;
8071     pr "\"ocaml_guestfs_%s\"" name
8072   );
8073   pr "\n"
8074
8075 (* Generate Perl xs code, a sort of crazy variation of C with macros. *)
8076 and generate_perl_xs () =
8077   generate_header CStyle LGPLv2plus;
8078
8079   pr "\
8080 #include \"EXTERN.h\"
8081 #include \"perl.h\"
8082 #include \"XSUB.h\"
8083
8084 #include <guestfs.h>
8085
8086 #ifndef PRId64
8087 #define PRId64 \"lld\"
8088 #endif
8089
8090 static SV *
8091 my_newSVll(long long val) {
8092 #ifdef USE_64_BIT_ALL
8093   return newSViv(val);
8094 #else
8095   char buf[100];
8096   int len;
8097   len = snprintf(buf, 100, \"%%\" PRId64, val);
8098   return newSVpv(buf, len);
8099 #endif
8100 }
8101
8102 #ifndef PRIu64
8103 #define PRIu64 \"llu\"
8104 #endif
8105
8106 static SV *
8107 my_newSVull(unsigned long long val) {
8108 #ifdef USE_64_BIT_ALL
8109   return newSVuv(val);
8110 #else
8111   char buf[100];
8112   int len;
8113   len = snprintf(buf, 100, \"%%\" PRIu64, val);
8114   return newSVpv(buf, len);
8115 #endif
8116 }
8117
8118 /* http://www.perlmonks.org/?node_id=680842 */
8119 static char **
8120 XS_unpack_charPtrPtr (SV *arg) {
8121   char **ret;
8122   AV *av;
8123   I32 i;
8124
8125   if (!arg || !SvOK (arg) || !SvROK (arg) || SvTYPE (SvRV (arg)) != SVt_PVAV)
8126     croak (\"array reference expected\");
8127
8128   av = (AV *)SvRV (arg);
8129   ret = malloc ((av_len (av) + 1 + 1) * sizeof (char *));
8130   if (!ret)
8131     croak (\"malloc failed\");
8132
8133   for (i = 0; i <= av_len (av); i++) {
8134     SV **elem = av_fetch (av, i, 0);
8135
8136     if (!elem || !*elem)
8137       croak (\"missing element in list\");
8138
8139     ret[i] = SvPV_nolen (*elem);
8140   }
8141
8142   ret[i] = NULL;
8143
8144   return ret;
8145 }
8146
8147 MODULE = Sys::Guestfs  PACKAGE = Sys::Guestfs
8148
8149 PROTOTYPES: ENABLE
8150
8151 guestfs_h *
8152 _create ()
8153    CODE:
8154       RETVAL = guestfs_create ();
8155       if (!RETVAL)
8156         croak (\"could not create guestfs handle\");
8157       guestfs_set_error_handler (RETVAL, NULL, NULL);
8158  OUTPUT:
8159       RETVAL
8160
8161 void
8162 DESTROY (g)
8163       guestfs_h *g;
8164  PPCODE:
8165       guestfs_close (g);
8166
8167 ";
8168
8169   List.iter (
8170     fun (name, style, _, _, _, _, _) ->
8171       (match fst style with
8172        | RErr -> pr "void\n"
8173        | RInt _ -> pr "SV *\n"
8174        | RInt64 _ -> pr "SV *\n"
8175        | RBool _ -> pr "SV *\n"
8176        | RConstString _ -> pr "SV *\n"
8177        | RConstOptString _ -> pr "SV *\n"
8178        | RString _ -> pr "SV *\n"
8179        | RBufferOut _ -> pr "SV *\n"
8180        | RStringList _
8181        | RStruct _ | RStructList _
8182        | RHashtable _ ->
8183            pr "void\n" (* all lists returned implictly on the stack *)
8184       );
8185       (* Call and arguments. *)
8186       pr "%s " name;
8187       generate_c_call_args ~handle:"g" ~decl:true style;
8188       pr "\n";
8189       pr "      guestfs_h *g;\n";
8190       iteri (
8191         fun i ->
8192           function
8193           | Pathname n | Device n | Dev_or_Path n | String n | FileIn n | FileOut n ->
8194               pr "      char *%s;\n" n
8195           | OptString n ->
8196               (* http://www.perlmonks.org/?node_id=554277
8197                * Note that the implicit handle argument means we have
8198                * to add 1 to the ST(x) operator.
8199                *)
8200               pr "      char *%s = SvOK(ST(%d)) ? SvPV_nolen(ST(%d)) : NULL;\n" n (i+1) (i+1)
8201           | StringList n | DeviceList n -> pr "      char **%s;\n" n
8202           | Bool n -> pr "      int %s;\n" n
8203           | Int n -> pr "      int %s;\n" n
8204           | Int64 n -> pr "      int64_t %s;\n" n
8205       ) (snd style);
8206
8207       let do_cleanups () =
8208         List.iter (
8209           function
8210           | Pathname _ | Device _ | Dev_or_Path _ | String _ | OptString _
8211           | Bool _ | Int _ | Int64 _
8212           | FileIn _ | FileOut _ -> ()
8213           | StringList n | DeviceList n -> pr "      free (%s);\n" n
8214         ) (snd style)
8215       in
8216
8217       (* Code. *)
8218       (match fst style with
8219        | RErr ->
8220            pr "PREINIT:\n";
8221            pr "      int r;\n";
8222            pr " PPCODE:\n";
8223            pr "      r = guestfs_%s " name;
8224            generate_c_call_args ~handle:"g" style;
8225            pr ";\n";
8226            do_cleanups ();
8227            pr "      if (r == -1)\n";
8228            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8229        | RInt n
8230        | RBool n ->
8231            pr "PREINIT:\n";
8232            pr "      int %s;\n" n;
8233            pr "   CODE:\n";
8234            pr "      %s = guestfs_%s " n name;
8235            generate_c_call_args ~handle:"g" style;
8236            pr ";\n";
8237            do_cleanups ();
8238            pr "      if (%s == -1)\n" n;
8239            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8240            pr "      RETVAL = newSViv (%s);\n" n;
8241            pr " OUTPUT:\n";
8242            pr "      RETVAL\n"
8243        | RInt64 n ->
8244            pr "PREINIT:\n";
8245            pr "      int64_t %s;\n" n;
8246            pr "   CODE:\n";
8247            pr "      %s = guestfs_%s " n name;
8248            generate_c_call_args ~handle:"g" style;
8249            pr ";\n";
8250            do_cleanups ();
8251            pr "      if (%s == -1)\n" n;
8252            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8253            pr "      RETVAL = my_newSVll (%s);\n" n;
8254            pr " OUTPUT:\n";
8255            pr "      RETVAL\n"
8256        | RConstString n ->
8257            pr "PREINIT:\n";
8258            pr "      const char *%s;\n" n;
8259            pr "   CODE:\n";
8260            pr "      %s = guestfs_%s " n name;
8261            generate_c_call_args ~handle:"g" style;
8262            pr ";\n";
8263            do_cleanups ();
8264            pr "      if (%s == NULL)\n" n;
8265            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8266            pr "      RETVAL = newSVpv (%s, 0);\n" n;
8267            pr " OUTPUT:\n";
8268            pr "      RETVAL\n"
8269        | RConstOptString n ->
8270            pr "PREINIT:\n";
8271            pr "      const char *%s;\n" n;
8272            pr "   CODE:\n";
8273            pr "      %s = guestfs_%s " n name;
8274            generate_c_call_args ~handle:"g" style;
8275            pr ";\n";
8276            do_cleanups ();
8277            pr "      if (%s == NULL)\n" n;
8278            pr "        RETVAL = &PL_sv_undef;\n";
8279            pr "      else\n";
8280            pr "        RETVAL = newSVpv (%s, 0);\n" n;
8281            pr " OUTPUT:\n";
8282            pr "      RETVAL\n"
8283        | RString n ->
8284            pr "PREINIT:\n";
8285            pr "      char *%s;\n" n;
8286            pr "   CODE:\n";
8287            pr "      %s = guestfs_%s " n name;
8288            generate_c_call_args ~handle:"g" style;
8289            pr ";\n";
8290            do_cleanups ();
8291            pr "      if (%s == NULL)\n" n;
8292            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8293            pr "      RETVAL = newSVpv (%s, 0);\n" n;
8294            pr "      free (%s);\n" n;
8295            pr " OUTPUT:\n";
8296            pr "      RETVAL\n"
8297        | RStringList n | RHashtable n ->
8298            pr "PREINIT:\n";
8299            pr "      char **%s;\n" n;
8300            pr "      int i, n;\n";
8301            pr " PPCODE:\n";
8302            pr "      %s = guestfs_%s " n name;
8303            generate_c_call_args ~handle:"g" style;
8304            pr ";\n";
8305            do_cleanups ();
8306            pr "      if (%s == NULL)\n" n;
8307            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8308            pr "      for (n = 0; %s[n] != NULL; ++n) /**/;\n" n;
8309            pr "      EXTEND (SP, n);\n";
8310            pr "      for (i = 0; i < n; ++i) {\n";
8311            pr "        PUSHs (sv_2mortal (newSVpv (%s[i], 0)));\n" n;
8312            pr "        free (%s[i]);\n" n;
8313            pr "      }\n";
8314            pr "      free (%s);\n" n;
8315        | RStruct (n, typ) ->
8316            let cols = cols_of_struct typ in
8317            generate_perl_struct_code typ cols name style n do_cleanups
8318        | RStructList (n, typ) ->
8319            let cols = cols_of_struct typ in
8320            generate_perl_struct_list_code typ cols name style n do_cleanups
8321        | RBufferOut n ->
8322            pr "PREINIT:\n";
8323            pr "      char *%s;\n" n;
8324            pr "      size_t size;\n";
8325            pr "   CODE:\n";
8326            pr "      %s = guestfs_%s " n name;
8327            generate_c_call_args ~handle:"g" style;
8328            pr ";\n";
8329            do_cleanups ();
8330            pr "      if (%s == NULL)\n" n;
8331            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8332            pr "      RETVAL = newSVpvn (%s, size);\n" n;
8333            pr "      free (%s);\n" n;
8334            pr " OUTPUT:\n";
8335            pr "      RETVAL\n"
8336       );
8337
8338       pr "\n"
8339   ) all_functions
8340
8341 and generate_perl_struct_list_code typ cols name style n do_cleanups =
8342   pr "PREINIT:\n";
8343   pr "      struct guestfs_%s_list *%s;\n" typ n;
8344   pr "      int i;\n";
8345   pr "      HV *hv;\n";
8346   pr " PPCODE:\n";
8347   pr "      %s = guestfs_%s " n name;
8348   generate_c_call_args ~handle:"g" style;
8349   pr ";\n";
8350   do_cleanups ();
8351   pr "      if (%s == NULL)\n" n;
8352   pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8353   pr "      EXTEND (SP, %s->len);\n" n;
8354   pr "      for (i = 0; i < %s->len; ++i) {\n" n;
8355   pr "        hv = newHV ();\n";
8356   List.iter (
8357     function
8358     | name, FString ->
8359         pr "        (void) hv_store (hv, \"%s\", %d, newSVpv (%s->val[i].%s, 0), 0);\n"
8360           name (String.length name) n name
8361     | name, FUUID ->
8362         pr "        (void) hv_store (hv, \"%s\", %d, newSVpv (%s->val[i].%s, 32), 0);\n"
8363           name (String.length name) n name
8364     | name, FBuffer ->
8365         pr "        (void) hv_store (hv, \"%s\", %d, newSVpvn (%s->val[i].%s, %s->val[i].%s_len), 0);\n"
8366           name (String.length name) n name n name
8367     | name, (FBytes|FUInt64) ->
8368         pr "        (void) hv_store (hv, \"%s\", %d, my_newSVull (%s->val[i].%s), 0);\n"
8369           name (String.length name) n name
8370     | name, FInt64 ->
8371         pr "        (void) hv_store (hv, \"%s\", %d, my_newSVll (%s->val[i].%s), 0);\n"
8372           name (String.length name) n name
8373     | name, (FInt32|FUInt32) ->
8374         pr "        (void) hv_store (hv, \"%s\", %d, newSVnv (%s->val[i].%s), 0);\n"
8375           name (String.length name) n name
8376     | name, FChar ->
8377         pr "        (void) hv_store (hv, \"%s\", %d, newSVpv (&%s->val[i].%s, 1), 0);\n"
8378           name (String.length name) n name
8379     | name, FOptPercent ->
8380         pr "        (void) hv_store (hv, \"%s\", %d, newSVnv (%s->val[i].%s), 0);\n"
8381           name (String.length name) n name
8382   ) cols;
8383   pr "        PUSHs (sv_2mortal (newRV ((SV *) hv)));\n";
8384   pr "      }\n";
8385   pr "      guestfs_free_%s_list (%s);\n" typ n
8386
8387 and generate_perl_struct_code typ cols name style n do_cleanups =
8388   pr "PREINIT:\n";
8389   pr "      struct guestfs_%s *%s;\n" typ n;
8390   pr " PPCODE:\n";
8391   pr "      %s = guestfs_%s " n name;
8392   generate_c_call_args ~handle:"g" style;
8393   pr ";\n";
8394   do_cleanups ();
8395   pr "      if (%s == NULL)\n" n;
8396   pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8397   pr "      EXTEND (SP, 2 * %d);\n" (List.length cols);
8398   List.iter (
8399     fun ((name, _) as col) ->
8400       pr "      PUSHs (sv_2mortal (newSVpv (\"%s\", 0)));\n" name;
8401
8402       match col with
8403       | name, FString ->
8404           pr "      PUSHs (sv_2mortal (newSVpv (%s->%s, 0)));\n"
8405             n name
8406       | name, FBuffer ->
8407           pr "      PUSHs (sv_2mortal (newSVpvn (%s->%s, %s->%s_len)));\n"
8408             n name n name
8409       | name, FUUID ->
8410           pr "      PUSHs (sv_2mortal (newSVpv (%s->%s, 32)));\n"
8411             n name
8412       | name, (FBytes|FUInt64) ->
8413           pr "      PUSHs (sv_2mortal (my_newSVull (%s->%s)));\n"
8414             n name
8415       | name, FInt64 ->
8416           pr "      PUSHs (sv_2mortal (my_newSVll (%s->%s)));\n"
8417             n name
8418       | name, (FInt32|FUInt32) ->
8419           pr "      PUSHs (sv_2mortal (newSVnv (%s->%s)));\n"
8420             n name
8421       | name, FChar ->
8422           pr "      PUSHs (sv_2mortal (newSVpv (&%s->%s, 1)));\n"
8423             n name
8424       | name, FOptPercent ->
8425           pr "      PUSHs (sv_2mortal (newSVnv (%s->%s)));\n"
8426             n name
8427   ) cols;
8428   pr "      free (%s);\n" n
8429
8430 (* Generate Sys/Guestfs.pm. *)
8431 and generate_perl_pm () =
8432   generate_header HashStyle LGPLv2plus;
8433
8434   pr "\
8435 =pod
8436
8437 =head1 NAME
8438
8439 Sys::Guestfs - Perl bindings for libguestfs
8440
8441 =head1 SYNOPSIS
8442
8443  use Sys::Guestfs;
8444
8445  my $h = Sys::Guestfs->new ();
8446  $h->add_drive ('guest.img');
8447  $h->launch ();
8448  $h->mount ('/dev/sda1', '/');
8449  $h->touch ('/hello');
8450  $h->sync ();
8451
8452 =head1 DESCRIPTION
8453
8454 The C<Sys::Guestfs> module provides a Perl XS binding to the
8455 libguestfs API for examining and modifying virtual machine
8456 disk images.
8457
8458 Amongst the things this is good for: making batch configuration
8459 changes to guests, getting disk used/free statistics (see also:
8460 virt-df), migrating between virtualization systems (see also:
8461 virt-p2v), performing partial backups, performing partial guest
8462 clones, cloning guests and changing registry/UUID/hostname info, and
8463 much else besides.
8464
8465 Libguestfs uses Linux kernel and qemu code, and can access any type of
8466 guest filesystem that Linux and qemu can, including but not limited
8467 to: ext2/3/4, btrfs, FAT and NTFS, LVM, many different disk partition
8468 schemes, qcow, qcow2, vmdk.
8469
8470 Libguestfs provides ways to enumerate guest storage (eg. partitions,
8471 LVs, what filesystem is in each LV, etc.).  It can also run commands
8472 in the context of the guest.  Also you can access filesystems over
8473 FUSE.
8474
8475 See also L<Sys::Guestfs::Lib(3)> for a set of useful library
8476 functions for using libguestfs from Perl, including integration
8477 with libvirt.
8478
8479 =head1 ERRORS
8480
8481 All errors turn into calls to C<croak> (see L<Carp(3)>).
8482
8483 =head1 METHODS
8484
8485 =over 4
8486
8487 =cut
8488
8489 package Sys::Guestfs;
8490
8491 use strict;
8492 use warnings;
8493
8494 require XSLoader;
8495 XSLoader::load ('Sys::Guestfs');
8496
8497 =item $h = Sys::Guestfs->new ();
8498
8499 Create a new guestfs handle.
8500
8501 =cut
8502
8503 sub new {
8504   my $proto = shift;
8505   my $class = ref ($proto) || $proto;
8506
8507   my $self = Sys::Guestfs::_create ();
8508   bless $self, $class;
8509   return $self;
8510 }
8511
8512 ";
8513
8514   (* Actions.  We only need to print documentation for these as
8515    * they are pulled in from the XS code automatically.
8516    *)
8517   List.iter (
8518     fun (name, style, _, flags, _, _, longdesc) ->
8519       if not (List.mem NotInDocs flags) then (
8520         let longdesc = replace_str longdesc "C<guestfs_" "C<$h-E<gt>" in
8521         pr "=item ";
8522         generate_perl_prototype name style;
8523         pr "\n\n";
8524         pr "%s\n\n" longdesc;
8525         if List.mem ProtocolLimitWarning flags then
8526           pr "%s\n\n" protocol_limit_warning;
8527         if List.mem DangerWillRobinson flags then
8528           pr "%s\n\n" danger_will_robinson;
8529         match deprecation_notice flags with
8530         | None -> ()
8531         | Some txt -> pr "%s\n\n" txt
8532       )
8533   ) all_functions_sorted;
8534
8535   (* End of file. *)
8536   pr "\
8537 =cut
8538
8539 1;
8540
8541 =back
8542
8543 =head1 COPYRIGHT
8544
8545 Copyright (C) %s Red Hat Inc.
8546
8547 =head1 LICENSE
8548
8549 Please see the file COPYING.LIB for the full license.
8550
8551 =head1 SEE ALSO
8552
8553 L<guestfs(3)>,
8554 L<guestfish(1)>,
8555 L<http://libguestfs.org>,
8556 L<Sys::Guestfs::Lib(3)>.
8557
8558 =cut
8559 " copyright_years
8560
8561 and generate_perl_prototype name style =
8562   (match fst style with
8563    | RErr -> ()
8564    | RBool n
8565    | RInt n
8566    | RInt64 n
8567    | RConstString n
8568    | RConstOptString n
8569    | RString n
8570    | RBufferOut n -> pr "$%s = " n
8571    | RStruct (n,_)
8572    | RHashtable n -> pr "%%%s = " n
8573    | RStringList n
8574    | RStructList (n,_) -> pr "@%s = " n
8575   );
8576   pr "$h->%s (" name;
8577   let comma = ref false in
8578   List.iter (
8579     fun arg ->
8580       if !comma then pr ", ";
8581       comma := true;
8582       match arg with
8583       | Pathname n | Device n | Dev_or_Path n | String n
8584       | OptString n | Bool n | Int n | Int64 n | FileIn n | FileOut n ->
8585           pr "$%s" n
8586       | StringList n | DeviceList n ->
8587           pr "\\@%s" n
8588   ) (snd style);
8589   pr ");"
8590
8591 (* Generate Python C module. *)
8592 and generate_python_c () =
8593   generate_header CStyle LGPLv2plus;
8594
8595   pr "\
8596 #include <Python.h>
8597
8598 #include <stdio.h>
8599 #include <stdlib.h>
8600 #include <assert.h>
8601
8602 #include \"guestfs.h\"
8603
8604 typedef struct {
8605   PyObject_HEAD
8606   guestfs_h *g;
8607 } Pyguestfs_Object;
8608
8609 static guestfs_h *
8610 get_handle (PyObject *obj)
8611 {
8612   assert (obj);
8613   assert (obj != Py_None);
8614   return ((Pyguestfs_Object *) obj)->g;
8615 }
8616
8617 static PyObject *
8618 put_handle (guestfs_h *g)
8619 {
8620   assert (g);
8621   return
8622     PyCObject_FromVoidPtrAndDesc ((void *) g, (char *) \"guestfs_h\", NULL);
8623 }
8624
8625 /* This list should be freed (but not the strings) after use. */
8626 static char **
8627 get_string_list (PyObject *obj)
8628 {
8629   int i, len;
8630   char **r;
8631
8632   assert (obj);
8633
8634   if (!PyList_Check (obj)) {
8635     PyErr_SetString (PyExc_RuntimeError, \"expecting a list parameter\");
8636     return NULL;
8637   }
8638
8639   len = PyList_Size (obj);
8640   r = malloc (sizeof (char *) * (len+1));
8641   if (r == NULL) {
8642     PyErr_SetString (PyExc_RuntimeError, \"get_string_list: out of memory\");
8643     return NULL;
8644   }
8645
8646   for (i = 0; i < len; ++i)
8647     r[i] = PyString_AsString (PyList_GetItem (obj, i));
8648   r[len] = NULL;
8649
8650   return r;
8651 }
8652
8653 static PyObject *
8654 put_string_list (char * const * const argv)
8655 {
8656   PyObject *list;
8657   int argc, i;
8658
8659   for (argc = 0; argv[argc] != NULL; ++argc)
8660     ;
8661
8662   list = PyList_New (argc);
8663   for (i = 0; i < argc; ++i)
8664     PyList_SetItem (list, i, PyString_FromString (argv[i]));
8665
8666   return list;
8667 }
8668
8669 static PyObject *
8670 put_table (char * const * const argv)
8671 {
8672   PyObject *list, *item;
8673   int argc, i;
8674
8675   for (argc = 0; argv[argc] != NULL; ++argc)
8676     ;
8677
8678   list = PyList_New (argc >> 1);
8679   for (i = 0; i < argc; i += 2) {
8680     item = PyTuple_New (2);
8681     PyTuple_SetItem (item, 0, PyString_FromString (argv[i]));
8682     PyTuple_SetItem (item, 1, PyString_FromString (argv[i+1]));
8683     PyList_SetItem (list, i >> 1, item);
8684   }
8685
8686   return list;
8687 }
8688
8689 static void
8690 free_strings (char **argv)
8691 {
8692   int argc;
8693
8694   for (argc = 0; argv[argc] != NULL; ++argc)
8695     free (argv[argc]);
8696   free (argv);
8697 }
8698
8699 static PyObject *
8700 py_guestfs_create (PyObject *self, PyObject *args)
8701 {
8702   guestfs_h *g;
8703
8704   g = guestfs_create ();
8705   if (g == NULL) {
8706     PyErr_SetString (PyExc_RuntimeError,
8707                      \"guestfs.create: failed to allocate handle\");
8708     return NULL;
8709   }
8710   guestfs_set_error_handler (g, NULL, NULL);
8711   return put_handle (g);
8712 }
8713
8714 static PyObject *
8715 py_guestfs_close (PyObject *self, PyObject *args)
8716 {
8717   PyObject *py_g;
8718   guestfs_h *g;
8719
8720   if (!PyArg_ParseTuple (args, (char *) \"O:guestfs_close\", &py_g))
8721     return NULL;
8722   g = get_handle (py_g);
8723
8724   guestfs_close (g);
8725
8726   Py_INCREF (Py_None);
8727   return Py_None;
8728 }
8729
8730 ";
8731
8732   let emit_put_list_function typ =
8733     pr "static PyObject *\n";
8734     pr "put_%s_list (struct guestfs_%s_list *%ss)\n" typ typ typ;
8735     pr "{\n";
8736     pr "  PyObject *list;\n";
8737     pr "  int i;\n";
8738     pr "\n";
8739     pr "  list = PyList_New (%ss->len);\n" typ;
8740     pr "  for (i = 0; i < %ss->len; ++i)\n" typ;
8741     pr "    PyList_SetItem (list, i, put_%s (&%ss->val[i]));\n" typ typ;
8742     pr "  return list;\n";
8743     pr "};\n";
8744     pr "\n"
8745   in
8746
8747   (* Structures, turned into Python dictionaries. *)
8748   List.iter (
8749     fun (typ, cols) ->
8750       pr "static PyObject *\n";
8751       pr "put_%s (struct guestfs_%s *%s)\n" typ typ typ;
8752       pr "{\n";
8753       pr "  PyObject *dict;\n";
8754       pr "\n";
8755       pr "  dict = PyDict_New ();\n";
8756       List.iter (
8757         function
8758         | name, FString ->
8759             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8760             pr "                        PyString_FromString (%s->%s));\n"
8761               typ name
8762         | name, FBuffer ->
8763             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8764             pr "                        PyString_FromStringAndSize (%s->%s, %s->%s_len));\n"
8765               typ name typ name
8766         | name, FUUID ->
8767             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8768             pr "                        PyString_FromStringAndSize (%s->%s, 32));\n"
8769               typ name
8770         | name, (FBytes|FUInt64) ->
8771             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8772             pr "                        PyLong_FromUnsignedLongLong (%s->%s));\n"
8773               typ name
8774         | name, FInt64 ->
8775             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8776             pr "                        PyLong_FromLongLong (%s->%s));\n"
8777               typ name
8778         | name, FUInt32 ->
8779             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8780             pr "                        PyLong_FromUnsignedLong (%s->%s));\n"
8781               typ name
8782         | name, FInt32 ->
8783             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8784             pr "                        PyLong_FromLong (%s->%s));\n"
8785               typ name
8786         | name, FOptPercent ->
8787             pr "  if (%s->%s >= 0)\n" typ name;
8788             pr "    PyDict_SetItemString (dict, \"%s\",\n" name;
8789             pr "                          PyFloat_FromDouble ((double) %s->%s));\n"
8790               typ name;
8791             pr "  else {\n";
8792             pr "    Py_INCREF (Py_None);\n";
8793             pr "    PyDict_SetItemString (dict, \"%s\", Py_None);\n" name;
8794             pr "  }\n"
8795         | name, FChar ->
8796             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8797             pr "                        PyString_FromStringAndSize (&dirent->%s, 1));\n" name
8798       ) cols;
8799       pr "  return dict;\n";
8800       pr "};\n";
8801       pr "\n";
8802
8803   ) structs;
8804
8805   (* Emit a put_TYPE_list function definition only if that function is used. *)
8806   List.iter (
8807     function
8808     | typ, (RStructListOnly | RStructAndList) ->
8809         (* generate the function for typ *)
8810         emit_put_list_function typ
8811     | typ, _ -> () (* empty *)
8812   ) (rstructs_used_by all_functions);
8813
8814   (* Python wrapper functions. *)
8815   List.iter (
8816     fun (name, style, _, _, _, _, _) ->
8817       pr "static PyObject *\n";
8818       pr "py_guestfs_%s (PyObject *self, PyObject *args)\n" name;
8819       pr "{\n";
8820
8821       pr "  PyObject *py_g;\n";
8822       pr "  guestfs_h *g;\n";
8823       pr "  PyObject *py_r;\n";
8824
8825       let error_code =
8826         match fst style with
8827         | RErr | RInt _ | RBool _ -> pr "  int r;\n"; "-1"
8828         | RInt64 _ -> pr "  int64_t r;\n"; "-1"
8829         | RConstString _ | RConstOptString _ ->
8830             pr "  const char *r;\n"; "NULL"
8831         | RString _ -> pr "  char *r;\n"; "NULL"
8832         | RStringList _ | RHashtable _ -> pr "  char **r;\n"; "NULL"
8833         | RStruct (_, typ) -> pr "  struct guestfs_%s *r;\n" typ; "NULL"
8834         | RStructList (_, typ) ->
8835             pr "  struct guestfs_%s_list *r;\n" typ; "NULL"
8836         | RBufferOut _ ->
8837             pr "  char *r;\n";
8838             pr "  size_t size;\n";
8839             "NULL" in
8840
8841       List.iter (
8842         function
8843         | Pathname n | Device n | Dev_or_Path n | String n | FileIn n | FileOut n ->
8844             pr "  const char *%s;\n" n
8845         | OptString n -> pr "  const char *%s;\n" n
8846         | StringList n | DeviceList n ->
8847             pr "  PyObject *py_%s;\n" n;
8848             pr "  char **%s;\n" n
8849         | Bool n -> pr "  int %s;\n" n
8850         | Int n -> pr "  int %s;\n" n
8851         | Int64 n -> pr "  long long %s;\n" n
8852       ) (snd style);
8853
8854       pr "\n";
8855
8856       (* Convert the parameters. *)
8857       pr "  if (!PyArg_ParseTuple (args, (char *) \"O";
8858       List.iter (
8859         function
8860         | Pathname _ | Device _ | Dev_or_Path _ | String _ | FileIn _ | FileOut _ -> pr "s"
8861         | OptString _ -> pr "z"
8862         | StringList _ | DeviceList _ -> pr "O"
8863         | Bool _ -> pr "i" (* XXX Python has booleans? *)
8864         | Int _ -> pr "i"
8865         | Int64 _ -> pr "L" (* XXX Whoever thought it was a good idea to
8866                              * emulate C's int/long/long long in Python?
8867                              *)
8868       ) (snd style);
8869       pr ":guestfs_%s\",\n" name;
8870       pr "                         &py_g";
8871       List.iter (
8872         function
8873         | Pathname n | Device n | Dev_or_Path n | String n | FileIn n | FileOut n -> pr ", &%s" n
8874         | OptString n -> pr ", &%s" n
8875         | StringList n | DeviceList n -> pr ", &py_%s" n
8876         | Bool n -> pr ", &%s" n
8877         | Int n -> pr ", &%s" n
8878         | Int64 n -> pr ", &%s" n
8879       ) (snd style);
8880
8881       pr "))\n";
8882       pr "    return NULL;\n";
8883
8884       pr "  g = get_handle (py_g);\n";
8885       List.iter (
8886         function
8887         | Pathname _ | Device _ | Dev_or_Path _ | String _
8888         | FileIn _ | FileOut _ | OptString _ | Bool _ | Int _ | Int64 _ -> ()
8889         | StringList n | DeviceList n ->
8890             pr "  %s = get_string_list (py_%s);\n" n n;
8891             pr "  if (!%s) return NULL;\n" n
8892       ) (snd style);
8893
8894       pr "\n";
8895
8896       pr "  r = guestfs_%s " name;
8897       generate_c_call_args ~handle:"g" style;
8898       pr ";\n";
8899
8900       List.iter (
8901         function
8902         | Pathname _ | Device _ | Dev_or_Path _ | String _
8903         | FileIn _ | FileOut _ | OptString _ | Bool _ | Int _ | Int64 _ -> ()
8904         | StringList n | DeviceList n ->
8905             pr "  free (%s);\n" n
8906       ) (snd style);
8907
8908       pr "  if (r == %s) {\n" error_code;
8909       pr "    PyErr_SetString (PyExc_RuntimeError, guestfs_last_error (g));\n";
8910       pr "    return NULL;\n";
8911       pr "  }\n";
8912       pr "\n";
8913
8914       (match fst style with
8915        | RErr ->
8916            pr "  Py_INCREF (Py_None);\n";
8917            pr "  py_r = Py_None;\n"
8918        | RInt _
8919        | RBool _ -> pr "  py_r = PyInt_FromLong ((long) r);\n"
8920        | RInt64 _ -> pr "  py_r = PyLong_FromLongLong (r);\n"
8921        | RConstString _ -> pr "  py_r = PyString_FromString (r);\n"
8922        | RConstOptString _ ->
8923            pr "  if (r)\n";
8924            pr "    py_r = PyString_FromString (r);\n";
8925            pr "  else {\n";
8926            pr "    Py_INCREF (Py_None);\n";
8927            pr "    py_r = Py_None;\n";
8928            pr "  }\n"
8929        | RString _ ->
8930            pr "  py_r = PyString_FromString (r);\n";
8931            pr "  free (r);\n"
8932        | RStringList _ ->
8933            pr "  py_r = put_string_list (r);\n";
8934            pr "  free_strings (r);\n"
8935        | RStruct (_, typ) ->
8936            pr "  py_r = put_%s (r);\n" typ;
8937            pr "  guestfs_free_%s (r);\n" typ
8938        | RStructList (_, typ) ->
8939            pr "  py_r = put_%s_list (r);\n" typ;
8940            pr "  guestfs_free_%s_list (r);\n" typ
8941        | RHashtable n ->
8942            pr "  py_r = put_table (r);\n";
8943            pr "  free_strings (r);\n"
8944        | RBufferOut _ ->
8945            pr "  py_r = PyString_FromStringAndSize (r, size);\n";
8946            pr "  free (r);\n"
8947       );
8948
8949       pr "  return py_r;\n";
8950       pr "}\n";
8951       pr "\n"
8952   ) all_functions;
8953
8954   (* Table of functions. *)
8955   pr "static PyMethodDef methods[] = {\n";
8956   pr "  { (char *) \"create\", py_guestfs_create, METH_VARARGS, NULL },\n";
8957   pr "  { (char *) \"close\", py_guestfs_close, METH_VARARGS, NULL },\n";
8958   List.iter (
8959     fun (name, _, _, _, _, _, _) ->
8960       pr "  { (char *) \"%s\", py_guestfs_%s, METH_VARARGS, NULL },\n"
8961         name name
8962   ) all_functions;
8963   pr "  { NULL, NULL, 0, NULL }\n";
8964   pr "};\n";
8965   pr "\n";
8966
8967   (* Init function. *)
8968   pr "\
8969 void
8970 initlibguestfsmod (void)
8971 {
8972   static int initialized = 0;
8973
8974   if (initialized) return;
8975   Py_InitModule ((char *) \"libguestfsmod\", methods);
8976   initialized = 1;
8977 }
8978 "
8979
8980 (* Generate Python module. *)
8981 and generate_python_py () =
8982   generate_header HashStyle LGPLv2plus;
8983
8984   pr "\
8985 u\"\"\"Python bindings for libguestfs
8986
8987 import guestfs
8988 g = guestfs.GuestFS ()
8989 g.add_drive (\"guest.img\")
8990 g.launch ()
8991 parts = g.list_partitions ()
8992
8993 The guestfs module provides a Python binding to the libguestfs API
8994 for examining and modifying virtual machine disk images.
8995
8996 Amongst the things this is good for: making batch configuration
8997 changes to guests, getting disk used/free statistics (see also:
8998 virt-df), migrating between virtualization systems (see also:
8999 virt-p2v), performing partial backups, performing partial guest
9000 clones, cloning guests and changing registry/UUID/hostname info, and
9001 much else besides.
9002
9003 Libguestfs uses Linux kernel and qemu code, and can access any type of
9004 guest filesystem that Linux and qemu can, including but not limited
9005 to: ext2/3/4, btrfs, FAT and NTFS, LVM, many different disk partition
9006 schemes, qcow, qcow2, vmdk.
9007
9008 Libguestfs provides ways to enumerate guest storage (eg. partitions,
9009 LVs, what filesystem is in each LV, etc.).  It can also run commands
9010 in the context of the guest.  Also you can access filesystems over
9011 FUSE.
9012
9013 Errors which happen while using the API are turned into Python
9014 RuntimeError exceptions.
9015
9016 To create a guestfs handle you usually have to perform the following
9017 sequence of calls:
9018
9019 # Create the handle, call add_drive at least once, and possibly
9020 # several times if the guest has multiple block devices:
9021 g = guestfs.GuestFS ()
9022 g.add_drive (\"guest.img\")
9023
9024 # Launch the qemu subprocess and wait for it to become ready:
9025 g.launch ()
9026
9027 # Now you can issue commands, for example:
9028 logvols = g.lvs ()
9029
9030 \"\"\"
9031
9032 import libguestfsmod
9033
9034 class GuestFS:
9035     \"\"\"Instances of this class are libguestfs API handles.\"\"\"
9036
9037     def __init__ (self):
9038         \"\"\"Create a new libguestfs handle.\"\"\"
9039         self._o = libguestfsmod.create ()
9040
9041     def __del__ (self):
9042         libguestfsmod.close (self._o)
9043
9044 ";
9045
9046   List.iter (
9047     fun (name, style, _, flags, _, _, longdesc) ->
9048       pr "    def %s " name;
9049       generate_py_call_args ~handle:"self" (snd style);
9050       pr ":\n";
9051
9052       if not (List.mem NotInDocs flags) then (
9053         let doc = replace_str longdesc "C<guestfs_" "C<g." in
9054         let doc =
9055           match fst style with
9056           | RErr | RInt _ | RInt64 _ | RBool _
9057           | RConstOptString _ | RConstString _
9058           | RString _ | RBufferOut _ -> doc
9059           | RStringList _ ->
9060               doc ^ "\n\nThis function returns a list of strings."
9061           | RStruct (_, typ) ->
9062               doc ^ sprintf "\n\nThis function returns a dictionary, with keys matching the various fields in the guestfs_%s structure." typ
9063           | RStructList (_, typ) ->
9064               doc ^ sprintf "\n\nThis function returns a list of %ss.  Each %s is represented as a dictionary." typ typ
9065           | RHashtable _ ->
9066               doc ^ "\n\nThis function returns a dictionary." in
9067         let doc =
9068           if List.mem ProtocolLimitWarning flags then
9069             doc ^ "\n\n" ^ protocol_limit_warning
9070           else doc in
9071         let doc =
9072           if List.mem DangerWillRobinson flags then
9073             doc ^ "\n\n" ^ danger_will_robinson
9074           else doc in
9075         let doc =
9076           match deprecation_notice flags with
9077           | None -> doc
9078           | Some txt -> doc ^ "\n\n" ^ txt in
9079         let doc = pod2text ~width:60 name doc in
9080         let doc = List.map (fun line -> replace_str line "\\" "\\\\") doc in
9081         let doc = String.concat "\n        " doc in
9082         pr "        u\"\"\"%s\"\"\"\n" doc;
9083       );
9084       pr "        return libguestfsmod.%s " name;
9085       generate_py_call_args ~handle:"self._o" (snd style);
9086       pr "\n";
9087       pr "\n";
9088   ) all_functions
9089
9090 (* Generate Python call arguments, eg "(handle, foo, bar)" *)
9091 and generate_py_call_args ~handle args =
9092   pr "(%s" handle;
9093   List.iter (fun arg -> pr ", %s" (name_of_argt arg)) args;
9094   pr ")"
9095
9096 (* Useful if you need the longdesc POD text as plain text.  Returns a
9097  * list of lines.
9098  *
9099  * Because this is very slow (the slowest part of autogeneration),
9100  * we memoize the results.
9101  *)
9102 and pod2text ~width name longdesc =
9103   let key = width, name, longdesc in
9104   try Hashtbl.find pod2text_memo key
9105   with Not_found ->
9106     let filename, chan = Filename.open_temp_file "gen" ".tmp" in
9107     fprintf chan "=head1 %s\n\n%s\n" name longdesc;
9108     close_out chan;
9109     let cmd = sprintf "pod2text -w %d %s" width (Filename.quote filename) in
9110     let chan = open_process_in cmd in
9111     let lines = ref [] in
9112     let rec loop i =
9113       let line = input_line chan in
9114       if i = 1 then             (* discard the first line of output *)
9115         loop (i+1)
9116       else (
9117         let line = triml line in
9118         lines := line :: !lines;
9119         loop (i+1)
9120       ) in
9121     let lines = try loop 1 with End_of_file -> List.rev !lines in
9122     unlink filename;
9123     (match close_process_in chan with
9124      | WEXITED 0 -> ()
9125      | WEXITED i ->
9126          failwithf "pod2text: process exited with non-zero status (%d)" i
9127      | WSIGNALED i | WSTOPPED i ->
9128          failwithf "pod2text: process signalled or stopped by signal %d" i
9129     );
9130     Hashtbl.add pod2text_memo key lines;
9131     pod2text_memo_updated ();
9132     lines
9133
9134 (* Generate ruby bindings. *)
9135 and generate_ruby_c () =
9136   generate_header CStyle LGPLv2plus;
9137
9138   pr "\
9139 #include <stdio.h>
9140 #include <stdlib.h>
9141
9142 #include <ruby.h>
9143
9144 #include \"guestfs.h\"
9145
9146 #include \"extconf.h\"
9147
9148 /* For Ruby < 1.9 */
9149 #ifndef RARRAY_LEN
9150 #define RARRAY_LEN(r) (RARRAY((r))->len)
9151 #endif
9152
9153 static VALUE m_guestfs;                 /* guestfs module */
9154 static VALUE c_guestfs;                 /* guestfs_h handle */
9155 static VALUE e_Error;                   /* used for all errors */
9156
9157 static void ruby_guestfs_free (void *p)
9158 {
9159   if (!p) return;
9160   guestfs_close ((guestfs_h *) p);
9161 }
9162
9163 static VALUE ruby_guestfs_create (VALUE m)
9164 {
9165   guestfs_h *g;
9166
9167   g = guestfs_create ();
9168   if (!g)
9169     rb_raise (e_Error, \"failed to create guestfs handle\");
9170
9171   /* Don't print error messages to stderr by default. */
9172   guestfs_set_error_handler (g, NULL, NULL);
9173
9174   /* Wrap it, and make sure the close function is called when the
9175    * handle goes away.
9176    */
9177   return Data_Wrap_Struct (c_guestfs, NULL, ruby_guestfs_free, g);
9178 }
9179
9180 static VALUE ruby_guestfs_close (VALUE gv)
9181 {
9182   guestfs_h *g;
9183   Data_Get_Struct (gv, guestfs_h, g);
9184
9185   ruby_guestfs_free (g);
9186   DATA_PTR (gv) = NULL;
9187
9188   return Qnil;
9189 }
9190
9191 ";
9192
9193   List.iter (
9194     fun (name, style, _, _, _, _, _) ->
9195       pr "static VALUE ruby_guestfs_%s (VALUE gv" name;
9196       List.iter (fun arg -> pr ", VALUE %sv" (name_of_argt arg)) (snd style);
9197       pr ")\n";
9198       pr "{\n";
9199       pr "  guestfs_h *g;\n";
9200       pr "  Data_Get_Struct (gv, guestfs_h, g);\n";
9201       pr "  if (!g)\n";
9202       pr "    rb_raise (rb_eArgError, \"%%s: used handle after closing it\", \"%s\");\n"
9203         name;
9204       pr "\n";
9205
9206       List.iter (
9207         function
9208         | Pathname n | Device n | Dev_or_Path n | String n | FileIn n | FileOut n ->
9209             pr "  Check_Type (%sv, T_STRING);\n" n;
9210             pr "  const char *%s = StringValueCStr (%sv);\n" n n;
9211             pr "  if (!%s)\n" n;
9212             pr "    rb_raise (rb_eTypeError, \"expected string for parameter %%s of %%s\",\n";
9213             pr "              \"%s\", \"%s\");\n" n name
9214         | OptString n ->
9215             pr "  const char *%s = !NIL_P (%sv) ? StringValueCStr (%sv) : NULL;\n" n n n
9216         | StringList n | DeviceList n ->
9217             pr "  char **%s;\n" n;
9218             pr "  Check_Type (%sv, T_ARRAY);\n" n;
9219             pr "  {\n";
9220             pr "    int i, len;\n";
9221             pr "    len = RARRAY_LEN (%sv);\n" n;
9222             pr "    %s = guestfs_safe_malloc (g, sizeof (char *) * (len+1));\n"
9223               n;
9224             pr "    for (i = 0; i < len; ++i) {\n";
9225             pr "      VALUE v = rb_ary_entry (%sv, i);\n" n;
9226             pr "      %s[i] = StringValueCStr (v);\n" n;
9227             pr "    }\n";
9228             pr "    %s[len] = NULL;\n" n;
9229             pr "  }\n";
9230         | Bool n ->
9231             pr "  int %s = RTEST (%sv);\n" n n
9232         | Int n ->
9233             pr "  int %s = NUM2INT (%sv);\n" n n
9234         | Int64 n ->
9235             pr "  long long %s = NUM2LL (%sv);\n" n n
9236       ) (snd style);
9237       pr "\n";
9238
9239       let error_code =
9240         match fst style with
9241         | RErr | RInt _ | RBool _ -> pr "  int r;\n"; "-1"
9242         | RInt64 _ -> pr "  int64_t r;\n"; "-1"
9243         | RConstString _ | RConstOptString _ ->
9244             pr "  const char *r;\n"; "NULL"
9245         | RString _ -> pr "  char *r;\n"; "NULL"
9246         | RStringList _ | RHashtable _ -> pr "  char **r;\n"; "NULL"
9247         | RStruct (_, typ) -> pr "  struct guestfs_%s *r;\n" typ; "NULL"
9248         | RStructList (_, typ) ->
9249             pr "  struct guestfs_%s_list *r;\n" typ; "NULL"
9250         | RBufferOut _ ->
9251             pr "  char *r;\n";
9252             pr "  size_t size;\n";
9253             "NULL" in
9254       pr "\n";
9255
9256       pr "  r = guestfs_%s " name;
9257       generate_c_call_args ~handle:"g" style;
9258       pr ";\n";
9259
9260       List.iter (
9261         function
9262         | Pathname _ | Device _ | Dev_or_Path _ | String _
9263         | FileIn _ | FileOut _ | OptString _ | Bool _ | Int _ | Int64 _ -> ()
9264         | StringList n | DeviceList n ->
9265             pr "  free (%s);\n" n
9266       ) (snd style);
9267
9268       pr "  if (r == %s)\n" error_code;
9269       pr "    rb_raise (e_Error, \"%%s\", guestfs_last_error (g));\n";
9270       pr "\n";
9271
9272       (match fst style with
9273        | RErr ->
9274            pr "  return Qnil;\n"
9275        | RInt _ | RBool _ ->
9276            pr "  return INT2NUM (r);\n"
9277        | RInt64 _ ->
9278            pr "  return ULL2NUM (r);\n"
9279        | RConstString _ ->
9280            pr "  return rb_str_new2 (r);\n";
9281        | RConstOptString _ ->
9282            pr "  if (r)\n";
9283            pr "    return rb_str_new2 (r);\n";
9284            pr "  else\n";
9285            pr "    return Qnil;\n";
9286        | RString _ ->
9287            pr "  VALUE rv = rb_str_new2 (r);\n";
9288            pr "  free (r);\n";
9289            pr "  return rv;\n";
9290        | RStringList _ ->
9291            pr "  int i, len = 0;\n";
9292            pr "  for (i = 0; r[i] != NULL; ++i) len++;\n";
9293            pr "  VALUE rv = rb_ary_new2 (len);\n";
9294            pr "  for (i = 0; r[i] != NULL; ++i) {\n";
9295            pr "    rb_ary_push (rv, rb_str_new2 (r[i]));\n";
9296            pr "    free (r[i]);\n";
9297            pr "  }\n";
9298            pr "  free (r);\n";
9299            pr "  return rv;\n"
9300        | RStruct (_, typ) ->
9301            let cols = cols_of_struct typ in
9302            generate_ruby_struct_code typ cols
9303        | RStructList (_, typ) ->
9304            let cols = cols_of_struct typ in
9305            generate_ruby_struct_list_code typ cols
9306        | RHashtable _ ->
9307            pr "  VALUE rv = rb_hash_new ();\n";
9308            pr "  int i;\n";
9309            pr "  for (i = 0; r[i] != NULL; i+=2) {\n";
9310            pr "    rb_hash_aset (rv, rb_str_new2 (r[i]), rb_str_new2 (r[i+1]));\n";
9311            pr "    free (r[i]);\n";
9312            pr "    free (r[i+1]);\n";
9313            pr "  }\n";
9314            pr "  free (r);\n";
9315            pr "  return rv;\n"
9316        | RBufferOut _ ->
9317            pr "  VALUE rv = rb_str_new (r, size);\n";
9318            pr "  free (r);\n";
9319            pr "  return rv;\n";
9320       );
9321
9322       pr "}\n";
9323       pr "\n"
9324   ) all_functions;
9325
9326   pr "\
9327 /* Initialize the module. */
9328 void Init__guestfs ()
9329 {
9330   m_guestfs = rb_define_module (\"Guestfs\");
9331   c_guestfs = rb_define_class_under (m_guestfs, \"Guestfs\", rb_cObject);
9332   e_Error = rb_define_class_under (m_guestfs, \"Error\", rb_eStandardError);
9333
9334   rb_define_module_function (m_guestfs, \"create\", ruby_guestfs_create, 0);
9335   rb_define_method (c_guestfs, \"close\", ruby_guestfs_close, 0);
9336
9337 ";
9338   (* Define the rest of the methods. *)
9339   List.iter (
9340     fun (name, style, _, _, _, _, _) ->
9341       pr "  rb_define_method (c_guestfs, \"%s\",\n" name;
9342       pr "        ruby_guestfs_%s, %d);\n" name (List.length (snd style))
9343   ) all_functions;
9344
9345   pr "}\n"
9346
9347 (* Ruby code to return a struct. *)
9348 and generate_ruby_struct_code typ cols =
9349   pr "  VALUE rv = rb_hash_new ();\n";
9350   List.iter (
9351     function
9352     | name, FString ->
9353         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), rb_str_new2 (r->%s));\n" name name
9354     | name, FBuffer ->
9355         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), rb_str_new (r->%s, r->%s_len));\n" name name name
9356     | name, FUUID ->
9357         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), rb_str_new (r->%s, 32));\n" name name
9358     | name, (FBytes|FUInt64) ->
9359         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), ULL2NUM (r->%s));\n" name name
9360     | name, FInt64 ->
9361         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), LL2NUM (r->%s));\n" name name
9362     | name, FUInt32 ->
9363         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), UINT2NUM (r->%s));\n" name name
9364     | name, FInt32 ->
9365         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), INT2NUM (r->%s));\n" name name
9366     | name, FOptPercent ->
9367         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), rb_dbl2big (r->%s));\n" name name
9368     | name, FChar -> (* XXX wrong? *)
9369         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), ULL2NUM (r->%s));\n" name name
9370   ) cols;
9371   pr "  guestfs_free_%s (r);\n" typ;
9372   pr "  return rv;\n"
9373
9374 (* Ruby code to return a struct list. *)
9375 and generate_ruby_struct_list_code typ cols =
9376   pr "  VALUE rv = rb_ary_new2 (r->len);\n";
9377   pr "  int i;\n";
9378   pr "  for (i = 0; i < r->len; ++i) {\n";
9379   pr "    VALUE hv = rb_hash_new ();\n";
9380   List.iter (
9381     function
9382     | name, FString ->
9383         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), rb_str_new2 (r->val[i].%s));\n" name name
9384     | name, FBuffer ->
9385         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), rb_str_new (r->val[i].%s, r->val[i].%s_len));\n" name name name
9386     | name, FUUID ->
9387         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), rb_str_new (r->val[i].%s, 32));\n" name name
9388     | name, (FBytes|FUInt64) ->
9389         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), ULL2NUM (r->val[i].%s));\n" name name
9390     | name, FInt64 ->
9391         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), LL2NUM (r->val[i].%s));\n" name name
9392     | name, FUInt32 ->
9393         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), UINT2NUM (r->val[i].%s));\n" name name
9394     | name, FInt32 ->
9395         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), INT2NUM (r->val[i].%s));\n" name name
9396     | name, FOptPercent ->
9397         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), rb_dbl2big (r->val[i].%s));\n" name name
9398     | name, FChar -> (* XXX wrong? *)
9399         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), ULL2NUM (r->val[i].%s));\n" name name
9400   ) cols;
9401   pr "    rb_ary_push (rv, hv);\n";
9402   pr "  }\n";
9403   pr "  guestfs_free_%s_list (r);\n" typ;
9404   pr "  return rv;\n"
9405
9406 (* Generate Java bindings GuestFS.java file. *)
9407 and generate_java_java () =
9408   generate_header CStyle LGPLv2plus;
9409
9410   pr "\
9411 package com.redhat.et.libguestfs;
9412
9413 import java.util.HashMap;
9414 import com.redhat.et.libguestfs.LibGuestFSException;
9415 import com.redhat.et.libguestfs.PV;
9416 import com.redhat.et.libguestfs.VG;
9417 import com.redhat.et.libguestfs.LV;
9418 import com.redhat.et.libguestfs.Stat;
9419 import com.redhat.et.libguestfs.StatVFS;
9420 import com.redhat.et.libguestfs.IntBool;
9421 import com.redhat.et.libguestfs.Dirent;
9422
9423 /**
9424  * The GuestFS object is a libguestfs handle.
9425  *
9426  * @author rjones
9427  */
9428 public class GuestFS {
9429   // Load the native code.
9430   static {
9431     System.loadLibrary (\"guestfs_jni\");
9432   }
9433
9434   /**
9435    * The native guestfs_h pointer.
9436    */
9437   long g;
9438
9439   /**
9440    * Create a libguestfs handle.
9441    *
9442    * @throws LibGuestFSException
9443    */
9444   public GuestFS () throws LibGuestFSException
9445   {
9446     g = _create ();
9447   }
9448   private native long _create () throws LibGuestFSException;
9449
9450   /**
9451    * Close a libguestfs handle.
9452    *
9453    * You can also leave handles to be collected by the garbage
9454    * collector, but this method ensures that the resources used
9455    * by the handle are freed up immediately.  If you call any
9456    * other methods after closing the handle, you will get an
9457    * exception.
9458    *
9459    * @throws LibGuestFSException
9460    */
9461   public void close () throws LibGuestFSException
9462   {
9463     if (g != 0)
9464       _close (g);
9465     g = 0;
9466   }
9467   private native void _close (long g) throws LibGuestFSException;
9468
9469   public void finalize () throws LibGuestFSException
9470   {
9471     close ();
9472   }
9473
9474 ";
9475
9476   List.iter (
9477     fun (name, style, _, flags, _, shortdesc, longdesc) ->
9478       if not (List.mem NotInDocs flags); then (
9479         let doc = replace_str longdesc "C<guestfs_" "C<g." in
9480         let doc =
9481           if List.mem ProtocolLimitWarning flags then
9482             doc ^ "\n\n" ^ protocol_limit_warning
9483           else doc in
9484         let doc =
9485           if List.mem DangerWillRobinson flags then
9486             doc ^ "\n\n" ^ danger_will_robinson
9487           else doc in
9488         let doc =
9489           match deprecation_notice flags with
9490           | None -> doc
9491           | Some txt -> doc ^ "\n\n" ^ txt in
9492         let doc = pod2text ~width:60 name doc in
9493         let doc = List.map (            (* RHBZ#501883 *)
9494           function
9495           | "" -> "<p>"
9496           | nonempty -> nonempty
9497         ) doc in
9498         let doc = String.concat "\n   * " doc in
9499
9500         pr "  /**\n";
9501         pr "   * %s\n" shortdesc;
9502         pr "   * <p>\n";
9503         pr "   * %s\n" doc;
9504         pr "   * @throws LibGuestFSException\n";
9505         pr "   */\n";
9506         pr "  ";
9507       );
9508       generate_java_prototype ~public:true ~semicolon:false name style;
9509       pr "\n";
9510       pr "  {\n";
9511       pr "    if (g == 0)\n";
9512       pr "      throw new LibGuestFSException (\"%s: handle is closed\");\n"
9513         name;
9514       pr "    ";
9515       if fst style <> RErr then pr "return ";
9516       pr "_%s " name;
9517       generate_java_call_args ~handle:"g" (snd style);
9518       pr ";\n";
9519       pr "  }\n";
9520       pr "  ";
9521       generate_java_prototype ~privat:true ~native:true name style;
9522       pr "\n";
9523       pr "\n";
9524   ) all_functions;
9525
9526   pr "}\n"
9527
9528 (* Generate Java call arguments, eg "(handle, foo, bar)" *)
9529 and generate_java_call_args ~handle args =
9530   pr "(%s" handle;
9531   List.iter (fun arg -> pr ", %s" (name_of_argt arg)) args;
9532   pr ")"
9533
9534 and generate_java_prototype ?(public=false) ?(privat=false) ?(native=false)
9535     ?(semicolon=true) name style =
9536   if privat then pr "private ";
9537   if public then pr "public ";
9538   if native then pr "native ";
9539
9540   (* return type *)
9541   (match fst style with
9542    | RErr -> pr "void ";
9543    | RInt _ -> pr "int ";
9544    | RInt64 _ -> pr "long ";
9545    | RBool _ -> pr "boolean ";
9546    | RConstString _ | RConstOptString _ | RString _
9547    | RBufferOut _ -> pr "String ";
9548    | RStringList _ -> pr "String[] ";
9549    | RStruct (_, typ) ->
9550        let name = java_name_of_struct typ in
9551        pr "%s " name;
9552    | RStructList (_, typ) ->
9553        let name = java_name_of_struct typ in
9554        pr "%s[] " name;
9555    | RHashtable _ -> pr "HashMap<String,String> ";
9556   );
9557
9558   if native then pr "_%s " name else pr "%s " name;
9559   pr "(";
9560   let needs_comma = ref false in
9561   if native then (
9562     pr "long g";
9563     needs_comma := true
9564   );
9565
9566   (* args *)
9567   List.iter (
9568     fun arg ->
9569       if !needs_comma then pr ", ";
9570       needs_comma := true;
9571
9572       match arg with
9573       | Pathname n
9574       | Device n | Dev_or_Path n
9575       | String n
9576       | OptString n
9577       | FileIn n
9578       | FileOut n ->
9579           pr "String %s" n
9580       | StringList n | DeviceList n ->
9581           pr "String[] %s" n
9582       | Bool n ->
9583           pr "boolean %s" n
9584       | Int n ->
9585           pr "int %s" n
9586       | Int64 n ->
9587           pr "long %s" n
9588   ) (snd style);
9589
9590   pr ")\n";
9591   pr "    throws LibGuestFSException";
9592   if semicolon then pr ";"
9593
9594 and generate_java_struct jtyp cols () =
9595   generate_header CStyle LGPLv2plus;
9596
9597   pr "\
9598 package com.redhat.et.libguestfs;
9599
9600 /**
9601  * Libguestfs %s structure.
9602  *
9603  * @author rjones
9604  * @see GuestFS
9605  */
9606 public class %s {
9607 " jtyp jtyp;
9608
9609   List.iter (
9610     function
9611     | name, FString
9612     | name, FUUID
9613     | name, FBuffer -> pr "  public String %s;\n" name
9614     | name, (FBytes|FUInt64|FInt64) -> pr "  public long %s;\n" name
9615     | name, (FUInt32|FInt32) -> pr "  public int %s;\n" name
9616     | name, FChar -> pr "  public char %s;\n" name
9617     | name, FOptPercent ->
9618         pr "  /* The next field is [0..100] or -1 meaning 'not present': */\n";
9619         pr "  public float %s;\n" name
9620   ) cols;
9621
9622   pr "}\n"
9623
9624 and generate_java_c () =
9625   generate_header CStyle LGPLv2plus;
9626
9627   pr "\
9628 #include <stdio.h>
9629 #include <stdlib.h>
9630 #include <string.h>
9631
9632 #include \"com_redhat_et_libguestfs_GuestFS.h\"
9633 #include \"guestfs.h\"
9634
9635 /* Note that this function returns.  The exception is not thrown
9636  * until after the wrapper function returns.
9637  */
9638 static void
9639 throw_exception (JNIEnv *env, const char *msg)
9640 {
9641   jclass cl;
9642   cl = (*env)->FindClass (env,
9643                           \"com/redhat/et/libguestfs/LibGuestFSException\");
9644   (*env)->ThrowNew (env, cl, msg);
9645 }
9646
9647 JNIEXPORT jlong JNICALL
9648 Java_com_redhat_et_libguestfs_GuestFS__1create
9649   (JNIEnv *env, jobject obj)
9650 {
9651   guestfs_h *g;
9652
9653   g = guestfs_create ();
9654   if (g == NULL) {
9655     throw_exception (env, \"GuestFS.create: failed to allocate handle\");
9656     return 0;
9657   }
9658   guestfs_set_error_handler (g, NULL, NULL);
9659   return (jlong) (long) g;
9660 }
9661
9662 JNIEXPORT void JNICALL
9663 Java_com_redhat_et_libguestfs_GuestFS__1close
9664   (JNIEnv *env, jobject obj, jlong jg)
9665 {
9666   guestfs_h *g = (guestfs_h *) (long) jg;
9667   guestfs_close (g);
9668 }
9669
9670 ";
9671
9672   List.iter (
9673     fun (name, style, _, _, _, _, _) ->
9674       pr "JNIEXPORT ";
9675       (match fst style with
9676        | RErr -> pr "void ";
9677        | RInt _ -> pr "jint ";
9678        | RInt64 _ -> pr "jlong ";
9679        | RBool _ -> pr "jboolean ";
9680        | RConstString _ | RConstOptString _ | RString _
9681        | RBufferOut _ -> pr "jstring ";
9682        | RStruct _ | RHashtable _ ->
9683            pr "jobject ";
9684        | RStringList _ | RStructList _ ->
9685            pr "jobjectArray ";
9686       );
9687       pr "JNICALL\n";
9688       pr "Java_com_redhat_et_libguestfs_GuestFS_";
9689       pr "%s" (replace_str ("_" ^ name) "_" "_1");
9690       pr "\n";
9691       pr "  (JNIEnv *env, jobject obj, jlong jg";
9692       List.iter (
9693         function
9694         | Pathname n
9695         | Device n | Dev_or_Path n
9696         | String n
9697         | OptString n
9698         | FileIn n
9699         | FileOut n ->
9700             pr ", jstring j%s" n
9701         | StringList n | DeviceList n ->
9702             pr ", jobjectArray j%s" n
9703         | Bool n ->
9704             pr ", jboolean j%s" n
9705         | Int n ->
9706             pr ", jint j%s" n
9707         | Int64 n ->
9708             pr ", jlong j%s" n
9709       ) (snd style);
9710       pr ")\n";
9711       pr "{\n";
9712       pr "  guestfs_h *g = (guestfs_h *) (long) jg;\n";
9713       let error_code, no_ret =
9714         match fst style with
9715         | RErr -> pr "  int r;\n"; "-1", ""
9716         | RBool _
9717         | RInt _ -> pr "  int r;\n"; "-1", "0"
9718         | RInt64 _ -> pr "  int64_t r;\n"; "-1", "0"
9719         | RConstString _ -> pr "  const char *r;\n"; "NULL", "NULL"
9720         | RConstOptString _ -> pr "  const char *r;\n"; "NULL", "NULL"
9721         | RString _ ->
9722             pr "  jstring jr;\n";
9723             pr "  char *r;\n"; "NULL", "NULL"
9724         | RStringList _ ->
9725             pr "  jobjectArray jr;\n";
9726             pr "  int r_len;\n";
9727             pr "  jclass cl;\n";
9728             pr "  jstring jstr;\n";
9729             pr "  char **r;\n"; "NULL", "NULL"
9730         | RStruct (_, typ) ->
9731             pr "  jobject jr;\n";
9732             pr "  jclass cl;\n";
9733             pr "  jfieldID fl;\n";
9734             pr "  struct guestfs_%s *r;\n" typ; "NULL", "NULL"
9735         | RStructList (_, typ) ->
9736             pr "  jobjectArray jr;\n";
9737             pr "  jclass cl;\n";
9738             pr "  jfieldID fl;\n";
9739             pr "  jobject jfl;\n";
9740             pr "  struct guestfs_%s_list *r;\n" typ; "NULL", "NULL"
9741         | RHashtable _ -> pr "  char **r;\n"; "NULL", "NULL"
9742         | RBufferOut _ ->
9743             pr "  jstring jr;\n";
9744             pr "  char *r;\n";
9745             pr "  size_t size;\n";
9746             "NULL", "NULL" in
9747       List.iter (
9748         function
9749         | Pathname n
9750         | Device n | Dev_or_Path n
9751         | String n
9752         | OptString n
9753         | FileIn n
9754         | FileOut n ->
9755             pr "  const char *%s;\n" n
9756         | StringList n | DeviceList n ->
9757             pr "  int %s_len;\n" n;
9758             pr "  const char **%s;\n" n
9759         | Bool n
9760         | Int n ->
9761             pr "  int %s;\n" n
9762         | Int64 n ->
9763             pr "  int64_t %s;\n" n
9764       ) (snd style);
9765
9766       let needs_i =
9767         (match fst style with
9768          | RStringList _ | RStructList _ -> true
9769          | RErr | RBool _ | RInt _ | RInt64 _ | RConstString _
9770          | RConstOptString _
9771          | RString _ | RBufferOut _ | RStruct _ | RHashtable _ -> false) ||
9772           List.exists (function
9773                        | StringList _ -> true
9774                        | DeviceList _ -> true
9775                        | _ -> false) (snd style) in
9776       if needs_i then
9777         pr "  int i;\n";
9778
9779       pr "\n";
9780
9781       (* Get the parameters. *)
9782       List.iter (
9783         function
9784         | Pathname n
9785         | Device n | Dev_or_Path n
9786         | String n
9787         | FileIn n
9788         | FileOut n ->
9789             pr "  %s = (*env)->GetStringUTFChars (env, j%s, NULL);\n" n n
9790         | OptString n ->
9791             (* This is completely undocumented, but Java null becomes
9792              * a NULL parameter.
9793              *)
9794             pr "  %s = j%s ? (*env)->GetStringUTFChars (env, j%s, NULL) : NULL;\n" n n n
9795         | StringList n | DeviceList n ->
9796             pr "  %s_len = (*env)->GetArrayLength (env, j%s);\n" n n;
9797             pr "  %s = guestfs_safe_malloc (g, sizeof (char *) * (%s_len+1));\n" n n;
9798             pr "  for (i = 0; i < %s_len; ++i) {\n" n;
9799             pr "    jobject o = (*env)->GetObjectArrayElement (env, j%s, i);\n"
9800               n;
9801             pr "    %s[i] = (*env)->GetStringUTFChars (env, o, NULL);\n" n;
9802             pr "  }\n";
9803             pr "  %s[%s_len] = NULL;\n" n n;
9804         | Bool n
9805         | Int n
9806         | Int64 n ->
9807             pr "  %s = j%s;\n" n n
9808       ) (snd style);
9809
9810       (* Make the call. *)
9811       pr "  r = guestfs_%s " name;
9812       generate_c_call_args ~handle:"g" style;
9813       pr ";\n";
9814
9815       (* Release the parameters. *)
9816       List.iter (
9817         function
9818         | Pathname n
9819         | Device n | Dev_or_Path n
9820         | String n
9821         | FileIn n
9822         | FileOut n ->
9823             pr "  (*env)->ReleaseStringUTFChars (env, j%s, %s);\n" n n
9824         | OptString n ->
9825             pr "  if (j%s)\n" n;
9826             pr "    (*env)->ReleaseStringUTFChars (env, j%s, %s);\n" n n
9827         | StringList n | DeviceList n ->
9828             pr "  for (i = 0; i < %s_len; ++i) {\n" n;
9829             pr "    jobject o = (*env)->GetObjectArrayElement (env, j%s, i);\n"
9830               n;
9831             pr "    (*env)->ReleaseStringUTFChars (env, o, %s[i]);\n" n;
9832             pr "  }\n";
9833             pr "  free (%s);\n" n
9834         | Bool n
9835         | Int n
9836         | Int64 n -> ()
9837       ) (snd style);
9838
9839       (* Check for errors. *)
9840       pr "  if (r == %s) {\n" error_code;
9841       pr "    throw_exception (env, guestfs_last_error (g));\n";
9842       pr "    return %s;\n" no_ret;
9843       pr "  }\n";
9844
9845       (* Return value. *)
9846       (match fst style with
9847        | RErr -> ()
9848        | RInt _ -> pr "  return (jint) r;\n"
9849        | RBool _ -> pr "  return (jboolean) r;\n"
9850        | RInt64 _ -> pr "  return (jlong) r;\n"
9851        | RConstString _ -> pr "  return (*env)->NewStringUTF (env, r);\n"
9852        | RConstOptString _ ->
9853            pr "  return (*env)->NewStringUTF (env, r); /* XXX r NULL? */\n"
9854        | RString _ ->
9855            pr "  jr = (*env)->NewStringUTF (env, r);\n";
9856            pr "  free (r);\n";
9857            pr "  return jr;\n"
9858        | RStringList _ ->
9859            pr "  for (r_len = 0; r[r_len] != NULL; ++r_len) ;\n";
9860            pr "  cl = (*env)->FindClass (env, \"java/lang/String\");\n";
9861            pr "  jstr = (*env)->NewStringUTF (env, \"\");\n";
9862            pr "  jr = (*env)->NewObjectArray (env, r_len, cl, jstr);\n";
9863            pr "  for (i = 0; i < r_len; ++i) {\n";
9864            pr "    jstr = (*env)->NewStringUTF (env, r[i]);\n";
9865            pr "    (*env)->SetObjectArrayElement (env, jr, i, jstr);\n";
9866            pr "    free (r[i]);\n";
9867            pr "  }\n";
9868            pr "  free (r);\n";
9869            pr "  return jr;\n"
9870        | RStruct (_, typ) ->
9871            let jtyp = java_name_of_struct typ in
9872            let cols = cols_of_struct typ in
9873            generate_java_struct_return typ jtyp cols
9874        | RStructList (_, typ) ->
9875            let jtyp = java_name_of_struct typ in
9876            let cols = cols_of_struct typ in
9877            generate_java_struct_list_return typ jtyp cols
9878        | RHashtable _ ->
9879            (* XXX *)
9880            pr "  throw_exception (env, \"%s: internal error: please let us know how to make a Java HashMap from JNI bindings!\");\n" name;
9881            pr "  return NULL;\n"
9882        | RBufferOut _ ->
9883            pr "  jr = (*env)->NewStringUTF (env, r); /* XXX size */\n";
9884            pr "  free (r);\n";
9885            pr "  return jr;\n"
9886       );
9887
9888       pr "}\n";
9889       pr "\n"
9890   ) all_functions
9891
9892 and generate_java_struct_return typ jtyp cols =
9893   pr "  cl = (*env)->FindClass (env, \"com/redhat/et/libguestfs/%s\");\n" jtyp;
9894   pr "  jr = (*env)->AllocObject (env, cl);\n";
9895   List.iter (
9896     function
9897     | name, FString ->
9898         pr "  fl = (*env)->GetFieldID (env, cl, \"%s\", \"Ljava/lang/String;\");\n" name;
9899         pr "  (*env)->SetObjectField (env, jr, fl, (*env)->NewStringUTF (env, r->%s));\n" name;
9900     | name, FUUID ->
9901         pr "  {\n";
9902         pr "    char s[33];\n";
9903         pr "    memcpy (s, r->%s, 32);\n" name;
9904         pr "    s[32] = 0;\n";
9905         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"Ljava/lang/String;\");\n" name;
9906         pr "    (*env)->SetObjectField (env, jr, fl, (*env)->NewStringUTF (env, s));\n";
9907         pr "  }\n";
9908     | name, FBuffer ->
9909         pr "  {\n";
9910         pr "    int len = r->%s_len;\n" name;
9911         pr "    char s[len+1];\n";
9912         pr "    memcpy (s, r->%s, len);\n" name;
9913         pr "    s[len] = 0;\n";
9914         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"Ljava/lang/String;\");\n" name;
9915         pr "    (*env)->SetObjectField (env, jr, fl, (*env)->NewStringUTF (env, s));\n";
9916         pr "  }\n";
9917     | name, (FBytes|FUInt64|FInt64) ->
9918         pr "  fl = (*env)->GetFieldID (env, cl, \"%s\", \"J\");\n" name;
9919         pr "  (*env)->SetLongField (env, jr, fl, r->%s);\n" name;
9920     | name, (FUInt32|FInt32) ->
9921         pr "  fl = (*env)->GetFieldID (env, cl, \"%s\", \"I\");\n" name;
9922         pr "  (*env)->SetLongField (env, jr, fl, r->%s);\n" name;
9923     | name, FOptPercent ->
9924         pr "  fl = (*env)->GetFieldID (env, cl, \"%s\", \"F\");\n" name;
9925         pr "  (*env)->SetFloatField (env, jr, fl, r->%s);\n" name;
9926     | name, FChar ->
9927         pr "  fl = (*env)->GetFieldID (env, cl, \"%s\", \"C\");\n" name;
9928         pr "  (*env)->SetLongField (env, jr, fl, r->%s);\n" name;
9929   ) cols;
9930   pr "  free (r);\n";
9931   pr "  return jr;\n"
9932
9933 and generate_java_struct_list_return typ jtyp cols =
9934   pr "  cl = (*env)->FindClass (env, \"com/redhat/et/libguestfs/%s\");\n" jtyp;
9935   pr "  jr = (*env)->NewObjectArray (env, r->len, cl, NULL);\n";
9936   pr "  for (i = 0; i < r->len; ++i) {\n";
9937   pr "    jfl = (*env)->AllocObject (env, cl);\n";
9938   List.iter (
9939     function
9940     | name, FString ->
9941         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"Ljava/lang/String;\");\n" name;
9942         pr "    (*env)->SetObjectField (env, jfl, fl, (*env)->NewStringUTF (env, r->val[i].%s));\n" name;
9943     | name, FUUID ->
9944         pr "    {\n";
9945         pr "      char s[33];\n";
9946         pr "      memcpy (s, r->val[i].%s, 32);\n" name;
9947         pr "      s[32] = 0;\n";
9948         pr "      fl = (*env)->GetFieldID (env, cl, \"%s\", \"Ljava/lang/String;\");\n" name;
9949         pr "      (*env)->SetObjectField (env, jfl, fl, (*env)->NewStringUTF (env, s));\n";
9950         pr "    }\n";
9951     | name, FBuffer ->
9952         pr "    {\n";
9953         pr "      int len = r->val[i].%s_len;\n" name;
9954         pr "      char s[len+1];\n";
9955         pr "      memcpy (s, r->val[i].%s, len);\n" name;
9956         pr "      s[len] = 0;\n";
9957         pr "      fl = (*env)->GetFieldID (env, cl, \"%s\", \"Ljava/lang/String;\");\n" name;
9958         pr "      (*env)->SetObjectField (env, jfl, fl, (*env)->NewStringUTF (env, s));\n";
9959         pr "    }\n";
9960     | name, (FBytes|FUInt64|FInt64) ->
9961         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"J\");\n" name;
9962         pr "    (*env)->SetLongField (env, jfl, fl, r->val[i].%s);\n" name;
9963     | name, (FUInt32|FInt32) ->
9964         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"I\");\n" name;
9965         pr "    (*env)->SetLongField (env, jfl, fl, r->val[i].%s);\n" name;
9966     | name, FOptPercent ->
9967         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"F\");\n" name;
9968         pr "    (*env)->SetFloatField (env, jfl, fl, r->val[i].%s);\n" name;
9969     | name, FChar ->
9970         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"C\");\n" name;
9971         pr "    (*env)->SetLongField (env, jfl, fl, r->val[i].%s);\n" name;
9972   ) cols;
9973   pr "    (*env)->SetObjectArrayElement (env, jfl, i, jfl);\n";
9974   pr "  }\n";
9975   pr "  guestfs_free_%s_list (r);\n" typ;
9976   pr "  return jr;\n"
9977
9978 and generate_java_makefile_inc () =
9979   generate_header HashStyle GPLv2plus;
9980
9981   pr "java_built_sources = \\\n";
9982   List.iter (
9983     fun (typ, jtyp) ->
9984         pr "\tcom/redhat/et/libguestfs/%s.java \\\n" jtyp;
9985   ) java_structs;
9986   pr "\tcom/redhat/et/libguestfs/GuestFS.java\n"
9987
9988 and generate_haskell_hs () =
9989   generate_header HaskellStyle LGPLv2plus;
9990
9991   (* XXX We only know how to generate partial FFI for Haskell
9992    * at the moment.  Please help out!
9993    *)
9994   let can_generate style =
9995     match style with
9996     | RErr, _
9997     | RInt _, _
9998     | RInt64 _, _ -> true
9999     | RBool _, _
10000     | RConstString _, _
10001     | RConstOptString _, _
10002     | RString _, _
10003     | RStringList _, _
10004     | RStruct _, _
10005     | RStructList _, _
10006     | RHashtable _, _
10007     | RBufferOut _, _ -> false in
10008
10009   pr "\
10010 {-# INCLUDE <guestfs.h> #-}
10011 {-# LANGUAGE ForeignFunctionInterface #-}
10012
10013 module Guestfs (
10014   create";
10015
10016   (* List out the names of the actions we want to export. *)
10017   List.iter (
10018     fun (name, style, _, _, _, _, _) ->
10019       if can_generate style then pr ",\n  %s" name
10020   ) all_functions;
10021
10022   pr "
10023   ) where
10024
10025 -- Unfortunately some symbols duplicate ones already present
10026 -- in Prelude.  We don't know which, so we hard-code a list
10027 -- here.
10028 import Prelude hiding (truncate)
10029
10030 import Foreign
10031 import Foreign.C
10032 import Foreign.C.Types
10033 import IO
10034 import Control.Exception
10035 import Data.Typeable
10036
10037 data GuestfsS = GuestfsS            -- represents the opaque C struct
10038 type GuestfsP = Ptr GuestfsS        -- guestfs_h *
10039 type GuestfsH = ForeignPtr GuestfsS -- guestfs_h * with attached finalizer
10040
10041 -- XXX define properly later XXX
10042 data PV = PV
10043 data VG = VG
10044 data LV = LV
10045 data IntBool = IntBool
10046 data Stat = Stat
10047 data StatVFS = StatVFS
10048 data Hashtable = Hashtable
10049
10050 foreign import ccall unsafe \"guestfs_create\" c_create
10051   :: IO GuestfsP
10052 foreign import ccall unsafe \"&guestfs_close\" c_close
10053   :: FunPtr (GuestfsP -> IO ())
10054 foreign import ccall unsafe \"guestfs_set_error_handler\" c_set_error_handler
10055   :: GuestfsP -> Ptr CInt -> Ptr CInt -> IO ()
10056
10057 create :: IO GuestfsH
10058 create = do
10059   p <- c_create
10060   c_set_error_handler p nullPtr nullPtr
10061   h <- newForeignPtr c_close p
10062   return h
10063
10064 foreign import ccall unsafe \"guestfs_last_error\" c_last_error
10065   :: GuestfsP -> IO CString
10066
10067 -- last_error :: GuestfsH -> IO (Maybe String)
10068 -- last_error h = do
10069 --   str <- withForeignPtr h (\\p -> c_last_error p)
10070 --   maybePeek peekCString str
10071
10072 last_error :: GuestfsH -> IO (String)
10073 last_error h = do
10074   str <- withForeignPtr h (\\p -> c_last_error p)
10075   if (str == nullPtr)
10076     then return \"no error\"
10077     else peekCString str
10078
10079 ";
10080
10081   (* Generate wrappers for each foreign function. *)
10082   List.iter (
10083     fun (name, style, _, _, _, _, _) ->
10084       if can_generate style then (
10085         pr "foreign import ccall unsafe \"guestfs_%s\" c_%s\n" name name;
10086         pr "  :: ";
10087         generate_haskell_prototype ~handle:"GuestfsP" style;
10088         pr "\n";
10089         pr "\n";
10090         pr "%s :: " name;
10091         generate_haskell_prototype ~handle:"GuestfsH" ~hs:true style;
10092         pr "\n";
10093         pr "%s %s = do\n" name
10094           (String.concat " " ("h" :: List.map name_of_argt (snd style)));
10095         pr "  r <- ";
10096         (* Convert pointer arguments using with* functions. *)
10097         List.iter (
10098           function
10099           | FileIn n
10100           | FileOut n
10101           | Pathname n | Device n | Dev_or_Path n | String n -> pr "withCString %s $ \\%s -> " n n
10102           | OptString n -> pr "maybeWith withCString %s $ \\%s -> " n n
10103           | StringList n | DeviceList n -> pr "withMany withCString %s $ \\%s -> withArray0 nullPtr %s $ \\%s -> " n n n n
10104           | Bool _ | Int _ | Int64 _ -> ()
10105         ) (snd style);
10106         (* Convert integer arguments. *)
10107         let args =
10108           List.map (
10109             function
10110             | Bool n -> sprintf "(fromBool %s)" n
10111             | Int n -> sprintf "(fromIntegral %s)" n
10112             | Int64 n -> sprintf "(fromIntegral %s)" n
10113             | FileIn n | FileOut n
10114             | Pathname n | Device n | Dev_or_Path n | String n | OptString n | StringList n | DeviceList n -> n
10115           ) (snd style) in
10116         pr "withForeignPtr h (\\p -> c_%s %s)\n" name
10117           (String.concat " " ("p" :: args));
10118         (match fst style with
10119          | RErr | RInt _ | RInt64 _ | RBool _ ->
10120              pr "  if (r == -1)\n";
10121              pr "    then do\n";
10122              pr "      err <- last_error h\n";
10123              pr "      fail err\n";
10124          | RConstString _ | RConstOptString _ | RString _
10125          | RStringList _ | RStruct _
10126          | RStructList _ | RHashtable _ | RBufferOut _ ->
10127              pr "  if (r == nullPtr)\n";
10128              pr "    then do\n";
10129              pr "      err <- last_error h\n";
10130              pr "      fail err\n";
10131         );
10132         (match fst style with
10133          | RErr ->
10134              pr "    else return ()\n"
10135          | RInt _ ->
10136              pr "    else return (fromIntegral r)\n"
10137          | RInt64 _ ->
10138              pr "    else return (fromIntegral r)\n"
10139          | RBool _ ->
10140              pr "    else return (toBool r)\n"
10141          | RConstString _
10142          | RConstOptString _
10143          | RString _
10144          | RStringList _
10145          | RStruct _
10146          | RStructList _
10147          | RHashtable _
10148          | RBufferOut _ ->
10149              pr "    else return ()\n" (* XXXXXXXXXXXXXXXXXXXX *)
10150         );
10151         pr "\n";
10152       )
10153   ) all_functions
10154
10155 and generate_haskell_prototype ~handle ?(hs = false) style =
10156   pr "%s -> " handle;
10157   let string = if hs then "String" else "CString" in
10158   let int = if hs then "Int" else "CInt" in
10159   let bool = if hs then "Bool" else "CInt" in
10160   let int64 = if hs then "Integer" else "Int64" in
10161   List.iter (
10162     fun arg ->
10163       (match arg with
10164        | Pathname _ | Device _ | Dev_or_Path _ | String _ -> pr "%s" string
10165        | OptString _ -> if hs then pr "Maybe String" else pr "CString"
10166        | StringList _ | DeviceList _ -> if hs then pr "[String]" else pr "Ptr CString"
10167        | Bool _ -> pr "%s" bool
10168        | Int _ -> pr "%s" int
10169        | Int64 _ -> pr "%s" int
10170        | FileIn _ -> pr "%s" string
10171        | FileOut _ -> pr "%s" string
10172       );
10173       pr " -> ";
10174   ) (snd style);
10175   pr "IO (";
10176   (match fst style with
10177    | RErr -> if not hs then pr "CInt"
10178    | RInt _ -> pr "%s" int
10179    | RInt64 _ -> pr "%s" int64
10180    | RBool _ -> pr "%s" bool
10181    | RConstString _ -> pr "%s" string
10182    | RConstOptString _ -> pr "Maybe %s" string
10183    | RString _ -> pr "%s" string
10184    | RStringList _ -> pr "[%s]" string
10185    | RStruct (_, typ) ->
10186        let name = java_name_of_struct typ in
10187        pr "%s" name
10188    | RStructList (_, typ) ->
10189        let name = java_name_of_struct typ in
10190        pr "[%s]" name
10191    | RHashtable _ -> pr "Hashtable"
10192    | RBufferOut _ -> pr "%s" string
10193   );
10194   pr ")"
10195
10196 and generate_csharp () =
10197   generate_header CPlusPlusStyle LGPLv2plus;
10198
10199   (* XXX Make this configurable by the C# assembly users. *)
10200   let library = "libguestfs.so.0" in
10201
10202   pr "\
10203 // These C# bindings are highly experimental at present.
10204 //
10205 // Firstly they only work on Linux (ie. Mono).  In order to get them
10206 // to work on Windows (ie. .Net) you would need to port the library
10207 // itself to Windows first.
10208 //
10209 // The second issue is that some calls are known to be incorrect and
10210 // can cause Mono to segfault.  Particularly: calls which pass or
10211 // return string[], or return any structure value.  This is because
10212 // we haven't worked out the correct way to do this from C#.
10213 //
10214 // The third issue is that when compiling you get a lot of warnings.
10215 // We are not sure whether the warnings are important or not.
10216 //
10217 // Fourthly we do not routinely build or test these bindings as part
10218 // of the make && make check cycle, which means that regressions might
10219 // go unnoticed.
10220 //
10221 // Suggestions and patches are welcome.
10222
10223 // To compile:
10224 //
10225 // gmcs Libguestfs.cs
10226 // mono Libguestfs.exe
10227 //
10228 // (You'll probably want to add a Test class / static main function
10229 // otherwise this won't do anything useful).
10230
10231 using System;
10232 using System.IO;
10233 using System.Runtime.InteropServices;
10234 using System.Runtime.Serialization;
10235 using System.Collections;
10236
10237 namespace Guestfs
10238 {
10239   class Error : System.ApplicationException
10240   {
10241     public Error (string message) : base (message) {}
10242     protected Error (SerializationInfo info, StreamingContext context) {}
10243   }
10244
10245   class Guestfs
10246   {
10247     IntPtr _handle;
10248
10249     [DllImport (\"%s\")]
10250     static extern IntPtr guestfs_create ();
10251
10252     public Guestfs ()
10253     {
10254       _handle = guestfs_create ();
10255       if (_handle == IntPtr.Zero)
10256         throw new Error (\"could not create guestfs handle\");
10257     }
10258
10259     [DllImport (\"%s\")]
10260     static extern void guestfs_close (IntPtr h);
10261
10262     ~Guestfs ()
10263     {
10264       guestfs_close (_handle);
10265     }
10266
10267     [DllImport (\"%s\")]
10268     static extern string guestfs_last_error (IntPtr h);
10269
10270 " library library library;
10271
10272   (* Generate C# structure bindings.  We prefix struct names with
10273    * underscore because C# cannot have conflicting struct names and
10274    * method names (eg. "class stat" and "stat").
10275    *)
10276   List.iter (
10277     fun (typ, cols) ->
10278       pr "    [StructLayout (LayoutKind.Sequential)]\n";
10279       pr "    public class _%s {\n" typ;
10280       List.iter (
10281         function
10282         | name, FChar -> pr "      char %s;\n" name
10283         | name, FString -> pr "      string %s;\n" name
10284         | name, FBuffer ->
10285             pr "      uint %s_len;\n" name;
10286             pr "      string %s;\n" name
10287         | name, FUUID ->
10288             pr "      [MarshalAs (UnmanagedType.ByValTStr, SizeConst=16)]\n";
10289             pr "      string %s;\n" name
10290         | name, FUInt32 -> pr "      uint %s;\n" name
10291         | name, FInt32 -> pr "      int %s;\n" name
10292         | name, (FUInt64|FBytes) -> pr "      ulong %s;\n" name
10293         | name, FInt64 -> pr "      long %s;\n" name
10294         | name, FOptPercent -> pr "      float %s; /* [0..100] or -1 */\n" name
10295       ) cols;
10296       pr "    }\n";
10297       pr "\n"
10298   ) structs;
10299
10300   (* Generate C# function bindings. *)
10301   List.iter (
10302     fun (name, style, _, _, _, shortdesc, _) ->
10303       let rec csharp_return_type () =
10304         match fst style with
10305         | RErr -> "void"
10306         | RBool n -> "bool"
10307         | RInt n -> "int"
10308         | RInt64 n -> "long"
10309         | RConstString n
10310         | RConstOptString n
10311         | RString n
10312         | RBufferOut n -> "string"
10313         | RStruct (_,n) -> "_" ^ n
10314         | RHashtable n -> "Hashtable"
10315         | RStringList n -> "string[]"
10316         | RStructList (_,n) -> sprintf "_%s[]" n
10317
10318       and c_return_type () =
10319         match fst style with
10320         | RErr
10321         | RBool _
10322         | RInt _ -> "int"
10323         | RInt64 _ -> "long"
10324         | RConstString _
10325         | RConstOptString _
10326         | RString _
10327         | RBufferOut _ -> "string"
10328         | RStruct (_,n) -> "_" ^ n
10329         | RHashtable _
10330         | RStringList _ -> "string[]"
10331         | RStructList (_,n) -> sprintf "_%s[]" n
10332
10333       and c_error_comparison () =
10334         match fst style with
10335         | RErr
10336         | RBool _
10337         | RInt _
10338         | RInt64 _ -> "== -1"
10339         | RConstString _
10340         | RConstOptString _
10341         | RString _
10342         | RBufferOut _
10343         | RStruct (_,_)
10344         | RHashtable _
10345         | RStringList _
10346         | RStructList (_,_) -> "== null"
10347
10348       and generate_extern_prototype () =
10349         pr "    static extern %s guestfs_%s (IntPtr h"
10350           (c_return_type ()) name;
10351         List.iter (
10352           function
10353           | Pathname n | Device n | Dev_or_Path n | String n | OptString n
10354           | FileIn n | FileOut n ->
10355               pr ", [In] string %s" n
10356           | StringList n | DeviceList n ->
10357               pr ", [In] string[] %s" n
10358           | Bool n ->
10359               pr ", bool %s" n
10360           | Int n ->
10361               pr ", int %s" n
10362           | Int64 n ->
10363               pr ", long %s" n
10364         ) (snd style);
10365         pr ");\n"
10366
10367       and generate_public_prototype () =
10368         pr "    public %s %s (" (csharp_return_type ()) name;
10369         let comma = ref false in
10370         let next () =
10371           if !comma then pr ", ";
10372           comma := true
10373         in
10374         List.iter (
10375           function
10376           | Pathname n | Device n | Dev_or_Path n | String n | OptString n
10377           | FileIn n | FileOut n ->
10378               next (); pr "string %s" n
10379           | StringList n | DeviceList n ->
10380               next (); pr "string[] %s" n
10381           | Bool n ->
10382               next (); pr "bool %s" n
10383           | Int n ->
10384               next (); pr "int %s" n
10385           | Int64 n ->
10386               next (); pr "long %s" n
10387         ) (snd style);
10388         pr ")\n"
10389
10390       and generate_call () =
10391         pr "guestfs_%s (_handle" name;
10392         List.iter (fun arg -> pr ", %s" (name_of_argt arg)) (snd style);
10393         pr ");\n";
10394       in
10395
10396       pr "    [DllImport (\"%s\")]\n" library;
10397       generate_extern_prototype ();
10398       pr "\n";
10399       pr "    /// <summary>\n";
10400       pr "    /// %s\n" shortdesc;
10401       pr "    /// </summary>\n";
10402       generate_public_prototype ();
10403       pr "    {\n";
10404       pr "      %s r;\n" (c_return_type ());
10405       pr "      r = ";
10406       generate_call ();
10407       pr "      if (r %s)\n" (c_error_comparison ());
10408       pr "        throw new Error (guestfs_last_error (_handle));\n";
10409       (match fst style with
10410        | RErr -> ()
10411        | RBool _ ->
10412            pr "      return r != 0 ? true : false;\n"
10413        | RHashtable _ ->
10414            pr "      Hashtable rr = new Hashtable ();\n";
10415            pr "      for (int i = 0; i < r.Length; i += 2)\n";
10416            pr "        rr.Add (r[i], r[i+1]);\n";
10417            pr "      return rr;\n"
10418        | RInt _ | RInt64 _ | RConstString _ | RConstOptString _
10419        | RString _ | RBufferOut _ | RStruct _ | RStringList _
10420        | RStructList _ ->
10421            pr "      return r;\n"
10422       );
10423       pr "    }\n";
10424       pr "\n";
10425   ) all_functions_sorted;
10426
10427   pr "  }
10428 }
10429 "
10430
10431 and generate_bindtests () =
10432   generate_header CStyle LGPLv2plus;
10433
10434   pr "\
10435 #include <stdio.h>
10436 #include <stdlib.h>
10437 #include <inttypes.h>
10438 #include <string.h>
10439
10440 #include \"guestfs.h\"
10441 #include \"guestfs-internal.h\"
10442 #include \"guestfs-internal-actions.h\"
10443 #include \"guestfs_protocol.h\"
10444
10445 #define error guestfs_error
10446 #define safe_calloc guestfs_safe_calloc
10447 #define safe_malloc guestfs_safe_malloc
10448
10449 static void
10450 print_strings (char *const *argv)
10451 {
10452   int argc;
10453
10454   printf (\"[\");
10455   for (argc = 0; argv[argc] != NULL; ++argc) {
10456     if (argc > 0) printf (\", \");
10457     printf (\"\\\"%%s\\\"\", argv[argc]);
10458   }
10459   printf (\"]\\n\");
10460 }
10461
10462 /* The test0 function prints its parameters to stdout. */
10463 ";
10464
10465   let test0, tests =
10466     match test_functions with
10467     | [] -> assert false
10468     | test0 :: tests -> test0, tests in
10469
10470   let () =
10471     let (name, style, _, _, _, _, _) = test0 in
10472     generate_prototype ~extern:false ~semicolon:false ~newline:true
10473       ~handle:"g" ~prefix:"guestfs__" name style;
10474     pr "{\n";
10475     List.iter (
10476       function
10477       | Pathname n
10478       | Device n | Dev_or_Path n
10479       | String n
10480       | FileIn n
10481       | FileOut n -> pr "  printf (\"%%s\\n\", %s);\n" n
10482       | OptString n -> pr "  printf (\"%%s\\n\", %s ? %s : \"null\");\n" n n
10483       | StringList n | DeviceList n -> pr "  print_strings (%s);\n" n
10484       | Bool n -> pr "  printf (\"%%s\\n\", %s ? \"true\" : \"false\");\n" n
10485       | Int n -> pr "  printf (\"%%d\\n\", %s);\n" n
10486       | Int64 n -> pr "  printf (\"%%\" PRIi64 \"\\n\", %s);\n" n
10487     ) (snd style);
10488     pr "  /* Java changes stdout line buffering so we need this: */\n";
10489     pr "  fflush (stdout);\n";
10490     pr "  return 0;\n";
10491     pr "}\n";
10492     pr "\n" in
10493
10494   List.iter (
10495     fun (name, style, _, _, _, _, _) ->
10496       if String.sub name (String.length name - 3) 3 <> "err" then (
10497         pr "/* Test normal return. */\n";
10498         generate_prototype ~extern:false ~semicolon:false ~newline:true
10499           ~handle:"g" ~prefix:"guestfs__" name style;
10500         pr "{\n";
10501         (match fst style with
10502          | RErr ->
10503              pr "  return 0;\n"
10504          | RInt _ ->
10505              pr "  int r;\n";
10506              pr "  sscanf (val, \"%%d\", &r);\n";
10507              pr "  return r;\n"
10508          | RInt64 _ ->
10509              pr "  int64_t r;\n";
10510              pr "  sscanf (val, \"%%\" SCNi64, &r);\n";
10511              pr "  return r;\n"
10512          | RBool _ ->
10513              pr "  return STREQ (val, \"true\");\n"
10514          | RConstString _
10515          | RConstOptString _ ->
10516              (* Can't return the input string here.  Return a static
10517               * string so we ensure we get a segfault if the caller
10518               * tries to free it.
10519               *)
10520              pr "  return \"static string\";\n"
10521          | RString _ ->
10522              pr "  return strdup (val);\n"
10523          | RStringList _ ->
10524              pr "  char **strs;\n";
10525              pr "  int n, i;\n";
10526              pr "  sscanf (val, \"%%d\", &n);\n";
10527              pr "  strs = safe_malloc (g, (n+1) * sizeof (char *));\n";
10528              pr "  for (i = 0; i < n; ++i) {\n";
10529              pr "    strs[i] = safe_malloc (g, 16);\n";
10530              pr "    snprintf (strs[i], 16, \"%%d\", i);\n";
10531              pr "  }\n";
10532              pr "  strs[n] = NULL;\n";
10533              pr "  return strs;\n"
10534          | RStruct (_, typ) ->
10535              pr "  struct guestfs_%s *r;\n" typ;
10536              pr "  r = safe_calloc (g, sizeof *r, 1);\n";
10537              pr "  return r;\n"
10538          | RStructList (_, typ) ->
10539              pr "  struct guestfs_%s_list *r;\n" typ;
10540              pr "  r = safe_calloc (g, sizeof *r, 1);\n";
10541              pr "  sscanf (val, \"%%d\", &r->len);\n";
10542              pr "  r->val = safe_calloc (g, r->len, sizeof *r->val);\n";
10543              pr "  return r;\n"
10544          | RHashtable _ ->
10545              pr "  char **strs;\n";
10546              pr "  int n, i;\n";
10547              pr "  sscanf (val, \"%%d\", &n);\n";
10548              pr "  strs = safe_malloc (g, (n*2+1) * sizeof (*strs));\n";
10549              pr "  for (i = 0; i < n; ++i) {\n";
10550              pr "    strs[i*2] = safe_malloc (g, 16);\n";
10551              pr "    strs[i*2+1] = safe_malloc (g, 16);\n";
10552              pr "    snprintf (strs[i*2], 16, \"%%d\", i);\n";
10553              pr "    snprintf (strs[i*2+1], 16, \"%%d\", i);\n";
10554              pr "  }\n";
10555              pr "  strs[n*2] = NULL;\n";
10556              pr "  return strs;\n"
10557          | RBufferOut _ ->
10558              pr "  return strdup (val);\n"
10559         );
10560         pr "}\n";
10561         pr "\n"
10562       ) else (
10563         pr "/* Test error return. */\n";
10564         generate_prototype ~extern:false ~semicolon:false ~newline:true
10565           ~handle:"g" ~prefix:"guestfs__" name style;
10566         pr "{\n";
10567         pr "  error (g, \"error\");\n";
10568         (match fst style with
10569          | RErr | RInt _ | RInt64 _ | RBool _ ->
10570              pr "  return -1;\n"
10571          | RConstString _ | RConstOptString _
10572          | RString _ | RStringList _ | RStruct _
10573          | RStructList _
10574          | RHashtable _
10575          | RBufferOut _ ->
10576              pr "  return NULL;\n"
10577         );
10578         pr "}\n";
10579         pr "\n"
10580       )
10581   ) tests
10582
10583 and generate_ocaml_bindtests () =
10584   generate_header OCamlStyle GPLv2plus;
10585
10586   pr "\
10587 let () =
10588   let g = Guestfs.create () in
10589 ";
10590
10591   let mkargs args =
10592     String.concat " " (
10593       List.map (
10594         function
10595         | CallString s -> "\"" ^ s ^ "\""
10596         | CallOptString None -> "None"
10597         | CallOptString (Some s) -> sprintf "(Some \"%s\")" s
10598         | CallStringList xs ->
10599             "[|" ^ String.concat ";" (List.map (sprintf "\"%s\"") xs) ^ "|]"
10600         | CallInt i when i >= 0 -> string_of_int i
10601         | CallInt i (* when i < 0 *) -> "(" ^ string_of_int i ^ ")"
10602         | CallInt64 i when i >= 0L -> Int64.to_string i ^ "L"
10603         | CallInt64 i (* when i < 0L *) -> "(" ^ Int64.to_string i ^ "L)"
10604         | CallBool b -> string_of_bool b
10605       ) args
10606     )
10607   in
10608
10609   generate_lang_bindtests (
10610     fun f args -> pr "  Guestfs.%s g %s;\n" f (mkargs args)
10611   );
10612
10613   pr "print_endline \"EOF\"\n"
10614
10615 and generate_perl_bindtests () =
10616   pr "#!/usr/bin/perl -w\n";
10617   generate_header HashStyle GPLv2plus;
10618
10619   pr "\
10620 use strict;
10621
10622 use Sys::Guestfs;
10623
10624 my $g = Sys::Guestfs->new ();
10625 ";
10626
10627   let mkargs args =
10628     String.concat ", " (
10629       List.map (
10630         function
10631         | CallString s -> "\"" ^ s ^ "\""
10632         | CallOptString None -> "undef"
10633         | CallOptString (Some s) -> sprintf "\"%s\"" s
10634         | CallStringList xs ->
10635             "[" ^ String.concat "," (List.map (sprintf "\"%s\"") xs) ^ "]"
10636         | CallInt i -> string_of_int i
10637         | CallInt64 i -> Int64.to_string i
10638         | CallBool b -> if b then "1" else "0"
10639       ) args
10640     )
10641   in
10642
10643   generate_lang_bindtests (
10644     fun f args -> pr "$g->%s (%s);\n" f (mkargs args)
10645   );
10646
10647   pr "print \"EOF\\n\"\n"
10648
10649 and generate_python_bindtests () =
10650   generate_header HashStyle GPLv2plus;
10651
10652   pr "\
10653 import guestfs
10654
10655 g = guestfs.GuestFS ()
10656 ";
10657
10658   let mkargs args =
10659     String.concat ", " (
10660       List.map (
10661         function
10662         | CallString s -> "\"" ^ s ^ "\""
10663         | CallOptString None -> "None"
10664         | CallOptString (Some s) -> sprintf "\"%s\"" s
10665         | CallStringList xs ->
10666             "[" ^ String.concat "," (List.map (sprintf "\"%s\"") xs) ^ "]"
10667         | CallInt i -> string_of_int i
10668         | CallInt64 i -> Int64.to_string i
10669         | CallBool b -> if b then "1" else "0"
10670       ) args
10671     )
10672   in
10673
10674   generate_lang_bindtests (
10675     fun f args -> pr "g.%s (%s)\n" f (mkargs args)
10676   );
10677
10678   pr "print \"EOF\"\n"
10679
10680 and generate_ruby_bindtests () =
10681   generate_header HashStyle GPLv2plus;
10682
10683   pr "\
10684 require 'guestfs'
10685
10686 g = Guestfs::create()
10687 ";
10688
10689   let mkargs args =
10690     String.concat ", " (
10691       List.map (
10692         function
10693         | CallString s -> "\"" ^ s ^ "\""
10694         | CallOptString None -> "nil"
10695         | CallOptString (Some s) -> sprintf "\"%s\"" s
10696         | CallStringList xs ->
10697             "[" ^ String.concat "," (List.map (sprintf "\"%s\"") xs) ^ "]"
10698         | CallInt i -> string_of_int i
10699         | CallInt64 i -> Int64.to_string i
10700         | CallBool b -> string_of_bool b
10701       ) args
10702     )
10703   in
10704
10705   generate_lang_bindtests (
10706     fun f args -> pr "g.%s(%s)\n" f (mkargs args)
10707   );
10708
10709   pr "print \"EOF\\n\"\n"
10710
10711 and generate_java_bindtests () =
10712   generate_header CStyle GPLv2plus;
10713
10714   pr "\
10715 import com.redhat.et.libguestfs.*;
10716
10717 public class Bindtests {
10718     public static void main (String[] argv)
10719     {
10720         try {
10721             GuestFS g = new GuestFS ();
10722 ";
10723
10724   let mkargs args =
10725     String.concat ", " (
10726       List.map (
10727         function
10728         | CallString s -> "\"" ^ s ^ "\""
10729         | CallOptString None -> "null"
10730         | CallOptString (Some s) -> sprintf "\"%s\"" s
10731         | CallStringList xs ->
10732             "new String[]{" ^
10733               String.concat "," (List.map (sprintf "\"%s\"") xs) ^ "}"
10734         | CallInt i -> string_of_int i
10735         | CallInt64 i -> Int64.to_string i
10736         | CallBool b -> string_of_bool b
10737       ) args
10738     )
10739   in
10740
10741   generate_lang_bindtests (
10742     fun f args -> pr "            g.%s (%s);\n" f (mkargs args)
10743   );
10744
10745   pr "
10746             System.out.println (\"EOF\");
10747         }
10748         catch (Exception exn) {
10749             System.err.println (exn);
10750             System.exit (1);
10751         }
10752     }
10753 }
10754 "
10755
10756 and generate_haskell_bindtests () =
10757   generate_header HaskellStyle GPLv2plus;
10758
10759   pr "\
10760 module Bindtests where
10761 import qualified Guestfs
10762
10763 main = do
10764   g <- Guestfs.create
10765 ";
10766
10767   let mkargs args =
10768     String.concat " " (
10769       List.map (
10770         function
10771         | CallString s -> "\"" ^ s ^ "\""
10772         | CallOptString None -> "Nothing"
10773         | CallOptString (Some s) -> sprintf "(Just \"%s\")" s
10774         | CallStringList xs ->
10775             "[" ^ String.concat "," (List.map (sprintf "\"%s\"") xs) ^ "]"
10776         | CallInt i when i < 0 -> "(" ^ string_of_int i ^ ")"
10777         | CallInt i -> string_of_int i
10778         | CallInt64 i when i < 0L -> "(" ^ Int64.to_string i ^ ")"
10779         | CallInt64 i -> Int64.to_string i
10780         | CallBool true -> "True"
10781         | CallBool false -> "False"
10782       ) args
10783     )
10784   in
10785
10786   generate_lang_bindtests (
10787     fun f args -> pr "  Guestfs.%s g %s\n" f (mkargs args)
10788   );
10789
10790   pr "  putStrLn \"EOF\"\n"
10791
10792 (* Language-independent bindings tests - we do it this way to
10793  * ensure there is parity in testing bindings across all languages.
10794  *)
10795 and generate_lang_bindtests call =
10796   call "test0" [CallString "abc"; CallOptString (Some "def");
10797                 CallStringList []; CallBool false;
10798                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10799   call "test0" [CallString "abc"; CallOptString None;
10800                 CallStringList []; CallBool false;
10801                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10802   call "test0" [CallString ""; CallOptString (Some "def");
10803                 CallStringList []; CallBool false;
10804                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10805   call "test0" [CallString ""; CallOptString (Some "");
10806                 CallStringList []; CallBool false;
10807                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10808   call "test0" [CallString "abc"; CallOptString (Some "def");
10809                 CallStringList ["1"]; CallBool false;
10810                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10811   call "test0" [CallString "abc"; CallOptString (Some "def");
10812                 CallStringList ["1"; "2"]; CallBool false;
10813                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10814   call "test0" [CallString "abc"; CallOptString (Some "def");
10815                 CallStringList ["1"]; CallBool true;
10816                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10817   call "test0" [CallString "abc"; CallOptString (Some "def");
10818                 CallStringList ["1"]; CallBool false;
10819                 CallInt (-1); CallInt64 (-1L); CallString "123"; CallString "456"];
10820   call "test0" [CallString "abc"; CallOptString (Some "def");
10821                 CallStringList ["1"]; CallBool false;
10822                 CallInt (-2); CallInt64 (-2L); CallString "123"; CallString "456"];
10823   call "test0" [CallString "abc"; CallOptString (Some "def");
10824                 CallStringList ["1"]; CallBool false;
10825                 CallInt 1; CallInt64 1L; CallString "123"; CallString "456"];
10826   call "test0" [CallString "abc"; CallOptString (Some "def");
10827                 CallStringList ["1"]; CallBool false;
10828                 CallInt 2; CallInt64 2L; CallString "123"; CallString "456"];
10829   call "test0" [CallString "abc"; CallOptString (Some "def");
10830                 CallStringList ["1"]; CallBool false;
10831                 CallInt 4095; CallInt64 4095L; CallString "123"; CallString "456"];
10832   call "test0" [CallString "abc"; CallOptString (Some "def");
10833                 CallStringList ["1"]; CallBool false;
10834                 CallInt 0; CallInt64 0L; CallString ""; CallString ""]
10835
10836 (* XXX Add here tests of the return and error functions. *)
10837
10838 (* Code to generator bindings for virt-inspector.  Currently only
10839  * implemented for OCaml code (for virt-p2v 2.0).
10840  *)
10841 let rng_input = "inspector/virt-inspector.rng"
10842
10843 (* Read the input file and parse it into internal structures.  This is
10844  * by no means a complete RELAX NG parser, but is just enough to be
10845  * able to parse the specific input file.
10846  *)
10847 type rng =
10848   | Element of string * rng list        (* <element name=name/> *)
10849   | Attribute of string * rng list        (* <attribute name=name/> *)
10850   | Interleave of rng list                (* <interleave/> *)
10851   | ZeroOrMore of rng                        (* <zeroOrMore/> *)
10852   | OneOrMore of rng                        (* <oneOrMore/> *)
10853   | Optional of rng                        (* <optional/> *)
10854   | Choice of string list                (* <choice><value/>*</choice> *)
10855   | Value of string                        (* <value>str</value> *)
10856   | Text                                (* <text/> *)
10857
10858 let rec string_of_rng = function
10859   | Element (name, xs) ->
10860       "Element (\"" ^ name ^ "\", (" ^ string_of_rng_list xs ^ "))"
10861   | Attribute (name, xs) ->
10862       "Attribute (\"" ^ name ^ "\", (" ^ string_of_rng_list xs ^ "))"
10863   | Interleave xs -> "Interleave (" ^ string_of_rng_list xs ^ ")"
10864   | ZeroOrMore rng -> "ZeroOrMore (" ^ string_of_rng rng ^ ")"
10865   | OneOrMore rng -> "OneOrMore (" ^ string_of_rng rng ^ ")"
10866   | Optional rng -> "Optional (" ^ string_of_rng rng ^ ")"
10867   | Choice values -> "Choice [" ^ String.concat ", " values ^ "]"
10868   | Value value -> "Value \"" ^ value ^ "\""
10869   | Text -> "Text"
10870
10871 and string_of_rng_list xs =
10872   String.concat ", " (List.map string_of_rng xs)
10873
10874 let rec parse_rng ?defines context = function
10875   | [] -> []
10876   | Xml.Element ("element", ["name", name], children) :: rest ->
10877       Element (name, parse_rng ?defines context children)
10878       :: parse_rng ?defines context rest
10879   | Xml.Element ("attribute", ["name", name], children) :: rest ->
10880       Attribute (name, parse_rng ?defines context children)
10881       :: parse_rng ?defines context rest
10882   | Xml.Element ("interleave", [], children) :: rest ->
10883       Interleave (parse_rng ?defines context children)
10884       :: parse_rng ?defines context rest
10885   | Xml.Element ("zeroOrMore", [], [child]) :: rest ->
10886       let rng = parse_rng ?defines context [child] in
10887       (match rng with
10888        | [child] -> ZeroOrMore child :: parse_rng ?defines context rest
10889        | _ ->
10890            failwithf "%s: <zeroOrMore> contains more than one child element"
10891              context
10892       )
10893   | Xml.Element ("oneOrMore", [], [child]) :: rest ->
10894       let rng = parse_rng ?defines context [child] in
10895       (match rng with
10896        | [child] -> OneOrMore child :: parse_rng ?defines context rest
10897        | _ ->
10898            failwithf "%s: <oneOrMore> contains more than one child element"
10899              context
10900       )
10901   | Xml.Element ("optional", [], [child]) :: rest ->
10902       let rng = parse_rng ?defines context [child] in
10903       (match rng with
10904        | [child] -> Optional child :: parse_rng ?defines context rest
10905        | _ ->
10906            failwithf "%s: <optional> contains more than one child element"
10907              context
10908       )
10909   | Xml.Element ("choice", [], children) :: rest ->
10910       let values = List.map (
10911         function Xml.Element ("value", [], [Xml.PCData value]) -> value
10912         | _ ->
10913             failwithf "%s: can't handle anything except <value> in <choice>"
10914               context
10915       ) children in
10916       Choice values
10917       :: parse_rng ?defines context rest
10918   | Xml.Element ("value", [], [Xml.PCData value]) :: rest ->
10919       Value value :: parse_rng ?defines context rest
10920   | Xml.Element ("text", [], []) :: rest ->
10921       Text :: parse_rng ?defines context rest
10922   | Xml.Element ("ref", ["name", name], []) :: rest ->
10923       (* Look up the reference.  Because of limitations in this parser,
10924        * we can't handle arbitrarily nested <ref> yet.  You can only
10925        * use <ref> from inside <start>.
10926        *)
10927       (match defines with
10928        | None ->
10929            failwithf "%s: contains <ref>, but no refs are defined yet" context
10930        | Some map ->
10931            let rng = StringMap.find name map in
10932            rng @ parse_rng ?defines context rest
10933       )
10934   | x :: _ ->
10935       failwithf "%s: can't handle '%s' in schema" context (Xml.to_string x)
10936
10937 let grammar =
10938   let xml = Xml.parse_file rng_input in
10939   match xml with
10940   | Xml.Element ("grammar", _,
10941                  Xml.Element ("start", _, gram) :: defines) ->
10942       (* The <define/> elements are referenced in the <start> section,
10943        * so build a map of those first.
10944        *)
10945       let defines = List.fold_left (
10946         fun map ->
10947           function Xml.Element ("define", ["name", name], defn) ->
10948             StringMap.add name defn map
10949           | _ ->
10950               failwithf "%s: expected <define name=name/>" rng_input
10951       ) StringMap.empty defines in
10952       let defines = StringMap.mapi parse_rng defines in
10953
10954       (* Parse the <start> clause, passing the defines. *)
10955       parse_rng ~defines "<start>" gram
10956   | _ ->
10957       failwithf "%s: input is not <grammar><start/><define>*</grammar>"
10958         rng_input
10959
10960 let name_of_field = function
10961   | Element (name, _) | Attribute (name, _)
10962   | ZeroOrMore (Element (name, _))
10963   | OneOrMore (Element (name, _))
10964   | Optional (Element (name, _)) -> name
10965   | Optional (Attribute (name, _)) -> name
10966   | Text -> (* an unnamed field in an element *)
10967       "data"
10968   | rng ->
10969       failwithf "name_of_field failed at: %s" (string_of_rng rng)
10970
10971 (* At the moment this function only generates OCaml types.  However we
10972  * should parameterize it later so it can generate types/structs in a
10973  * variety of languages.
10974  *)
10975 let generate_types xs =
10976   (* A simple type is one that can be printed out directly, eg.
10977    * "string option".  A complex type is one which has a name and has
10978    * to be defined via another toplevel definition, eg. a struct.
10979    *
10980    * generate_type generates code for either simple or complex types.
10981    * In the simple case, it returns the string ("string option").  In
10982    * the complex case, it returns the name ("mountpoint").  In the
10983    * complex case it has to print out the definition before returning,
10984    * so it should only be called when we are at the beginning of a
10985    * new line (BOL context).
10986    *)
10987   let rec generate_type = function
10988     | Text ->                                (* string *)
10989         "string", true
10990     | Choice values ->                        (* [`val1|`val2|...] *)
10991         "[" ^ String.concat "|" (List.map ((^)"`") values) ^ "]", true
10992     | ZeroOrMore rng ->                        (* <rng> list *)
10993         let t, is_simple = generate_type rng in
10994         t ^ " list (* 0 or more *)", is_simple
10995     | OneOrMore rng ->                        (* <rng> list *)
10996         let t, is_simple = generate_type rng in
10997         t ^ " list (* 1 or more *)", is_simple
10998                                         (* virt-inspector hack: bool *)
10999     | Optional (Attribute (name, [Value "1"])) ->
11000         "bool", true
11001     | Optional rng ->                        (* <rng> list *)
11002         let t, is_simple = generate_type rng in
11003         t ^ " option", is_simple
11004                                         (* type name = { fields ... } *)
11005     | Element (name, fields) when is_attrs_interleave fields ->
11006         generate_type_struct name (get_attrs_interleave fields)
11007     | Element (name, [field])                (* type name = field *)
11008     | Attribute (name, [field]) ->
11009         let t, is_simple = generate_type field in
11010         if is_simple then (t, true)
11011         else (
11012           pr "type %s = %s\n" name t;
11013           name, false
11014         )
11015     | Element (name, fields) ->              (* type name = { fields ... } *)
11016         generate_type_struct name fields
11017     | rng ->
11018         failwithf "generate_type failed at: %s" (string_of_rng rng)
11019
11020   and is_attrs_interleave = function
11021     | [Interleave _] -> true
11022     | Attribute _ :: fields -> is_attrs_interleave fields
11023     | Optional (Attribute _) :: fields -> is_attrs_interleave fields
11024     | _ -> false
11025
11026   and get_attrs_interleave = function
11027     | [Interleave fields] -> fields
11028     | ((Attribute _) as field) :: fields
11029     | ((Optional (Attribute _)) as field) :: fields ->
11030         field :: get_attrs_interleave fields
11031     | _ -> assert false
11032
11033   and generate_types xs =
11034     List.iter (fun x -> ignore (generate_type x)) xs
11035
11036   and generate_type_struct name fields =
11037     (* Calculate the types of the fields first.  We have to do this
11038      * before printing anything so we are still in BOL context.
11039      *)
11040     let types = List.map fst (List.map generate_type fields) in
11041
11042     (* Special case of a struct containing just a string and another
11043      * field.  Turn it into an assoc list.
11044      *)
11045     match types with
11046     | ["string"; other] ->
11047         let fname1, fname2 =
11048           match fields with
11049           | [f1; f2] -> name_of_field f1, name_of_field f2
11050           | _ -> assert false in
11051         pr "type %s = string * %s (* %s -> %s *)\n" name other fname1 fname2;
11052         name, false
11053
11054     | types ->
11055         pr "type %s = {\n" name;
11056         List.iter (
11057           fun (field, ftype) ->
11058             let fname = name_of_field field in
11059             pr "  %s_%s : %s;\n" name fname ftype
11060         ) (List.combine fields types);
11061         pr "}\n";
11062         (* Return the name of this type, and
11063          * false because it's not a simple type.
11064          *)
11065         name, false
11066   in
11067
11068   generate_types xs
11069
11070 let generate_parsers xs =
11071   (* As for generate_type above, generate_parser makes a parser for
11072    * some type, and returns the name of the parser it has generated.
11073    * Because it (may) need to print something, it should always be
11074    * called in BOL context.
11075    *)
11076   let rec generate_parser = function
11077     | Text ->                                (* string *)
11078         "string_child_or_empty"
11079     | Choice values ->                        (* [`val1|`val2|...] *)
11080         sprintf "(fun x -> match Xml.pcdata (first_child x) with %s | str -> failwith (\"unexpected field value: \" ^ str))"
11081           (String.concat "|"
11082              (List.map (fun v -> sprintf "%S -> `%s" v v) values))
11083     | ZeroOrMore rng ->                        (* <rng> list *)
11084         let pa = generate_parser rng in
11085         sprintf "(fun x -> List.map %s (Xml.children x))" pa
11086     | OneOrMore rng ->                        (* <rng> list *)
11087         let pa = generate_parser rng in
11088         sprintf "(fun x -> List.map %s (Xml.children x))" pa
11089                                         (* virt-inspector hack: bool *)
11090     | Optional (Attribute (name, [Value "1"])) ->
11091         sprintf "(fun x -> try ignore (Xml.attrib x %S); true with Xml.No_attribute _ -> false)" name
11092     | Optional rng ->                        (* <rng> list *)
11093         let pa = generate_parser rng in
11094         sprintf "(function None -> None | Some x -> Some (%s x))" pa
11095                                         (* type name = { fields ... } *)
11096     | Element (name, fields) when is_attrs_interleave fields ->
11097         generate_parser_struct name (get_attrs_interleave fields)
11098     | Element (name, [field]) ->        (* type name = field *)
11099         let pa = generate_parser field in
11100         let parser_name = sprintf "parse_%s_%d" name (unique ()) in
11101         pr "let %s =\n" parser_name;
11102         pr "  %s\n" pa;
11103         pr "let parse_%s = %s\n" name parser_name;
11104         parser_name
11105     | Attribute (name, [field]) ->
11106         let pa = generate_parser field in
11107         let parser_name = sprintf "parse_%s_%d" name (unique ()) in
11108         pr "let %s =\n" parser_name;
11109         pr "  %s\n" pa;
11110         pr "let parse_%s = %s\n" name parser_name;
11111         parser_name
11112     | Element (name, fields) ->              (* type name = { fields ... } *)
11113         generate_parser_struct name ([], fields)
11114     | rng ->
11115         failwithf "generate_parser failed at: %s" (string_of_rng rng)
11116
11117   and is_attrs_interleave = function
11118     | [Interleave _] -> true
11119     | Attribute _ :: fields -> is_attrs_interleave fields
11120     | Optional (Attribute _) :: fields -> is_attrs_interleave fields
11121     | _ -> false
11122
11123   and get_attrs_interleave = function
11124     | [Interleave fields] -> [], fields
11125     | ((Attribute _) as field) :: fields
11126     | ((Optional (Attribute _)) as field) :: fields ->
11127         let attrs, interleaves = get_attrs_interleave fields in
11128         (field :: attrs), interleaves
11129     | _ -> assert false
11130
11131   and generate_parsers xs =
11132     List.iter (fun x -> ignore (generate_parser x)) xs
11133
11134   and generate_parser_struct name (attrs, interleaves) =
11135     (* Generate parsers for the fields first.  We have to do this
11136      * before printing anything so we are still in BOL context.
11137      *)
11138     let fields = attrs @ interleaves in
11139     let pas = List.map generate_parser fields in
11140
11141     (* Generate an intermediate tuple from all the fields first.
11142      * If the type is just a string + another field, then we will
11143      * return this directly, otherwise it is turned into a record.
11144      *
11145      * RELAX NG note: This code treats <interleave> and plain lists of
11146      * fields the same.  In other words, it doesn't bother enforcing
11147      * any ordering of fields in the XML.
11148      *)
11149     pr "let parse_%s x =\n" name;
11150     pr "  let t = (\n    ";
11151     let comma = ref false in
11152     List.iter (
11153       fun x ->
11154         if !comma then pr ",\n    ";
11155         comma := true;
11156         match x with
11157         | Optional (Attribute (fname, [field])), pa ->
11158             pr "%s x" pa
11159         | Optional (Element (fname, [field])), pa ->
11160             pr "%s (optional_child %S x)" pa fname
11161         | Attribute (fname, [Text]), _ ->
11162             pr "attribute %S x" fname
11163         | (ZeroOrMore _ | OneOrMore _), pa ->
11164             pr "%s x" pa
11165         | Text, pa ->
11166             pr "%s x" pa
11167         | (field, pa) ->
11168             let fname = name_of_field field in
11169             pr "%s (child %S x)" pa fname
11170     ) (List.combine fields pas);
11171     pr "\n  ) in\n";
11172
11173     (match fields with
11174      | [Element (_, [Text]) | Attribute (_, [Text]); _] ->
11175          pr "  t\n"
11176
11177      | _ ->
11178          pr "  (Obj.magic t : %s)\n" name
11179 (*
11180          List.iter (
11181            function
11182            | (Optional (Attribute (fname, [field])), pa) ->
11183                pr "  %s_%s =\n" name fname;
11184                pr "    %s x;\n" pa
11185            | (Optional (Element (fname, [field])), pa) ->
11186                pr "  %s_%s =\n" name fname;
11187                pr "    (let x = optional_child %S x in\n" fname;
11188                pr "     %s x);\n" pa
11189            | (field, pa) ->
11190                let fname = name_of_field field in
11191                pr "  %s_%s =\n" name fname;
11192                pr "    (let x = child %S x in\n" fname;
11193                pr "     %s x);\n" pa
11194          ) (List.combine fields pas);
11195          pr "}\n"
11196 *)
11197     );
11198     sprintf "parse_%s" name
11199   in
11200
11201   generate_parsers xs
11202
11203 (* Generate ocaml/guestfs_inspector.mli. *)
11204 let generate_ocaml_inspector_mli () =
11205   generate_header ~extra_inputs:[rng_input] OCamlStyle LGPLv2plus;
11206
11207   pr "\
11208 (** This is an OCaml language binding to the external [virt-inspector]
11209     program.
11210
11211     For more information, please read the man page [virt-inspector(1)].
11212 *)
11213
11214 ";
11215
11216   generate_types grammar;
11217   pr "(** The nested information returned from the {!inspect} function. *)\n";
11218   pr "\n";
11219
11220   pr "\
11221 val inspect : ?connect:string -> ?xml:string -> string list -> operatingsystems
11222 (** To inspect a libvirt domain called [name], pass a singleton
11223     list: [inspect [name]].  When using libvirt only, you may
11224     optionally pass a libvirt URI using [inspect ~connect:uri ...].
11225
11226     To inspect a disk image or images, pass a list of the filenames
11227     of the disk images: [inspect filenames]
11228
11229     This function inspects the given guest or disk images and
11230     returns a list of operating system(s) found and a large amount
11231     of information about them.  In the vast majority of cases,
11232     a virtual machine only contains a single operating system.
11233
11234     If the optional [~xml] parameter is given, then this function
11235     skips running the external virt-inspector program and just
11236     parses the given XML directly (which is expected to be XML
11237     produced from a previous run of virt-inspector).  The list of
11238     names and connect URI are ignored in this case.
11239
11240     This function can throw a wide variety of exceptions, for example
11241     if the external virt-inspector program cannot be found, or if
11242     it doesn't generate valid XML.
11243 *)
11244 "
11245
11246 (* Generate ocaml/guestfs_inspector.ml. *)
11247 let generate_ocaml_inspector_ml () =
11248   generate_header ~extra_inputs:[rng_input] OCamlStyle LGPLv2plus;
11249
11250   pr "open Unix\n";
11251   pr "\n";
11252
11253   generate_types grammar;
11254   pr "\n";
11255
11256   pr "\
11257 (* Misc functions which are used by the parser code below. *)
11258 let first_child = function
11259   | Xml.Element (_, _, c::_) -> c
11260   | Xml.Element (name, _, []) ->
11261       failwith (\"expected <\" ^ name ^ \"/> to have a child node\")
11262   | Xml.PCData str ->
11263       failwith (\"expected XML tag, but read PCDATA '\" ^ str ^ \"' instead\")
11264
11265 let string_child_or_empty = function
11266   | Xml.Element (_, _, [Xml.PCData s]) -> s
11267   | Xml.Element (_, _, []) -> \"\"
11268   | Xml.Element (x, _, _) ->
11269       failwith (\"expected XML tag with a single PCDATA child, but got \" ^
11270                 x ^ \" instead\")
11271   | Xml.PCData str ->
11272       failwith (\"expected XML tag, but read PCDATA '\" ^ str ^ \"' instead\")
11273
11274 let optional_child name xml =
11275   let children = Xml.children xml in
11276   try
11277     Some (List.find (function
11278                      | Xml.Element (n, _, _) when n = name -> true
11279                      | _ -> false) children)
11280   with
11281     Not_found -> None
11282
11283 let child name xml =
11284   match optional_child name xml with
11285   | Some c -> c
11286   | None ->
11287       failwith (\"mandatory field <\" ^ name ^ \"/> missing in XML output\")
11288
11289 let attribute name xml =
11290   try Xml.attrib xml name
11291   with Xml.No_attribute _ ->
11292     failwith (\"mandatory attribute \" ^ name ^ \" missing in XML output\")
11293
11294 ";
11295
11296   generate_parsers grammar;
11297   pr "\n";
11298
11299   pr "\
11300 (* Run external virt-inspector, then use parser to parse the XML. *)
11301 let inspect ?connect ?xml names =
11302   let xml =
11303     match xml with
11304     | None ->
11305         if names = [] then invalid_arg \"inspect: no names given\";
11306         let cmd = [ \"virt-inspector\"; \"--xml\" ] @
11307           (match connect with None -> [] | Some uri -> [ \"--connect\"; uri ]) @
11308           names in
11309         let cmd = List.map Filename.quote cmd in
11310         let cmd = String.concat \" \" cmd in
11311         let chan = open_process_in cmd in
11312         let xml = Xml.parse_in chan in
11313         (match close_process_in chan with
11314          | WEXITED 0 -> ()
11315          | WEXITED _ -> failwith \"external virt-inspector command failed\"
11316          | WSIGNALED i | WSTOPPED i ->
11317              failwith (\"external virt-inspector command died or stopped on sig \" ^
11318                        string_of_int i)
11319         );
11320         xml
11321     | Some doc ->
11322         Xml.parse_string doc in
11323   parse_operatingsystems xml
11324 "
11325
11326 (* This is used to generate the src/MAX_PROC_NR file which
11327  * contains the maximum procedure number, a surrogate for the
11328  * ABI version number.  See src/Makefile.am for the details.
11329  *)
11330 and generate_max_proc_nr () =
11331   let proc_nrs = List.map (
11332     fun (_, _, proc_nr, _, _, _, _) -> proc_nr
11333   ) daemon_functions in
11334
11335   let max_proc_nr = List.fold_left max 0 proc_nrs in
11336
11337   pr "%d\n" max_proc_nr
11338
11339 let output_to filename k =
11340   let filename_new = filename ^ ".new" in
11341   chan := open_out filename_new;
11342   k ();
11343   close_out !chan;
11344   chan := Pervasives.stdout;
11345
11346   (* Is the new file different from the current file? *)
11347   if Sys.file_exists filename && files_equal filename filename_new then
11348     unlink filename_new                 (* same, so skip it *)
11349   else (
11350     (* different, overwrite old one *)
11351     (try chmod filename 0o644 with Unix_error _ -> ());
11352     rename filename_new filename;
11353     chmod filename 0o444;
11354     printf "written %s\n%!" filename;
11355   )
11356
11357 let perror msg = function
11358   | Unix_error (err, _, _) ->
11359       eprintf "%s: %s\n" msg (error_message err)
11360   | exn ->
11361       eprintf "%s: %s\n" msg (Printexc.to_string exn)
11362
11363 (* Main program. *)
11364 let () =
11365   let lock_fd =
11366     try openfile "HACKING" [O_RDWR] 0
11367     with
11368     | Unix_error (ENOENT, _, _) ->
11369         eprintf "\
11370 You are probably running this from the wrong directory.
11371 Run it from the top source directory using the command
11372   src/generator.ml
11373 ";
11374         exit 1
11375     | exn ->
11376         perror "open: HACKING" exn;
11377         exit 1 in
11378
11379   (* Acquire a lock so parallel builds won't try to run the generator
11380    * twice at the same time.  Subsequent builds will wait for the first
11381    * one to finish.  Note the lock is released implicitly when the
11382    * program exits.
11383    *)
11384   (try lockf lock_fd F_LOCK 1
11385    with exn ->
11386      perror "lock: HACKING" exn;
11387      exit 1);
11388
11389   check_functions ();
11390
11391   output_to "src/guestfs_protocol.x" generate_xdr;
11392   output_to "src/guestfs-structs.h" generate_structs_h;
11393   output_to "src/guestfs-actions.h" generate_actions_h;
11394   output_to "src/guestfs-internal-actions.h" generate_internal_actions_h;
11395   output_to "src/guestfs-actions.c" generate_client_actions;
11396   output_to "src/guestfs-bindtests.c" generate_bindtests;
11397   output_to "src/guestfs-structs.pod" generate_structs_pod;
11398   output_to "src/guestfs-actions.pod" generate_actions_pod;
11399   output_to "src/guestfs-availability.pod" generate_availability_pod;
11400   output_to "src/MAX_PROC_NR" generate_max_proc_nr;
11401   output_to "src/libguestfs.syms" generate_linker_script;
11402   output_to "daemon/actions.h" generate_daemon_actions_h;
11403   output_to "daemon/stubs.c" generate_daemon_actions;
11404   output_to "daemon/names.c" generate_daemon_names;
11405   output_to "daemon/optgroups.c" generate_daemon_optgroups_c;
11406   output_to "daemon/optgroups.h" generate_daemon_optgroups_h;
11407   output_to "capitests/tests.c" generate_tests;
11408   output_to "fish/cmds.c" generate_fish_cmds;
11409   output_to "fish/completion.c" generate_fish_completion;
11410   output_to "fish/guestfish-actions.pod" generate_fish_actions_pod;
11411   output_to "ocaml/guestfs.mli" generate_ocaml_mli;
11412   output_to "ocaml/guestfs.ml" generate_ocaml_ml;
11413   output_to "ocaml/guestfs_c_actions.c" generate_ocaml_c;
11414   output_to "ocaml/bindtests.ml" generate_ocaml_bindtests;
11415   output_to "ocaml/guestfs_inspector.mli" generate_ocaml_inspector_mli;
11416   output_to "ocaml/guestfs_inspector.ml" generate_ocaml_inspector_ml;
11417   output_to "perl/Guestfs.xs" generate_perl_xs;
11418   output_to "perl/lib/Sys/Guestfs.pm" generate_perl_pm;
11419   output_to "perl/bindtests.pl" generate_perl_bindtests;
11420   output_to "python/guestfs-py.c" generate_python_c;
11421   output_to "python/guestfs.py" generate_python_py;
11422   output_to "python/bindtests.py" generate_python_bindtests;
11423   output_to "ruby/ext/guestfs/_guestfs.c" generate_ruby_c;
11424   output_to "ruby/bindtests.rb" generate_ruby_bindtests;
11425   output_to "java/com/redhat/et/libguestfs/GuestFS.java" generate_java_java;
11426
11427   List.iter (
11428     fun (typ, jtyp) ->
11429       let cols = cols_of_struct typ in
11430       let filename = sprintf "java/com/redhat/et/libguestfs/%s.java" jtyp in
11431       output_to filename (generate_java_struct jtyp cols);
11432   ) java_structs;
11433
11434   output_to "java/Makefile.inc" generate_java_makefile_inc;
11435   output_to "java/com_redhat_et_libguestfs_GuestFS.c" generate_java_c;
11436   output_to "java/Bindtests.java" generate_java_bindtests;
11437   output_to "haskell/Guestfs.hs" generate_haskell_hs;
11438   output_to "haskell/Bindtests.hs" generate_haskell_bindtests;
11439   output_to "csharp/Libguestfs.cs" generate_csharp;
11440
11441   (* Always generate this file last, and unconditionally.  It's used
11442    * by the Makefile to know when we must re-run the generator.
11443    *)
11444   let chan = open_out "src/stamp-generator" in
11445   fprintf chan "1\n";
11446   close_out chan;
11447
11448   printf "generated %d lines of code\n" !lines