416a66fd96b1c106e6c541d122c0cfe19ec4e6b2
[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_zero_device>, 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   ("zero_device", (RErr, [Device "device"]), 228, [DangerWillRobinson],
4350    [InitBasicFSonLVM, Always, TestRun (
4351       [["zero_device"; "/dev/VG/LV"]])],
4352    "write zeroes to an entire device",
4353    "\
4354 This command writes zeroes over the entire C<device>.  Compare
4355 with C<guestfs_zero> which just zeroes the first few blocks of
4356 a device.");
4357
4358 ]
4359
4360 let all_functions = non_daemon_functions @ daemon_functions
4361
4362 (* In some places we want the functions to be displayed sorted
4363  * alphabetically, so this is useful:
4364  *)
4365 let all_functions_sorted =
4366   List.sort (fun (n1,_,_,_,_,_,_) (n2,_,_,_,_,_,_) ->
4367                compare n1 n2) all_functions
4368
4369 (* Field types for structures. *)
4370 type field =
4371   | FChar                       (* C 'char' (really, a 7 bit byte). *)
4372   | FString                     (* nul-terminated ASCII string, NOT NULL. *)
4373   | FBuffer                     (* opaque buffer of bytes, (char *, int) pair *)
4374   | FUInt32
4375   | FInt32
4376   | FUInt64
4377   | FInt64
4378   | FBytes                      (* Any int measure that counts bytes. *)
4379   | FUUID                       (* 32 bytes long, NOT nul-terminated. *)
4380   | FOptPercent                 (* [0..100], or -1 meaning "not present". *)
4381
4382 (* Because we generate extra parsing code for LVM command line tools,
4383  * we have to pull out the LVM columns separately here.
4384  *)
4385 let lvm_pv_cols = [
4386   "pv_name", FString;
4387   "pv_uuid", FUUID;
4388   "pv_fmt", FString;
4389   "pv_size", FBytes;
4390   "dev_size", FBytes;
4391   "pv_free", FBytes;
4392   "pv_used", FBytes;
4393   "pv_attr", FString (* XXX *);
4394   "pv_pe_count", FInt64;
4395   "pv_pe_alloc_count", FInt64;
4396   "pv_tags", FString;
4397   "pe_start", FBytes;
4398   "pv_mda_count", FInt64;
4399   "pv_mda_free", FBytes;
4400   (* Not in Fedora 10:
4401      "pv_mda_size", FBytes;
4402   *)
4403 ]
4404 let lvm_vg_cols = [
4405   "vg_name", FString;
4406   "vg_uuid", FUUID;
4407   "vg_fmt", FString;
4408   "vg_attr", FString (* XXX *);
4409   "vg_size", FBytes;
4410   "vg_free", FBytes;
4411   "vg_sysid", FString;
4412   "vg_extent_size", FBytes;
4413   "vg_extent_count", FInt64;
4414   "vg_free_count", FInt64;
4415   "max_lv", FInt64;
4416   "max_pv", FInt64;
4417   "pv_count", FInt64;
4418   "lv_count", FInt64;
4419   "snap_count", FInt64;
4420   "vg_seqno", FInt64;
4421   "vg_tags", FString;
4422   "vg_mda_count", FInt64;
4423   "vg_mda_free", FBytes;
4424   (* Not in Fedora 10:
4425      "vg_mda_size", FBytes;
4426   *)
4427 ]
4428 let lvm_lv_cols = [
4429   "lv_name", FString;
4430   "lv_uuid", FUUID;
4431   "lv_attr", FString (* XXX *);
4432   "lv_major", FInt64;
4433   "lv_minor", FInt64;
4434   "lv_kernel_major", FInt64;
4435   "lv_kernel_minor", FInt64;
4436   "lv_size", FBytes;
4437   "seg_count", FInt64;
4438   "origin", FString;
4439   "snap_percent", FOptPercent;
4440   "copy_percent", FOptPercent;
4441   "move_pv", FString;
4442   "lv_tags", FString;
4443   "mirror_log", FString;
4444   "modules", FString;
4445 ]
4446
4447 (* Names and fields in all structures (in RStruct and RStructList)
4448  * that we support.
4449  *)
4450 let structs = [
4451   (* The old RIntBool return type, only ever used for aug_defnode.  Do
4452    * not use this struct in any new code.
4453    *)
4454   "int_bool", [
4455     "i", FInt32;                (* for historical compatibility *)
4456     "b", FInt32;                (* for historical compatibility *)
4457   ];
4458
4459   (* LVM PVs, VGs, LVs. *)
4460   "lvm_pv", lvm_pv_cols;
4461   "lvm_vg", lvm_vg_cols;
4462   "lvm_lv", lvm_lv_cols;
4463
4464   (* Column names and types from stat structures.
4465    * NB. Can't use things like 'st_atime' because glibc header files
4466    * define some of these as macros.  Ugh.
4467    *)
4468   "stat", [
4469     "dev", FInt64;
4470     "ino", FInt64;
4471     "mode", FInt64;
4472     "nlink", FInt64;
4473     "uid", FInt64;
4474     "gid", FInt64;
4475     "rdev", FInt64;
4476     "size", FInt64;
4477     "blksize", FInt64;
4478     "blocks", FInt64;
4479     "atime", FInt64;
4480     "mtime", FInt64;
4481     "ctime", FInt64;
4482   ];
4483   "statvfs", [
4484     "bsize", FInt64;
4485     "frsize", FInt64;
4486     "blocks", FInt64;
4487     "bfree", FInt64;
4488     "bavail", FInt64;
4489     "files", FInt64;
4490     "ffree", FInt64;
4491     "favail", FInt64;
4492     "fsid", FInt64;
4493     "flag", FInt64;
4494     "namemax", FInt64;
4495   ];
4496
4497   (* Column names in dirent structure. *)
4498   "dirent", [
4499     "ino", FInt64;
4500     (* 'b' 'c' 'd' 'f' (FIFO) 'l' 'r' (regular file) 's' 'u' '?' *)
4501     "ftyp", FChar;
4502     "name", FString;
4503   ];
4504
4505   (* Version numbers. *)
4506   "version", [
4507     "major", FInt64;
4508     "minor", FInt64;
4509     "release", FInt64;
4510     "extra", FString;
4511   ];
4512
4513   (* Extended attribute. *)
4514   "xattr", [
4515     "attrname", FString;
4516     "attrval", FBuffer;
4517   ];
4518
4519   (* Inotify events. *)
4520   "inotify_event", [
4521     "in_wd", FInt64;
4522     "in_mask", FUInt32;
4523     "in_cookie", FUInt32;
4524     "in_name", FString;
4525   ];
4526
4527   (* Partition table entry. *)
4528   "partition", [
4529     "part_num", FInt32;
4530     "part_start", FBytes;
4531     "part_end", FBytes;
4532     "part_size", FBytes;
4533   ];
4534 ] (* end of structs *)
4535
4536 (* Ugh, Java has to be different ..
4537  * These names are also used by the Haskell bindings.
4538  *)
4539 let java_structs = [
4540   "int_bool", "IntBool";
4541   "lvm_pv", "PV";
4542   "lvm_vg", "VG";
4543   "lvm_lv", "LV";
4544   "stat", "Stat";
4545   "statvfs", "StatVFS";
4546   "dirent", "Dirent";
4547   "version", "Version";
4548   "xattr", "XAttr";
4549   "inotify_event", "INotifyEvent";
4550   "partition", "Partition";
4551 ]
4552
4553 (* What structs are actually returned. *)
4554 type rstructs_used_t = RStructOnly | RStructListOnly | RStructAndList
4555
4556 (* Returns a list of RStruct/RStructList structs that are returned
4557  * by any function.  Each element of returned list is a pair:
4558  *
4559  * (structname, RStructOnly)
4560  *    == there exists function which returns RStruct (_, structname)
4561  * (structname, RStructListOnly)
4562  *    == there exists function which returns RStructList (_, structname)
4563  * (structname, RStructAndList)
4564  *    == there are functions returning both RStruct (_, structname)
4565  *                                      and RStructList (_, structname)
4566  *)
4567 let rstructs_used_by functions =
4568   (* ||| is a "logical OR" for rstructs_used_t *)
4569   let (|||) a b =
4570     match a, b with
4571     | RStructAndList, _
4572     | _, RStructAndList -> RStructAndList
4573     | RStructOnly, RStructListOnly
4574     | RStructListOnly, RStructOnly -> RStructAndList
4575     | RStructOnly, RStructOnly -> RStructOnly
4576     | RStructListOnly, RStructListOnly -> RStructListOnly
4577   in
4578
4579   let h = Hashtbl.create 13 in
4580
4581   (* if elem->oldv exists, update entry using ||| operator,
4582    * else just add elem->newv to the hash
4583    *)
4584   let update elem newv =
4585     try  let oldv = Hashtbl.find h elem in
4586          Hashtbl.replace h elem (newv ||| oldv)
4587     with Not_found -> Hashtbl.add h elem newv
4588   in
4589
4590   List.iter (
4591     fun (_, style, _, _, _, _, _) ->
4592       match fst style with
4593       | RStruct (_, structname) -> update structname RStructOnly
4594       | RStructList (_, structname) -> update structname RStructListOnly
4595       | _ -> ()
4596   ) functions;
4597
4598   (* return key->values as a list of (key,value) *)
4599   Hashtbl.fold (fun key value xs -> (key, value) :: xs) h []
4600
4601 (* Used for testing language bindings. *)
4602 type callt =
4603   | CallString of string
4604   | CallOptString of string option
4605   | CallStringList of string list
4606   | CallInt of int
4607   | CallInt64 of int64
4608   | CallBool of bool
4609
4610 (* Used to memoize the result of pod2text. *)
4611 let pod2text_memo_filename = "src/.pod2text.data"
4612 let pod2text_memo : ((int * string * string), string list) Hashtbl.t =
4613   try
4614     let chan = open_in pod2text_memo_filename in
4615     let v = input_value chan in
4616     close_in chan;
4617     v
4618   with
4619     _ -> Hashtbl.create 13
4620 let pod2text_memo_updated () =
4621   let chan = open_out pod2text_memo_filename in
4622   output_value chan pod2text_memo;
4623   close_out chan
4624
4625 (* Useful functions.
4626  * Note we don't want to use any external OCaml libraries which
4627  * makes this a bit harder than it should be.
4628  *)
4629 module StringMap = Map.Make (String)
4630
4631 let failwithf fs = ksprintf failwith fs
4632
4633 let unique = let i = ref 0 in fun () -> incr i; !i
4634
4635 let replace_char s c1 c2 =
4636   let s2 = String.copy s in
4637   let r = ref false in
4638   for i = 0 to String.length s2 - 1 do
4639     if String.unsafe_get s2 i = c1 then (
4640       String.unsafe_set s2 i c2;
4641       r := true
4642     )
4643   done;
4644   if not !r then s else s2
4645
4646 let isspace c =
4647   c = ' '
4648   (* || c = '\f' *) || c = '\n' || c = '\r' || c = '\t' (* || c = '\v' *)
4649
4650 let triml ?(test = isspace) str =
4651   let i = ref 0 in
4652   let n = ref (String.length str) in
4653   while !n > 0 && test str.[!i]; do
4654     decr n;
4655     incr i
4656   done;
4657   if !i = 0 then str
4658   else String.sub str !i !n
4659
4660 let trimr ?(test = isspace) str =
4661   let n = ref (String.length str) in
4662   while !n > 0 && test str.[!n-1]; do
4663     decr n
4664   done;
4665   if !n = String.length str then str
4666   else String.sub str 0 !n
4667
4668 let trim ?(test = isspace) str =
4669   trimr ~test (triml ~test str)
4670
4671 let rec find s sub =
4672   let len = String.length s in
4673   let sublen = String.length sub in
4674   let rec loop i =
4675     if i <= len-sublen then (
4676       let rec loop2 j =
4677         if j < sublen then (
4678           if s.[i+j] = sub.[j] then loop2 (j+1)
4679           else -1
4680         ) else
4681           i (* found *)
4682       in
4683       let r = loop2 0 in
4684       if r = -1 then loop (i+1) else r
4685     ) else
4686       -1 (* not found *)
4687   in
4688   loop 0
4689
4690 let rec replace_str s s1 s2 =
4691   let len = String.length s in
4692   let sublen = String.length s1 in
4693   let i = find s s1 in
4694   if i = -1 then s
4695   else (
4696     let s' = String.sub s 0 i in
4697     let s'' = String.sub s (i+sublen) (len-i-sublen) in
4698     s' ^ s2 ^ replace_str s'' s1 s2
4699   )
4700
4701 let rec string_split sep str =
4702   let len = String.length str in
4703   let seplen = String.length sep in
4704   let i = find str sep in
4705   if i = -1 then [str]
4706   else (
4707     let s' = String.sub str 0 i in
4708     let s'' = String.sub str (i+seplen) (len-i-seplen) in
4709     s' :: string_split sep s''
4710   )
4711
4712 let files_equal n1 n2 =
4713   let cmd = sprintf "cmp -s %s %s" (Filename.quote n1) (Filename.quote n2) in
4714   match Sys.command cmd with
4715   | 0 -> true
4716   | 1 -> false
4717   | i -> failwithf "%s: failed with error code %d" cmd i
4718
4719 let rec filter_map f = function
4720   | [] -> []
4721   | x :: xs ->
4722       match f x with
4723       | Some y -> y :: filter_map f xs
4724       | None -> filter_map f xs
4725
4726 let rec find_map f = function
4727   | [] -> raise Not_found
4728   | x :: xs ->
4729       match f x with
4730       | Some y -> y
4731       | None -> find_map f xs
4732
4733 let iteri f xs =
4734   let rec loop i = function
4735     | [] -> ()
4736     | x :: xs -> f i x; loop (i+1) xs
4737   in
4738   loop 0 xs
4739
4740 let mapi f xs =
4741   let rec loop i = function
4742     | [] -> []
4743     | x :: xs -> let r = f i x in r :: loop (i+1) xs
4744   in
4745   loop 0 xs
4746
4747 let count_chars c str =
4748   let count = ref 0 in
4749   for i = 0 to String.length str - 1 do
4750     if c = String.unsafe_get str i then incr count
4751   done;
4752   !count
4753
4754 let name_of_argt = function
4755   | Pathname n | Device n | Dev_or_Path n | String n | OptString n
4756   | StringList n | DeviceList n | Bool n | Int n | Int64 n
4757   | FileIn n | FileOut n -> n
4758
4759 let java_name_of_struct typ =
4760   try List.assoc typ java_structs
4761   with Not_found ->
4762     failwithf
4763       "java_name_of_struct: no java_structs entry corresponding to %s" typ
4764
4765 let cols_of_struct typ =
4766   try List.assoc typ structs
4767   with Not_found ->
4768     failwithf "cols_of_struct: unknown struct %s" typ
4769
4770 let seq_of_test = function
4771   | TestRun s | TestOutput (s, _) | TestOutputList (s, _)
4772   | TestOutputListOfDevices (s, _)
4773   | TestOutputInt (s, _) | TestOutputIntOp (s, _, _)
4774   | TestOutputTrue s | TestOutputFalse s
4775   | TestOutputLength (s, _) | TestOutputBuffer (s, _)
4776   | TestOutputStruct (s, _)
4777   | TestLastFail s -> s
4778
4779 (* Handling for function flags. *)
4780 let protocol_limit_warning =
4781   "Because of the message protocol, there is a transfer limit
4782 of somewhere between 2MB and 4MB.  See L<guestfs(3)/PROTOCOL LIMITS>."
4783
4784 let danger_will_robinson =
4785   "B<This command is dangerous.  Without careful use you
4786 can easily destroy all your data>."
4787
4788 let deprecation_notice flags =
4789   try
4790     let alt =
4791       find_map (function DeprecatedBy str -> Some str | _ -> None) flags in
4792     let txt =
4793       sprintf "This function is deprecated.
4794 In new code, use the C<%s> call instead.
4795
4796 Deprecated functions will not be removed from the API, but the
4797 fact that they are deprecated indicates that there are problems
4798 with correct use of these functions." alt in
4799     Some txt
4800   with
4801     Not_found -> None
4802
4803 (* Create list of optional groups. *)
4804 let optgroups =
4805   let h = Hashtbl.create 13 in
4806   List.iter (
4807     fun (name, _, _, flags, _, _, _) ->
4808       List.iter (
4809         function
4810         | Optional group ->
4811             let names = try Hashtbl.find h group with Not_found -> [] in
4812             Hashtbl.replace h group (name :: names)
4813         | _ -> ()
4814       ) flags
4815   ) daemon_functions;
4816   let groups = Hashtbl.fold (fun k _ ks -> k :: ks) h [] in
4817   let groups =
4818     List.map (
4819       fun group -> group, List.sort compare (Hashtbl.find h group)
4820     ) groups in
4821   List.sort (fun x y -> compare (fst x) (fst y)) groups
4822
4823 (* Check function names etc. for consistency. *)
4824 let check_functions () =
4825   let contains_uppercase str =
4826     let len = String.length str in
4827     let rec loop i =
4828       if i >= len then false
4829       else (
4830         let c = str.[i] in
4831         if c >= 'A' && c <= 'Z' then true
4832         else loop (i+1)
4833       )
4834     in
4835     loop 0
4836   in
4837
4838   (* Check function names. *)
4839   List.iter (
4840     fun (name, _, _, _, _, _, _) ->
4841       if String.length name >= 7 && String.sub name 0 7 = "guestfs" then
4842         failwithf "function name %s does not need 'guestfs' prefix" name;
4843       if name = "" then
4844         failwithf "function name is empty";
4845       if name.[0] < 'a' || name.[0] > 'z' then
4846         failwithf "function name %s must start with lowercase a-z" name;
4847       if String.contains name '-' then
4848         failwithf "function name %s should not contain '-', use '_' instead."
4849           name
4850   ) all_functions;
4851
4852   (* Check function parameter/return names. *)
4853   List.iter (
4854     fun (name, style, _, _, _, _, _) ->
4855       let check_arg_ret_name n =
4856         if contains_uppercase n then
4857           failwithf "%s param/ret %s should not contain uppercase chars"
4858             name n;
4859         if String.contains n '-' || String.contains n '_' then
4860           failwithf "%s param/ret %s should not contain '-' or '_'"
4861             name n;
4862         if n = "value" then
4863           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;
4864         if n = "int" || n = "char" || n = "short" || n = "long" then
4865           failwithf "%s has a param/ret which conflicts with a C type (eg. 'int', 'char' etc.)" name;
4866         if n = "i" || n = "n" then
4867           failwithf "%s has a param/ret called 'i' or 'n', which will cause some conflicts in the generated code" name;
4868         if n = "argv" || n = "args" then
4869           failwithf "%s has a param/ret called 'argv' or 'args', which will cause some conflicts in the generated code" name;
4870
4871         (* List Haskell, OCaml and C keywords here.
4872          * http://www.haskell.org/haskellwiki/Keywords
4873          * http://caml.inria.fr/pub/docs/manual-ocaml/lex.html#operator-char
4874          * http://en.wikipedia.org/wiki/C_syntax#Reserved_keywords
4875          * Formatted via: cat c haskell ocaml|sort -u|grep -vE '_|^val$' \
4876          *   |perl -pe 's/(.+)/"$1";/'|fmt -70
4877          * Omitting _-containing words, since they're handled above.
4878          * Omitting the OCaml reserved word, "val", is ok,
4879          * and saves us from renaming several parameters.
4880          *)
4881         let reserved = [
4882           "and"; "as"; "asr"; "assert"; "auto"; "begin"; "break"; "case";
4883           "char"; "class"; "const"; "constraint"; "continue"; "data";
4884           "default"; "deriving"; "do"; "done"; "double"; "downto"; "else";
4885           "end"; "enum"; "exception"; "extern"; "external"; "false"; "float";
4886           "for"; "forall"; "foreign"; "fun"; "function"; "functor"; "goto";
4887           "hiding"; "if"; "import"; "in"; "include"; "infix"; "infixl";
4888           "infixr"; "inherit"; "initializer"; "inline"; "instance"; "int";
4889           "interface";
4890           "land"; "lazy"; "let"; "long"; "lor"; "lsl"; "lsr"; "lxor";
4891           "match"; "mdo"; "method"; "mod"; "module"; "mutable"; "new";
4892           "newtype"; "object"; "of"; "open"; "or"; "private"; "qualified";
4893           "rec"; "register"; "restrict"; "return"; "short"; "sig"; "signed";
4894           "sizeof"; "static"; "struct"; "switch"; "then"; "to"; "true"; "try";
4895           "type"; "typedef"; "union"; "unsigned"; "virtual"; "void";
4896           "volatile"; "when"; "where"; "while";
4897           ] in
4898         if List.mem n reserved then
4899           failwithf "%s has param/ret using reserved word %s" name n;
4900       in
4901
4902       (match fst style with
4903        | RErr -> ()
4904        | RInt n | RInt64 n | RBool n
4905        | RConstString n | RConstOptString n | RString n
4906        | RStringList n | RStruct (n, _) | RStructList (n, _)
4907        | RHashtable n | RBufferOut n ->
4908            check_arg_ret_name n
4909       );
4910       List.iter (fun arg -> check_arg_ret_name (name_of_argt arg)) (snd style)
4911   ) all_functions;
4912
4913   (* Check short descriptions. *)
4914   List.iter (
4915     fun (name, _, _, _, _, shortdesc, _) ->
4916       if shortdesc.[0] <> Char.lowercase shortdesc.[0] then
4917         failwithf "short description of %s should begin with lowercase." name;
4918       let c = shortdesc.[String.length shortdesc-1] in
4919       if c = '\n' || c = '.' then
4920         failwithf "short description of %s should not end with . or \\n." name
4921   ) all_functions;
4922
4923   (* Check long dscriptions. *)
4924   List.iter (
4925     fun (name, _, _, _, _, _, longdesc) ->
4926       if longdesc.[String.length longdesc-1] = '\n' then
4927         failwithf "long description of %s should not end with \\n." name
4928   ) all_functions;
4929
4930   (* Check proc_nrs. *)
4931   List.iter (
4932     fun (name, _, proc_nr, _, _, _, _) ->
4933       if proc_nr <= 0 then
4934         failwithf "daemon function %s should have proc_nr > 0" name
4935   ) daemon_functions;
4936
4937   List.iter (
4938     fun (name, _, proc_nr, _, _, _, _) ->
4939       if proc_nr <> -1 then
4940         failwithf "non-daemon function %s should have proc_nr -1" name
4941   ) non_daemon_functions;
4942
4943   let proc_nrs =
4944     List.map (fun (name, _, proc_nr, _, _, _, _) -> name, proc_nr)
4945       daemon_functions in
4946   let proc_nrs =
4947     List.sort (fun (_,nr1) (_,nr2) -> compare nr1 nr2) proc_nrs in
4948   let rec loop = function
4949     | [] -> ()
4950     | [_] -> ()
4951     | (name1,nr1) :: ((name2,nr2) :: _ as rest) when nr1 < nr2 ->
4952         loop rest
4953     | (name1,nr1) :: (name2,nr2) :: _ ->
4954         failwithf "%s and %s have conflicting procedure numbers (%d, %d)"
4955           name1 name2 nr1 nr2
4956   in
4957   loop proc_nrs;
4958
4959   (* Check tests. *)
4960   List.iter (
4961     function
4962       (* Ignore functions that have no tests.  We generate a
4963        * warning when the user does 'make check' instead.
4964        *)
4965     | name, _, _, _, [], _, _ -> ()
4966     | name, _, _, _, tests, _, _ ->
4967         let funcs =
4968           List.map (
4969             fun (_, _, test) ->
4970               match seq_of_test test with
4971               | [] ->
4972                   failwithf "%s has a test containing an empty sequence" name
4973               | cmds -> List.map List.hd cmds
4974           ) tests in
4975         let funcs = List.flatten funcs in
4976
4977         let tested = List.mem name funcs in
4978
4979         if not tested then
4980           failwithf "function %s has tests but does not test itself" name
4981   ) all_functions
4982
4983 (* 'pr' prints to the current output file. *)
4984 let chan = ref Pervasives.stdout
4985 let lines = ref 0
4986 let pr fs =
4987   ksprintf
4988     (fun str ->
4989        let i = count_chars '\n' str in
4990        lines := !lines + i;
4991        output_string !chan str
4992     ) fs
4993
4994 let copyright_years =
4995   let this_year = 1900 + (localtime (time ())).tm_year in
4996   if this_year > 2009 then sprintf "2009-%04d" this_year else "2009"
4997
4998 (* Generate a header block in a number of standard styles. *)
4999 type comment_style =
5000     CStyle | CPlusPlusStyle | HashStyle | OCamlStyle | HaskellStyle
5001 type license = GPLv2plus | LGPLv2plus
5002
5003 let generate_header ?(extra_inputs = []) comment license =
5004   let inputs = "src/generator.ml" :: extra_inputs in
5005   let c = match comment with
5006     | CStyle ->         pr "/* "; " *"
5007     | CPlusPlusStyle -> pr "// "; "//"
5008     | HashStyle ->      pr "# ";  "#"
5009     | OCamlStyle ->     pr "(* "; " *"
5010     | HaskellStyle ->   pr "{- "; "  " in
5011   pr "libguestfs generated file\n";
5012   pr "%s WARNING: THIS FILE IS GENERATED FROM:\n" c;
5013   List.iter (pr "%s   %s\n" c) inputs;
5014   pr "%s ANY CHANGES YOU MAKE TO THIS FILE WILL BE LOST.\n" c;
5015   pr "%s\n" c;
5016   pr "%s Copyright (C) %s Red Hat Inc.\n" c copyright_years;
5017   pr "%s\n" c;
5018   (match license with
5019    | GPLv2plus ->
5020        pr "%s This program is free software; you can redistribute it and/or modify\n" c;
5021        pr "%s it under the terms of the GNU General Public License as published by\n" c;
5022        pr "%s the Free Software Foundation; either version 2 of the License, or\n" c;
5023        pr "%s (at your option) any later version.\n" c;
5024        pr "%s\n" c;
5025        pr "%s This program is distributed in the hope that it will be useful,\n" c;
5026        pr "%s but WITHOUT ANY WARRANTY; without even the implied warranty of\n" c;
5027        pr "%s MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\n" c;
5028        pr "%s GNU General Public License for more details.\n" c;
5029        pr "%s\n" c;
5030        pr "%s You should have received a copy of the GNU General Public License along\n" c;
5031        pr "%s with this program; if not, write to the Free Software Foundation, Inc.,\n" c;
5032        pr "%s 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.\n" c;
5033
5034    | LGPLv2plus ->
5035        pr "%s This library is free software; you can redistribute it and/or\n" c;
5036        pr "%s modify it under the terms of the GNU Lesser General Public\n" c;
5037        pr "%s License as published by the Free Software Foundation; either\n" c;
5038        pr "%s version 2 of the License, or (at your option) any later version.\n" c;
5039        pr "%s\n" c;
5040        pr "%s This library is distributed in the hope that it will be useful,\n" c;
5041        pr "%s but WITHOUT ANY WARRANTY; without even the implied warranty of\n" c;
5042        pr "%s MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n" c;
5043        pr "%s Lesser General Public License for more details.\n" c;
5044        pr "%s\n" c;
5045        pr "%s You should have received a copy of the GNU Lesser General Public\n" c;
5046        pr "%s License along with this library; if not, write to the Free Software\n" c;
5047        pr "%s Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA\n" c;
5048   );
5049   (match comment with
5050    | CStyle -> pr " */\n"
5051    | CPlusPlusStyle
5052    | HashStyle -> ()
5053    | OCamlStyle -> pr " *)\n"
5054    | HaskellStyle -> pr "-}\n"
5055   );
5056   pr "\n"
5057
5058 (* Start of main code generation functions below this line. *)
5059
5060 (* Generate the pod documentation for the C API. *)
5061 let rec generate_actions_pod () =
5062   List.iter (
5063     fun (shortname, style, _, flags, _, _, longdesc) ->
5064       if not (List.mem NotInDocs flags) then (
5065         let name = "guestfs_" ^ shortname in
5066         pr "=head2 %s\n\n" name;
5067         pr " ";
5068         generate_prototype ~extern:false ~handle:"handle" name style;
5069         pr "\n\n";
5070         pr "%s\n\n" longdesc;
5071         (match fst style with
5072          | RErr ->
5073              pr "This function returns 0 on success or -1 on error.\n\n"
5074          | RInt _ ->
5075              pr "On error this function returns -1.\n\n"
5076          | RInt64 _ ->
5077              pr "On error this function returns -1.\n\n"
5078          | RBool _ ->
5079              pr "This function returns a C truth value on success or -1 on error.\n\n"
5080          | RConstString _ ->
5081              pr "This function returns a string, or NULL on error.
5082 The string is owned by the guest handle and must I<not> be freed.\n\n"
5083          | RConstOptString _ ->
5084              pr "This function returns a string which may be NULL.
5085 There is way to return an error from this function.
5086 The string is owned by the guest handle and must I<not> be freed.\n\n"
5087          | RString _ ->
5088              pr "This function returns a string, or NULL on error.
5089 I<The caller must free the returned string after use>.\n\n"
5090          | RStringList _ ->
5091              pr "This function returns a NULL-terminated array of strings
5092 (like L<environ(3)>), or NULL if there was an error.
5093 I<The caller must free the strings and the array after use>.\n\n"
5094          | RStruct (_, typ) ->
5095              pr "This function returns a C<struct guestfs_%s *>,
5096 or NULL if there was an error.
5097 I<The caller must call C<guestfs_free_%s> after use>.\n\n" typ typ
5098          | RStructList (_, typ) ->
5099              pr "This function returns a C<struct guestfs_%s_list *>
5100 (see E<lt>guestfs-structs.hE<gt>),
5101 or NULL if there was an error.
5102 I<The caller must call C<guestfs_free_%s_list> after use>.\n\n" typ typ
5103          | RHashtable _ ->
5104              pr "This function returns a NULL-terminated array of
5105 strings, or NULL if there was an error.
5106 The array of strings will always have length C<2n+1>, where
5107 C<n> keys and values alternate, followed by the trailing NULL entry.
5108 I<The caller must free the strings and the array after use>.\n\n"
5109          | RBufferOut _ ->
5110              pr "This function returns a buffer, or NULL on error.
5111 The size of the returned buffer is written to C<*size_r>.
5112 I<The caller must free the returned buffer after use>.\n\n"
5113         );
5114         if List.mem ProtocolLimitWarning flags then
5115           pr "%s\n\n" protocol_limit_warning;
5116         if List.mem DangerWillRobinson flags then
5117           pr "%s\n\n" danger_will_robinson;
5118         match deprecation_notice flags with
5119         | None -> ()
5120         | Some txt -> pr "%s\n\n" txt
5121       )
5122   ) all_functions_sorted
5123
5124 and generate_structs_pod () =
5125   (* Structs documentation. *)
5126   List.iter (
5127     fun (typ, cols) ->
5128       pr "=head2 guestfs_%s\n" typ;
5129       pr "\n";
5130       pr " struct guestfs_%s {\n" typ;
5131       List.iter (
5132         function
5133         | name, FChar -> pr "   char %s;\n" name
5134         | name, FUInt32 -> pr "   uint32_t %s;\n" name
5135         | name, FInt32 -> pr "   int32_t %s;\n" name
5136         | name, (FUInt64|FBytes) -> pr "   uint64_t %s;\n" name
5137         | name, FInt64 -> pr "   int64_t %s;\n" name
5138         | name, FString -> pr "   char *%s;\n" name
5139         | name, FBuffer ->
5140             pr "   /* The next two fields describe a byte array. */\n";
5141             pr "   uint32_t %s_len;\n" name;
5142             pr "   char *%s;\n" name
5143         | name, FUUID ->
5144             pr "   /* The next field is NOT nul-terminated, be careful when printing it: */\n";
5145             pr "   char %s[32];\n" name
5146         | name, FOptPercent ->
5147             pr "   /* The next field is [0..100] or -1 meaning 'not present': */\n";
5148             pr "   float %s;\n" name
5149       ) cols;
5150       pr " };\n";
5151       pr " \n";
5152       pr " struct guestfs_%s_list {\n" typ;
5153       pr "   uint32_t len; /* Number of elements in list. */\n";
5154       pr "   struct guestfs_%s *val; /* Elements. */\n" typ;
5155       pr " };\n";
5156       pr " \n";
5157       pr " void guestfs_free_%s (struct guestfs_free_%s *);\n" typ typ;
5158       pr " void guestfs_free_%s_list (struct guestfs_free_%s_list *);\n"
5159         typ typ;
5160       pr "\n"
5161   ) structs
5162
5163 and generate_availability_pod () =
5164   (* Availability documentation. *)
5165   pr "=over 4\n";
5166   pr "\n";
5167   List.iter (
5168     fun (group, functions) ->
5169       pr "=item B<%s>\n" group;
5170       pr "\n";
5171       pr "The following functions:\n";
5172       List.iter (pr "L</guestfs_%s>\n") functions;
5173       pr "\n"
5174   ) optgroups;
5175   pr "=back\n";
5176   pr "\n"
5177
5178 (* Generate the protocol (XDR) file, 'guestfs_protocol.x' and
5179  * indirectly 'guestfs_protocol.h' and 'guestfs_protocol.c'.
5180  *
5181  * We have to use an underscore instead of a dash because otherwise
5182  * rpcgen generates incorrect code.
5183  *
5184  * This header is NOT exported to clients, but see also generate_structs_h.
5185  *)
5186 and generate_xdr () =
5187   generate_header CStyle LGPLv2plus;
5188
5189   (* This has to be defined to get around a limitation in Sun's rpcgen. *)
5190   pr "typedef string str<>;\n";
5191   pr "\n";
5192
5193   (* Internal structures. *)
5194   List.iter (
5195     function
5196     | typ, cols ->
5197         pr "struct guestfs_int_%s {\n" typ;
5198         List.iter (function
5199                    | name, FChar -> pr "  char %s;\n" name
5200                    | name, FString -> pr "  string %s<>;\n" name
5201                    | name, FBuffer -> pr "  opaque %s<>;\n" name
5202                    | name, FUUID -> pr "  opaque %s[32];\n" name
5203                    | name, (FInt32|FUInt32) -> pr "  int %s;\n" name
5204                    | name, (FInt64|FUInt64|FBytes) -> pr "  hyper %s;\n" name
5205                    | name, FOptPercent -> pr "  float %s;\n" name
5206                   ) cols;
5207         pr "};\n";
5208         pr "\n";
5209         pr "typedef struct guestfs_int_%s guestfs_int_%s_list<>;\n" typ typ;
5210         pr "\n";
5211   ) structs;
5212
5213   List.iter (
5214     fun (shortname, style, _, _, _, _, _) ->
5215       let name = "guestfs_" ^ shortname in
5216
5217       (match snd style with
5218        | [] -> ()
5219        | args ->
5220            pr "struct %s_args {\n" name;
5221            List.iter (
5222              function
5223              | Pathname n | Device n | Dev_or_Path n | String n ->
5224                  pr "  string %s<>;\n" n
5225              | OptString n -> pr "  str *%s;\n" n
5226              | StringList n | DeviceList n -> pr "  str %s<>;\n" n
5227              | Bool n -> pr "  bool %s;\n" n
5228              | Int n -> pr "  int %s;\n" n
5229              | Int64 n -> pr "  hyper %s;\n" n
5230              | FileIn _ | FileOut _ -> ()
5231            ) args;
5232            pr "};\n\n"
5233       );
5234       (match fst style with
5235        | RErr -> ()
5236        | RInt n ->
5237            pr "struct %s_ret {\n" name;
5238            pr "  int %s;\n" n;
5239            pr "};\n\n"
5240        | RInt64 n ->
5241            pr "struct %s_ret {\n" name;
5242            pr "  hyper %s;\n" n;
5243            pr "};\n\n"
5244        | RBool n ->
5245            pr "struct %s_ret {\n" name;
5246            pr "  bool %s;\n" n;
5247            pr "};\n\n"
5248        | RConstString _ | RConstOptString _ ->
5249            failwithf "RConstString|RConstOptString cannot be used by daemon functions"
5250        | RString n ->
5251            pr "struct %s_ret {\n" name;
5252            pr "  string %s<>;\n" n;
5253            pr "};\n\n"
5254        | RStringList n ->
5255            pr "struct %s_ret {\n" name;
5256            pr "  str %s<>;\n" n;
5257            pr "};\n\n"
5258        | RStruct (n, typ) ->
5259            pr "struct %s_ret {\n" name;
5260            pr "  guestfs_int_%s %s;\n" typ n;
5261            pr "};\n\n"
5262        | RStructList (n, typ) ->
5263            pr "struct %s_ret {\n" name;
5264            pr "  guestfs_int_%s_list %s;\n" typ n;
5265            pr "};\n\n"
5266        | RHashtable n ->
5267            pr "struct %s_ret {\n" name;
5268            pr "  str %s<>;\n" n;
5269            pr "};\n\n"
5270        | RBufferOut n ->
5271            pr "struct %s_ret {\n" name;
5272            pr "  opaque %s<>;\n" n;
5273            pr "};\n\n"
5274       );
5275   ) daemon_functions;
5276
5277   (* Table of procedure numbers. *)
5278   pr "enum guestfs_procedure {\n";
5279   List.iter (
5280     fun (shortname, _, proc_nr, _, _, _, _) ->
5281       pr "  GUESTFS_PROC_%s = %d,\n" (String.uppercase shortname) proc_nr
5282   ) daemon_functions;
5283   pr "  GUESTFS_PROC_NR_PROCS\n";
5284   pr "};\n";
5285   pr "\n";
5286
5287   (* Having to choose a maximum message size is annoying for several
5288    * reasons (it limits what we can do in the API), but it (a) makes
5289    * the protocol a lot simpler, and (b) provides a bound on the size
5290    * of the daemon which operates in limited memory space.
5291    *)
5292   pr "const GUESTFS_MESSAGE_MAX = %d;\n" (4 * 1024 * 1024);
5293   pr "\n";
5294
5295   (* Message header, etc. *)
5296   pr "\
5297 /* The communication protocol is now documented in the guestfs(3)
5298  * manpage.
5299  */
5300
5301 const GUESTFS_PROGRAM = 0x2000F5F5;
5302 const GUESTFS_PROTOCOL_VERSION = 1;
5303
5304 /* These constants must be larger than any possible message length. */
5305 const GUESTFS_LAUNCH_FLAG = 0xf5f55ff5;
5306 const GUESTFS_CANCEL_FLAG = 0xffffeeee;
5307
5308 enum guestfs_message_direction {
5309   GUESTFS_DIRECTION_CALL = 0,        /* client -> daemon */
5310   GUESTFS_DIRECTION_REPLY = 1        /* daemon -> client */
5311 };
5312
5313 enum guestfs_message_status {
5314   GUESTFS_STATUS_OK = 0,
5315   GUESTFS_STATUS_ERROR = 1
5316 };
5317
5318 const GUESTFS_ERROR_LEN = 256;
5319
5320 struct guestfs_message_error {
5321   string error_message<GUESTFS_ERROR_LEN>;
5322 };
5323
5324 struct guestfs_message_header {
5325   unsigned prog;                     /* GUESTFS_PROGRAM */
5326   unsigned vers;                     /* GUESTFS_PROTOCOL_VERSION */
5327   guestfs_procedure proc;            /* GUESTFS_PROC_x */
5328   guestfs_message_direction direction;
5329   unsigned serial;                   /* message serial number */
5330   guestfs_message_status status;
5331 };
5332
5333 const GUESTFS_MAX_CHUNK_SIZE = 8192;
5334
5335 struct guestfs_chunk {
5336   int cancel;                        /* if non-zero, transfer is cancelled */
5337   /* data size is 0 bytes if the transfer has finished successfully */
5338   opaque data<GUESTFS_MAX_CHUNK_SIZE>;
5339 };
5340 "
5341
5342 (* Generate the guestfs-structs.h file. *)
5343 and generate_structs_h () =
5344   generate_header CStyle LGPLv2plus;
5345
5346   (* This is a public exported header file containing various
5347    * structures.  The structures are carefully written to have
5348    * exactly the same in-memory format as the XDR structures that
5349    * we use on the wire to the daemon.  The reason for creating
5350    * copies of these structures here is just so we don't have to
5351    * export the whole of guestfs_protocol.h (which includes much
5352    * unrelated and XDR-dependent stuff that we don't want to be
5353    * public, or required by clients).
5354    *
5355    * To reiterate, we will pass these structures to and from the
5356    * client with a simple assignment or memcpy, so the format
5357    * must be identical to what rpcgen / the RFC defines.
5358    *)
5359
5360   (* Public structures. *)
5361   List.iter (
5362     fun (typ, cols) ->
5363       pr "struct guestfs_%s {\n" typ;
5364       List.iter (
5365         function
5366         | name, FChar -> pr "  char %s;\n" name
5367         | name, FString -> pr "  char *%s;\n" name
5368         | name, FBuffer ->
5369             pr "  uint32_t %s_len;\n" name;
5370             pr "  char *%s;\n" name
5371         | name, FUUID -> pr "  char %s[32]; /* this is NOT nul-terminated, be careful when printing */\n" name
5372         | name, FUInt32 -> pr "  uint32_t %s;\n" name
5373         | name, FInt32 -> pr "  int32_t %s;\n" name
5374         | name, (FUInt64|FBytes) -> pr "  uint64_t %s;\n" name
5375         | name, FInt64 -> pr "  int64_t %s;\n" name
5376         | name, FOptPercent -> pr "  float %s; /* [0..100] or -1 */\n" name
5377       ) cols;
5378       pr "};\n";
5379       pr "\n";
5380       pr "struct guestfs_%s_list {\n" typ;
5381       pr "  uint32_t len;\n";
5382       pr "  struct guestfs_%s *val;\n" typ;
5383       pr "};\n";
5384       pr "\n";
5385       pr "extern void guestfs_free_%s (struct guestfs_%s *);\n" typ typ;
5386       pr "extern void guestfs_free_%s_list (struct guestfs_%s_list *);\n" typ typ;
5387       pr "\n"
5388   ) structs
5389
5390 (* Generate the guestfs-actions.h file. *)
5391 and generate_actions_h () =
5392   generate_header CStyle LGPLv2plus;
5393   List.iter (
5394     fun (shortname, style, _, _, _, _, _) ->
5395       let name = "guestfs_" ^ shortname in
5396       generate_prototype ~single_line:true ~newline:true ~handle:"handle"
5397         name style
5398   ) all_functions
5399
5400 (* Generate the guestfs-internal-actions.h file. *)
5401 and generate_internal_actions_h () =
5402   generate_header CStyle LGPLv2plus;
5403   List.iter (
5404     fun (shortname, style, _, _, _, _, _) ->
5405       let name = "guestfs__" ^ shortname in
5406       generate_prototype ~single_line:true ~newline:true ~handle:"handle"
5407         name style
5408   ) non_daemon_functions
5409
5410 (* Generate the client-side dispatch stubs. *)
5411 and generate_client_actions () =
5412   generate_header CStyle LGPLv2plus;
5413
5414   pr "\
5415 #include <stdio.h>
5416 #include <stdlib.h>
5417 #include <stdint.h>
5418 #include <string.h>
5419 #include <inttypes.h>
5420
5421 #include \"guestfs.h\"
5422 #include \"guestfs-internal.h\"
5423 #include \"guestfs-internal-actions.h\"
5424 #include \"guestfs_protocol.h\"
5425
5426 #define error guestfs_error
5427 //#define perrorf guestfs_perrorf
5428 #define safe_malloc guestfs_safe_malloc
5429 #define safe_realloc guestfs_safe_realloc
5430 //#define safe_strdup guestfs_safe_strdup
5431 #define safe_memdup guestfs_safe_memdup
5432
5433 /* Check the return message from a call for validity. */
5434 static int
5435 check_reply_header (guestfs_h *g,
5436                     const struct guestfs_message_header *hdr,
5437                     unsigned int proc_nr, unsigned int serial)
5438 {
5439   if (hdr->prog != GUESTFS_PROGRAM) {
5440     error (g, \"wrong program (%%d/%%d)\", hdr->prog, GUESTFS_PROGRAM);
5441     return -1;
5442   }
5443   if (hdr->vers != GUESTFS_PROTOCOL_VERSION) {
5444     error (g, \"wrong protocol version (%%d/%%d)\",
5445            hdr->vers, GUESTFS_PROTOCOL_VERSION);
5446     return -1;
5447   }
5448   if (hdr->direction != GUESTFS_DIRECTION_REPLY) {
5449     error (g, \"unexpected message direction (%%d/%%d)\",
5450            hdr->direction, GUESTFS_DIRECTION_REPLY);
5451     return -1;
5452   }
5453   if (hdr->proc != proc_nr) {
5454     error (g, \"unexpected procedure number (%%d/%%d)\", hdr->proc, proc_nr);
5455     return -1;
5456   }
5457   if (hdr->serial != serial) {
5458     error (g, \"unexpected serial (%%d/%%d)\", hdr->serial, serial);
5459     return -1;
5460   }
5461
5462   return 0;
5463 }
5464
5465 /* Check we are in the right state to run a high-level action. */
5466 static int
5467 check_state (guestfs_h *g, const char *caller)
5468 {
5469   if (!guestfs__is_ready (g)) {
5470     if (guestfs__is_config (g) || guestfs__is_launching (g))
5471       error (g, \"%%s: call launch before using this function\\n(in guestfish, don't forget to use the 'run' command)\",
5472         caller);
5473     else
5474       error (g, \"%%s called from the wrong state, %%d != READY\",
5475         caller, guestfs__get_state (g));
5476     return -1;
5477   }
5478   return 0;
5479 }
5480
5481 ";
5482
5483   (* Generate code to generate guestfish call traces. *)
5484   let trace_call shortname style =
5485     pr "  if (guestfs__get_trace (g)) {\n";
5486
5487     let needs_i =
5488       List.exists (function
5489                    | StringList _ | DeviceList _ -> true
5490                    | _ -> false) (snd style) in
5491     if needs_i then (
5492       pr "    int i;\n";
5493       pr "\n"
5494     );
5495
5496     pr "    printf (\"%s\");\n" shortname;
5497     List.iter (
5498       function
5499       | String n                        (* strings *)
5500       | Device n
5501       | Pathname n
5502       | Dev_or_Path n
5503       | FileIn n
5504       | FileOut n ->
5505           (* guestfish doesn't support string escaping, so neither do we *)
5506           pr "    printf (\" \\\"%%s\\\"\", %s);\n" n
5507       | OptString n ->                  (* string option *)
5508           pr "    if (%s) printf (\" \\\"%%s\\\"\", %s);\n" n n;
5509           pr "    else printf (\" null\");\n"
5510       | StringList n
5511       | DeviceList n ->                 (* string list *)
5512           pr "    putchar (' ');\n";
5513           pr "    putchar ('\"');\n";
5514           pr "    for (i = 0; %s[i]; ++i) {\n" n;
5515           pr "      if (i > 0) putchar (' ');\n";
5516           pr "      fputs (%s[i], stdout);\n" n;
5517           pr "    }\n";
5518           pr "    putchar ('\"');\n";
5519       | Bool n ->                       (* boolean *)
5520           pr "    fputs (%s ? \" true\" : \" false\", stdout);\n" n
5521       | Int n ->                        (* int *)
5522           pr "    printf (\" %%d\", %s);\n" n
5523       | Int64 n ->
5524           pr "    printf (\" %%\" PRIi64, %s);\n" n
5525     ) (snd style);
5526     pr "    putchar ('\\n');\n";
5527     pr "  }\n";
5528     pr "\n";
5529   in
5530
5531   (* For non-daemon functions, generate a wrapper around each function. *)
5532   List.iter (
5533     fun (shortname, style, _, _, _, _, _) ->
5534       let name = "guestfs_" ^ shortname in
5535
5536       generate_prototype ~extern:false ~semicolon:false ~newline:true
5537         ~handle:"g" name style;
5538       pr "{\n";
5539       trace_call shortname style;
5540       pr "  return guestfs__%s " shortname;
5541       generate_c_call_args ~handle:"g" style;
5542       pr ";\n";
5543       pr "}\n";
5544       pr "\n"
5545   ) non_daemon_functions;
5546
5547   (* Client-side stubs for each function. *)
5548   List.iter (
5549     fun (shortname, style, _, _, _, _, _) ->
5550       let name = "guestfs_" ^ shortname in
5551
5552       (* Generate the action stub. *)
5553       generate_prototype ~extern:false ~semicolon:false ~newline:true
5554         ~handle:"g" name style;
5555
5556       let error_code =
5557         match fst style with
5558         | RErr | RInt _ | RInt64 _ | RBool _ -> "-1"
5559         | RConstString _ | RConstOptString _ ->
5560             failwithf "RConstString|RConstOptString cannot be used by daemon functions"
5561         | RString _ | RStringList _
5562         | RStruct _ | RStructList _
5563         | RHashtable _ | RBufferOut _ ->
5564             "NULL" in
5565
5566       pr "{\n";
5567
5568       (match snd style with
5569        | [] -> ()
5570        | _ -> pr "  struct %s_args args;\n" name
5571       );
5572
5573       pr "  guestfs_message_header hdr;\n";
5574       pr "  guestfs_message_error err;\n";
5575       let has_ret =
5576         match fst style with
5577         | RErr -> false
5578         | RConstString _ | RConstOptString _ ->
5579             failwithf "RConstString|RConstOptString cannot be used by daemon functions"
5580         | RInt _ | RInt64 _
5581         | RBool _ | RString _ | RStringList _
5582         | RStruct _ | RStructList _
5583         | RHashtable _ | RBufferOut _ ->
5584             pr "  struct %s_ret ret;\n" name;
5585             true in
5586
5587       pr "  int serial;\n";
5588       pr "  int r;\n";
5589       pr "\n";
5590       trace_call shortname style;
5591       pr "  if (check_state (g, \"%s\") == -1) return %s;\n" name error_code;
5592       pr "  guestfs___set_busy (g);\n";
5593       pr "\n";
5594
5595       (* Send the main header and arguments. *)
5596       (match snd style with
5597        | [] ->
5598            pr "  serial = guestfs___send (g, GUESTFS_PROC_%s, NULL, NULL);\n"
5599              (String.uppercase shortname)
5600        | args ->
5601            List.iter (
5602              function
5603              | Pathname n | Device n | Dev_or_Path n | String n ->
5604                  pr "  args.%s = (char *) %s;\n" n n
5605              | OptString n ->
5606                  pr "  args.%s = %s ? (char **) &%s : NULL;\n" n n n
5607              | StringList n | DeviceList n ->
5608                  pr "  args.%s.%s_val = (char **) %s;\n" n n n;
5609                  pr "  for (args.%s.%s_len = 0; %s[args.%s.%s_len]; args.%s.%s_len++) ;\n" n n n n n n n;
5610              | Bool n ->
5611                  pr "  args.%s = %s;\n" n n
5612              | Int n ->
5613                  pr "  args.%s = %s;\n" n n
5614              | Int64 n ->
5615                  pr "  args.%s = %s;\n" n n
5616              | FileIn _ | FileOut _ -> ()
5617            ) args;
5618            pr "  serial = guestfs___send (g, GUESTFS_PROC_%s,\n"
5619              (String.uppercase shortname);
5620            pr "        (xdrproc_t) xdr_%s_args, (char *) &args);\n"
5621              name;
5622       );
5623       pr "  if (serial == -1) {\n";
5624       pr "    guestfs___end_busy (g);\n";
5625       pr "    return %s;\n" error_code;
5626       pr "  }\n";
5627       pr "\n";
5628
5629       (* Send any additional files (FileIn) requested. *)
5630       let need_read_reply_label = ref false in
5631       List.iter (
5632         function
5633         | FileIn n ->
5634             pr "  r = guestfs___send_file (g, %s);\n" n;
5635             pr "  if (r == -1) {\n";
5636             pr "    guestfs___end_busy (g);\n";
5637             pr "    return %s;\n" error_code;
5638             pr "  }\n";
5639             pr "  if (r == -2) /* daemon cancelled */\n";
5640             pr "    goto read_reply;\n";
5641             need_read_reply_label := true;
5642             pr "\n";
5643         | _ -> ()
5644       ) (snd style);
5645
5646       (* Wait for the reply from the remote end. *)
5647       if !need_read_reply_label then pr " read_reply:\n";
5648       pr "  memset (&hdr, 0, sizeof hdr);\n";
5649       pr "  memset (&err, 0, sizeof err);\n";
5650       if has_ret then pr "  memset (&ret, 0, sizeof ret);\n";
5651       pr "\n";
5652       pr "  r = guestfs___recv (g, \"%s\", &hdr, &err,\n        " shortname;
5653       if not has_ret then
5654         pr "NULL, NULL"
5655       else
5656         pr "(xdrproc_t) xdr_guestfs_%s_ret, (char *) &ret" shortname;
5657       pr ");\n";
5658
5659       pr "  if (r == -1) {\n";
5660       pr "    guestfs___end_busy (g);\n";
5661       pr "    return %s;\n" error_code;
5662       pr "  }\n";
5663       pr "\n";
5664
5665       pr "  if (check_reply_header (g, &hdr, GUESTFS_PROC_%s, serial) == -1) {\n"
5666         (String.uppercase shortname);
5667       pr "    guestfs___end_busy (g);\n";
5668       pr "    return %s;\n" error_code;
5669       pr "  }\n";
5670       pr "\n";
5671
5672       pr "  if (hdr.status == GUESTFS_STATUS_ERROR) {\n";
5673       pr "    error (g, \"%%s: %%s\", \"%s\", err.error_message);\n" shortname;
5674       pr "    free (err.error_message);\n";
5675       pr "    guestfs___end_busy (g);\n";
5676       pr "    return %s;\n" error_code;
5677       pr "  }\n";
5678       pr "\n";
5679
5680       (* Expecting to receive further files (FileOut)? *)
5681       List.iter (
5682         function
5683         | FileOut n ->
5684             pr "  if (guestfs___recv_file (g, %s) == -1) {\n" n;
5685             pr "    guestfs___end_busy (g);\n";
5686             pr "    return %s;\n" error_code;
5687             pr "  }\n";
5688             pr "\n";
5689         | _ -> ()
5690       ) (snd style);
5691
5692       pr "  guestfs___end_busy (g);\n";
5693
5694       (match fst style with
5695        | RErr -> pr "  return 0;\n"
5696        | RInt n | RInt64 n | RBool n ->
5697            pr "  return ret.%s;\n" n
5698        | RConstString _ | RConstOptString _ ->
5699            failwithf "RConstString|RConstOptString cannot be used by daemon functions"
5700        | RString n ->
5701            pr "  return ret.%s; /* caller will free */\n" n
5702        | RStringList n | RHashtable n ->
5703            pr "  /* caller will free this, but we need to add a NULL entry */\n";
5704            pr "  ret.%s.%s_val =\n" n n;
5705            pr "    safe_realloc (g, ret.%s.%s_val,\n" n n;
5706            pr "                  sizeof (char *) * (ret.%s.%s_len + 1));\n"
5707              n n;
5708            pr "  ret.%s.%s_val[ret.%s.%s_len] = NULL;\n" n n n n;
5709            pr "  return ret.%s.%s_val;\n" n n
5710        | RStruct (n, _) ->
5711            pr "  /* caller will free this */\n";
5712            pr "  return safe_memdup (g, &ret.%s, sizeof (ret.%s));\n" n n
5713        | RStructList (n, _) ->
5714            pr "  /* caller will free this */\n";
5715            pr "  return safe_memdup (g, &ret.%s, sizeof (ret.%s));\n" n n
5716        | RBufferOut n ->
5717            pr "  /* RBufferOut is tricky: If the buffer is zero-length, then\n";
5718            pr "   * _val might be NULL here.  To make the API saner for\n";
5719            pr "   * callers, we turn this case into a unique pointer (using\n";
5720            pr "   * malloc(1)).\n";
5721            pr "   */\n";
5722            pr "  if (ret.%s.%s_len > 0) {\n" n n;
5723            pr "    *size_r = ret.%s.%s_len;\n" n n;
5724            pr "    return ret.%s.%s_val; /* caller will free */\n" n n;
5725            pr "  } else {\n";
5726            pr "    free (ret.%s.%s_val);\n" n n;
5727            pr "    char *p = safe_malloc (g, 1);\n";
5728            pr "    *size_r = ret.%s.%s_len;\n" n n;
5729            pr "    return p;\n";
5730            pr "  }\n";
5731       );
5732
5733       pr "}\n\n"
5734   ) daemon_functions;
5735
5736   (* Functions to free structures. *)
5737   pr "/* Structure-freeing functions.  These rely on the fact that the\n";
5738   pr " * structure format is identical to the XDR format.  See note in\n";
5739   pr " * generator.ml.\n";
5740   pr " */\n";
5741   pr "\n";
5742
5743   List.iter (
5744     fun (typ, _) ->
5745       pr "void\n";
5746       pr "guestfs_free_%s (struct guestfs_%s *x)\n" typ typ;
5747       pr "{\n";
5748       pr "  xdr_free ((xdrproc_t) xdr_guestfs_int_%s, (char *) x);\n" typ;
5749       pr "  free (x);\n";
5750       pr "}\n";
5751       pr "\n";
5752
5753       pr "void\n";
5754       pr "guestfs_free_%s_list (struct guestfs_%s_list *x)\n" typ typ;
5755       pr "{\n";
5756       pr "  xdr_free ((xdrproc_t) xdr_guestfs_int_%s_list, (char *) x);\n" typ;
5757       pr "  free (x);\n";
5758       pr "}\n";
5759       pr "\n";
5760
5761   ) structs;
5762
5763 (* Generate daemon/actions.h. *)
5764 and generate_daemon_actions_h () =
5765   generate_header CStyle GPLv2plus;
5766
5767   pr "#include \"../src/guestfs_protocol.h\"\n";
5768   pr "\n";
5769
5770   List.iter (
5771     fun (name, style, _, _, _, _, _) ->
5772       generate_prototype
5773         ~single_line:true ~newline:true ~in_daemon:true ~prefix:"do_"
5774         name style;
5775   ) daemon_functions
5776
5777 (* Generate the linker script which controls the visibility of
5778  * symbols in the public ABI and ensures no other symbols get
5779  * exported accidentally.
5780  *)
5781 and generate_linker_script () =
5782   generate_header HashStyle GPLv2plus;
5783
5784   let globals = [
5785     "guestfs_create";
5786     "guestfs_close";
5787     "guestfs_get_error_handler";
5788     "guestfs_get_out_of_memory_handler";
5789     "guestfs_last_error";
5790     "guestfs_set_error_handler";
5791     "guestfs_set_launch_done_callback";
5792     "guestfs_set_log_message_callback";
5793     "guestfs_set_out_of_memory_handler";
5794     "guestfs_set_subprocess_quit_callback";
5795
5796     (* Unofficial parts of the API: the bindings code use these
5797      * functions, so it is useful to export them.
5798      *)
5799     "guestfs_safe_calloc";
5800     "guestfs_safe_malloc";
5801   ] in
5802   let functions =
5803     List.map (fun (name, _, _, _, _, _, _) -> "guestfs_" ^ name)
5804       all_functions in
5805   let structs =
5806     List.concat (
5807       List.map (fun (typ, _) ->
5808                   ["guestfs_free_" ^ typ; "guestfs_free_" ^ typ ^ "_list"])
5809         structs
5810     ) in
5811   let globals = List.sort compare (globals @ functions @ structs) in
5812
5813   pr "{\n";
5814   pr "    global:\n";
5815   List.iter (pr "        %s;\n") globals;
5816   pr "\n";
5817
5818   pr "    local:\n";
5819   pr "        *;\n";
5820   pr "};\n"
5821
5822 (* Generate the server-side stubs. *)
5823 and generate_daemon_actions () =
5824   generate_header CStyle GPLv2plus;
5825
5826   pr "#include <config.h>\n";
5827   pr "\n";
5828   pr "#include <stdio.h>\n";
5829   pr "#include <stdlib.h>\n";
5830   pr "#include <string.h>\n";
5831   pr "#include <inttypes.h>\n";
5832   pr "#include <rpc/types.h>\n";
5833   pr "#include <rpc/xdr.h>\n";
5834   pr "\n";
5835   pr "#include \"daemon.h\"\n";
5836   pr "#include \"c-ctype.h\"\n";
5837   pr "#include \"../src/guestfs_protocol.h\"\n";
5838   pr "#include \"actions.h\"\n";
5839   pr "\n";
5840
5841   List.iter (
5842     fun (name, style, _, _, _, _, _) ->
5843       (* Generate server-side stubs. *)
5844       pr "static void %s_stub (XDR *xdr_in)\n" name;
5845       pr "{\n";
5846       let error_code =
5847         match fst style with
5848         | RErr | RInt _ -> pr "  int r;\n"; "-1"
5849         | RInt64 _ -> pr "  int64_t r;\n"; "-1"
5850         | RBool _ -> pr "  int r;\n"; "-1"
5851         | RConstString _ | RConstOptString _ ->
5852             failwithf "RConstString|RConstOptString cannot be used by daemon functions"
5853         | RString _ -> pr "  char *r;\n"; "NULL"
5854         | RStringList _ | RHashtable _ -> pr "  char **r;\n"; "NULL"
5855         | RStruct (_, typ) -> pr "  guestfs_int_%s *r;\n" typ; "NULL"
5856         | RStructList (_, typ) -> pr "  guestfs_int_%s_list *r;\n" typ; "NULL"
5857         | RBufferOut _ ->
5858             pr "  size_t size = 1;\n";
5859             pr "  char *r;\n";
5860             "NULL" in
5861
5862       (match snd style with
5863        | [] -> ()
5864        | args ->
5865            pr "  struct guestfs_%s_args args;\n" name;
5866            List.iter (
5867              function
5868              | Device n | Dev_or_Path n
5869              | Pathname n
5870              | String n -> ()
5871              | OptString n -> pr "  char *%s;\n" n
5872              | StringList n | DeviceList n -> pr "  char **%s;\n" n
5873              | Bool n -> pr "  int %s;\n" n
5874              | Int n -> pr "  int %s;\n" n
5875              | Int64 n -> pr "  int64_t %s;\n" n
5876              | FileIn _ | FileOut _ -> ()
5877            ) args
5878       );
5879       pr "\n";
5880
5881       (match snd style with
5882        | [] -> ()
5883        | args ->
5884            pr "  memset (&args, 0, sizeof args);\n";
5885            pr "\n";
5886            pr "  if (!xdr_guestfs_%s_args (xdr_in, &args)) {\n" name;
5887            pr "    reply_with_error (\"daemon failed to decode procedure arguments\");\n";
5888            pr "    return;\n";
5889            pr "  }\n";
5890            let pr_args n =
5891              pr "  char *%s = args.%s;\n" n n
5892            in
5893            let pr_list_handling_code n =
5894              pr "  %s = realloc (args.%s.%s_val,\n" n n n;
5895              pr "                sizeof (char *) * (args.%s.%s_len+1));\n" n n;
5896              pr "  if (%s == NULL) {\n" n;
5897              pr "    reply_with_perror (\"realloc\");\n";
5898              pr "    goto done;\n";
5899              pr "  }\n";
5900              pr "  %s[args.%s.%s_len] = NULL;\n" n n n;
5901              pr "  args.%s.%s_val = %s;\n" n n n;
5902            in
5903            List.iter (
5904              function
5905              | Pathname n ->
5906                  pr_args n;
5907                  pr "  ABS_PATH (%s, goto done);\n" n;
5908              | Device n ->
5909                  pr_args n;
5910                  pr "  RESOLVE_DEVICE (%s, goto done);\n" n;
5911              | Dev_or_Path n ->
5912                  pr_args n;
5913                  pr "  REQUIRE_ROOT_OR_RESOLVE_DEVICE (%s, goto done);\n" n;
5914              | String n -> pr_args n
5915              | OptString n -> pr "  %s = args.%s ? *args.%s : NULL;\n" n n n
5916              | StringList n ->
5917                  pr_list_handling_code n;
5918              | DeviceList n ->
5919                  pr_list_handling_code n;
5920                  pr "  /* Ensure that each is a device,\n";
5921                  pr "   * and perform device name translation. */\n";
5922                  pr "  { int pvi; for (pvi = 0; physvols[pvi] != NULL; ++pvi)\n";
5923                  pr "    RESOLVE_DEVICE (physvols[pvi], goto done);\n";
5924                  pr "  }\n";
5925              | Bool n -> pr "  %s = args.%s;\n" n n
5926              | Int n -> pr "  %s = args.%s;\n" n n
5927              | Int64 n -> pr "  %s = args.%s;\n" n n
5928              | FileIn _ | FileOut _ -> ()
5929            ) args;
5930            pr "\n"
5931       );
5932
5933
5934       (* this is used at least for do_equal *)
5935       if List.exists (function Pathname _ -> true | _ -> false) (snd style) then (
5936         (* Emit NEED_ROOT just once, even when there are two or
5937            more Pathname args *)
5938         pr "  NEED_ROOT (goto done);\n";
5939       );
5940
5941       (* Don't want to call the impl with any FileIn or FileOut
5942        * parameters, since these go "outside" the RPC protocol.
5943        *)
5944       let args' =
5945         List.filter (function FileIn _ | FileOut _ -> false | _ -> true)
5946           (snd style) in
5947       pr "  r = do_%s " name;
5948       generate_c_call_args (fst style, args');
5949       pr ";\n";
5950
5951       (match fst style with
5952        | RErr | RInt _ | RInt64 _ | RBool _
5953        | RConstString _ | RConstOptString _
5954        | RString _ | RStringList _ | RHashtable _
5955        | RStruct (_, _) | RStructList (_, _) ->
5956            pr "  if (r == %s)\n" error_code;
5957            pr "    /* do_%s has already called reply_with_error */\n" name;
5958            pr "    goto done;\n";
5959            pr "\n"
5960        | RBufferOut _ ->
5961            pr "  /* size == 0 && r == NULL could be a non-error case (just\n";
5962            pr "   * an ordinary zero-length buffer), so be careful ...\n";
5963            pr "   */\n";
5964            pr "  if (size == 1 && r == %s)\n" error_code;
5965            pr "    /* do_%s has already called reply_with_error */\n" name;
5966            pr "    goto done;\n";
5967            pr "\n"
5968       );
5969
5970       (* If there are any FileOut parameters, then the impl must
5971        * send its own reply.
5972        *)
5973       let no_reply =
5974         List.exists (function FileOut _ -> true | _ -> false) (snd style) in
5975       if no_reply then
5976         pr "  /* do_%s has already sent a reply */\n" name
5977       else (
5978         match fst style with
5979         | RErr -> pr "  reply (NULL, NULL);\n"
5980         | RInt n | RInt64 n | RBool n ->
5981             pr "  struct guestfs_%s_ret ret;\n" name;
5982             pr "  ret.%s = r;\n" n;
5983             pr "  reply ((xdrproc_t) &xdr_guestfs_%s_ret, (char *) &ret);\n"
5984               name
5985         | RConstString _ | RConstOptString _ ->
5986             failwithf "RConstString|RConstOptString cannot be used by daemon functions"
5987         | RString n ->
5988             pr "  struct guestfs_%s_ret ret;\n" name;
5989             pr "  ret.%s = r;\n" n;
5990             pr "  reply ((xdrproc_t) &xdr_guestfs_%s_ret, (char *) &ret);\n"
5991               name;
5992             pr "  free (r);\n"
5993         | RStringList n | RHashtable n ->
5994             pr "  struct guestfs_%s_ret ret;\n" name;
5995             pr "  ret.%s.%s_len = count_strings (r);\n" n n;
5996             pr "  ret.%s.%s_val = r;\n" n n;
5997             pr "  reply ((xdrproc_t) &xdr_guestfs_%s_ret, (char *) &ret);\n"
5998               name;
5999             pr "  free_strings (r);\n"
6000         | RStruct (n, _) ->
6001             pr "  struct guestfs_%s_ret ret;\n" name;
6002             pr "  ret.%s = *r;\n" n;
6003             pr "  reply ((xdrproc_t) xdr_guestfs_%s_ret, (char *) &ret);\n"
6004               name;
6005             pr "  xdr_free ((xdrproc_t) xdr_guestfs_%s_ret, (char *) &ret);\n"
6006               name
6007         | RStructList (n, _) ->
6008             pr "  struct guestfs_%s_ret ret;\n" name;
6009             pr "  ret.%s = *r;\n" n;
6010             pr "  reply ((xdrproc_t) xdr_guestfs_%s_ret, (char *) &ret);\n"
6011               name;
6012             pr "  xdr_free ((xdrproc_t) xdr_guestfs_%s_ret, (char *) &ret);\n"
6013               name
6014         | RBufferOut n ->
6015             pr "  struct guestfs_%s_ret ret;\n" name;
6016             pr "  ret.%s.%s_val = r;\n" n n;
6017             pr "  ret.%s.%s_len = size;\n" n n;
6018             pr "  reply ((xdrproc_t) &xdr_guestfs_%s_ret, (char *) &ret);\n"
6019               name;
6020             pr "  free (r);\n"
6021       );
6022
6023       (* Free the args. *)
6024       (match snd style with
6025        | [] ->
6026            pr "done: ;\n";
6027        | _ ->
6028            pr "done:\n";
6029            pr "  xdr_free ((xdrproc_t) xdr_guestfs_%s_args, (char *) &args);\n"
6030              name
6031       );
6032
6033       pr "}\n\n";
6034   ) daemon_functions;
6035
6036   (* Dispatch function. *)
6037   pr "void dispatch_incoming_message (XDR *xdr_in)\n";
6038   pr "{\n";
6039   pr "  switch (proc_nr) {\n";
6040
6041   List.iter (
6042     fun (name, style, _, _, _, _, _) ->
6043       pr "    case GUESTFS_PROC_%s:\n" (String.uppercase name);
6044       pr "      %s_stub (xdr_in);\n" name;
6045       pr "      break;\n"
6046   ) daemon_functions;
6047
6048   pr "    default:\n";
6049   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";
6050   pr "  }\n";
6051   pr "}\n";
6052   pr "\n";
6053
6054   (* LVM columns and tokenization functions. *)
6055   (* XXX This generates crap code.  We should rethink how we
6056    * do this parsing.
6057    *)
6058   List.iter (
6059     function
6060     | typ, cols ->
6061         pr "static const char *lvm_%s_cols = \"%s\";\n"
6062           typ (String.concat "," (List.map fst cols));
6063         pr "\n";
6064
6065         pr "static int lvm_tokenize_%s (char *str, guestfs_int_lvm_%s *r)\n" typ typ;
6066         pr "{\n";
6067         pr "  char *tok, *p, *next;\n";
6068         pr "  int i, j;\n";
6069         pr "\n";
6070         (*
6071           pr "  fprintf (stderr, \"%%s: <<%%s>>\\n\", __func__, str);\n";
6072           pr "\n";
6073         *)
6074         pr "  if (!str) {\n";
6075         pr "    fprintf (stderr, \"%%s: failed: passed a NULL string\\n\", __func__);\n";
6076         pr "    return -1;\n";
6077         pr "  }\n";
6078         pr "  if (!*str || c_isspace (*str)) {\n";
6079         pr "    fprintf (stderr, \"%%s: failed: passed a empty string or one beginning with whitespace\\n\", __func__);\n";
6080         pr "    return -1;\n";
6081         pr "  }\n";
6082         pr "  tok = str;\n";
6083         List.iter (
6084           fun (name, coltype) ->
6085             pr "  if (!tok) {\n";
6086             pr "    fprintf (stderr, \"%%s: failed: string finished early, around token %%s\\n\", __func__, \"%s\");\n" name;
6087             pr "    return -1;\n";
6088             pr "  }\n";
6089             pr "  p = strchrnul (tok, ',');\n";
6090             pr "  if (*p) next = p+1; else next = NULL;\n";
6091             pr "  *p = '\\0';\n";
6092             (match coltype with
6093              | FString ->
6094                  pr "  r->%s = strdup (tok);\n" name;
6095                  pr "  if (r->%s == NULL) {\n" name;
6096                  pr "    perror (\"strdup\");\n";
6097                  pr "    return -1;\n";
6098                  pr "  }\n"
6099              | FUUID ->
6100                  pr "  for (i = j = 0; i < 32; ++j) {\n";
6101                  pr "    if (tok[j] == '\\0') {\n";
6102                  pr "      fprintf (stderr, \"%%s: failed to parse UUID from '%%s'\\n\", __func__, tok);\n";
6103                  pr "      return -1;\n";
6104                  pr "    } else if (tok[j] != '-')\n";
6105                  pr "      r->%s[i++] = tok[j];\n" name;
6106                  pr "  }\n";
6107              | FBytes ->
6108                  pr "  if (sscanf (tok, \"%%\"SCNu64, &r->%s) != 1) {\n" name;
6109                  pr "    fprintf (stderr, \"%%s: failed to parse size '%%s' from token %%s\\n\", __func__, tok, \"%s\");\n" name;
6110                  pr "    return -1;\n";
6111                  pr "  }\n";
6112              | FInt64 ->
6113                  pr "  if (sscanf (tok, \"%%\"SCNi64, &r->%s) != 1) {\n" name;
6114                  pr "    fprintf (stderr, \"%%s: failed to parse int '%%s' from token %%s\\n\", __func__, tok, \"%s\");\n" name;
6115                  pr "    return -1;\n";
6116                  pr "  }\n";
6117              | FOptPercent ->
6118                  pr "  if (tok[0] == '\\0')\n";
6119                  pr "    r->%s = -1;\n" name;
6120                  pr "  else if (sscanf (tok, \"%%f\", &r->%s) != 1) {\n" name;
6121                  pr "    fprintf (stderr, \"%%s: failed to parse float '%%s' from token %%s\\n\", __func__, tok, \"%s\");\n" name;
6122                  pr "    return -1;\n";
6123                  pr "  }\n";
6124              | FBuffer | FInt32 | FUInt32 | FUInt64 | FChar ->
6125                  assert false (* can never be an LVM column *)
6126             );
6127             pr "  tok = next;\n";
6128         ) cols;
6129
6130         pr "  if (tok != NULL) {\n";
6131         pr "    fprintf (stderr, \"%%s: failed: extra tokens at end of string\\n\", __func__);\n";
6132         pr "    return -1;\n";
6133         pr "  }\n";
6134         pr "  return 0;\n";
6135         pr "}\n";
6136         pr "\n";
6137
6138         pr "guestfs_int_lvm_%s_list *\n" typ;
6139         pr "parse_command_line_%ss (void)\n" typ;
6140         pr "{\n";
6141         pr "  char *out, *err;\n";
6142         pr "  char *p, *pend;\n";
6143         pr "  int r, i;\n";
6144         pr "  guestfs_int_lvm_%s_list *ret;\n" typ;
6145         pr "  void *newp;\n";
6146         pr "\n";
6147         pr "  ret = malloc (sizeof *ret);\n";
6148         pr "  if (!ret) {\n";
6149         pr "    reply_with_perror (\"malloc\");\n";
6150         pr "    return NULL;\n";
6151         pr "  }\n";
6152         pr "\n";
6153         pr "  ret->guestfs_int_lvm_%s_list_len = 0;\n" typ;
6154         pr "  ret->guestfs_int_lvm_%s_list_val = NULL;\n" typ;
6155         pr "\n";
6156         pr "  r = command (&out, &err,\n";
6157         pr "           \"lvm\", \"%ss\",\n" typ;
6158         pr "           \"-o\", lvm_%s_cols, \"--unbuffered\", \"--noheadings\",\n" typ;
6159         pr "           \"--nosuffix\", \"--separator\", \",\", \"--units\", \"b\", NULL);\n";
6160         pr "  if (r == -1) {\n";
6161         pr "    reply_with_error (\"%%s\", err);\n";
6162         pr "    free (out);\n";
6163         pr "    free (err);\n";
6164         pr "    free (ret);\n";
6165         pr "    return NULL;\n";
6166         pr "  }\n";
6167         pr "\n";
6168         pr "  free (err);\n";
6169         pr "\n";
6170         pr "  /* Tokenize each line of the output. */\n";
6171         pr "  p = out;\n";
6172         pr "  i = 0;\n";
6173         pr "  while (p) {\n";
6174         pr "    pend = strchr (p, '\\n');       /* Get the next line of output. */\n";
6175         pr "    if (pend) {\n";
6176         pr "      *pend = '\\0';\n";
6177         pr "      pend++;\n";
6178         pr "    }\n";
6179         pr "\n";
6180         pr "    while (*p && c_isspace (*p))    /* Skip any leading whitespace. */\n";
6181         pr "      p++;\n";
6182         pr "\n";
6183         pr "    if (!*p) {                      /* Empty line?  Skip it. */\n";
6184         pr "      p = pend;\n";
6185         pr "      continue;\n";
6186         pr "    }\n";
6187         pr "\n";
6188         pr "    /* Allocate some space to store this next entry. */\n";
6189         pr "    newp = realloc (ret->guestfs_int_lvm_%s_list_val,\n" typ;
6190         pr "                sizeof (guestfs_int_lvm_%s) * (i+1));\n" typ;
6191         pr "    if (newp == NULL) {\n";
6192         pr "      reply_with_perror (\"realloc\");\n";
6193         pr "      free (ret->guestfs_int_lvm_%s_list_val);\n" typ;
6194         pr "      free (ret);\n";
6195         pr "      free (out);\n";
6196         pr "      return NULL;\n";
6197         pr "    }\n";
6198         pr "    ret->guestfs_int_lvm_%s_list_val = newp;\n" typ;
6199         pr "\n";
6200         pr "    /* Tokenize the next entry. */\n";
6201         pr "    r = lvm_tokenize_%s (p, &ret->guestfs_int_lvm_%s_list_val[i]);\n" typ typ;
6202         pr "    if (r == -1) {\n";
6203         pr "      reply_with_error (\"failed to parse output of '%ss' command\");\n" typ;
6204         pr "      free (ret->guestfs_int_lvm_%s_list_val);\n" typ;
6205         pr "      free (ret);\n";
6206         pr "      free (out);\n";
6207         pr "      return NULL;\n";
6208         pr "    }\n";
6209         pr "\n";
6210         pr "    ++i;\n";
6211         pr "    p = pend;\n";
6212         pr "  }\n";
6213         pr "\n";
6214         pr "  ret->guestfs_int_lvm_%s_list_len = i;\n" typ;
6215         pr "\n";
6216         pr "  free (out);\n";
6217         pr "  return ret;\n";
6218         pr "}\n"
6219
6220   ) ["pv", lvm_pv_cols; "vg", lvm_vg_cols; "lv", lvm_lv_cols]
6221
6222 (* Generate a list of function names, for debugging in the daemon.. *)
6223 and generate_daemon_names () =
6224   generate_header CStyle GPLv2plus;
6225
6226   pr "#include <config.h>\n";
6227   pr "\n";
6228   pr "#include \"daemon.h\"\n";
6229   pr "\n";
6230
6231   pr "/* This array is indexed by proc_nr.  See guestfs_protocol.x. */\n";
6232   pr "const char *function_names[] = {\n";
6233   List.iter (
6234     fun (name, _, proc_nr, _, _, _, _) -> pr "  [%d] = \"%s\",\n" proc_nr name
6235   ) daemon_functions;
6236   pr "};\n";
6237
6238 (* Generate the optional groups for the daemon to implement
6239  * guestfs_available.
6240  *)
6241 and generate_daemon_optgroups_c () =
6242   generate_header CStyle GPLv2plus;
6243
6244   pr "#include <config.h>\n";
6245   pr "\n";
6246   pr "#include \"daemon.h\"\n";
6247   pr "#include \"optgroups.h\"\n";
6248   pr "\n";
6249
6250   pr "struct optgroup optgroups[] = {\n";
6251   List.iter (
6252     fun (group, _) ->
6253       pr "  { \"%s\", optgroup_%s_available },\n" group group
6254   ) optgroups;
6255   pr "  { NULL, NULL }\n";
6256   pr "};\n"
6257
6258 and generate_daemon_optgroups_h () =
6259   generate_header CStyle GPLv2plus;
6260
6261   List.iter (
6262     fun (group, _) ->
6263       pr "extern int optgroup_%s_available (void);\n" group
6264   ) optgroups
6265
6266 (* Generate the tests. *)
6267 and generate_tests () =
6268   generate_header CStyle GPLv2plus;
6269
6270   pr "\
6271 #include <stdio.h>
6272 #include <stdlib.h>
6273 #include <string.h>
6274 #include <unistd.h>
6275 #include <sys/types.h>
6276 #include <fcntl.h>
6277
6278 #include \"guestfs.h\"
6279 #include \"guestfs-internal.h\"
6280
6281 static guestfs_h *g;
6282 static int suppress_error = 0;
6283
6284 static void print_error (guestfs_h *g, void *data, const char *msg)
6285 {
6286   if (!suppress_error)
6287     fprintf (stderr, \"%%s\\n\", msg);
6288 }
6289
6290 /* FIXME: nearly identical code appears in fish.c */
6291 static void print_strings (char *const *argv)
6292 {
6293   int argc;
6294
6295   for (argc = 0; argv[argc] != NULL; ++argc)
6296     printf (\"\\t%%s\\n\", argv[argc]);
6297 }
6298
6299 /*
6300 static void print_table (char const *const *argv)
6301 {
6302   int i;
6303
6304   for (i = 0; argv[i] != NULL; i += 2)
6305     printf (\"%%s: %%s\\n\", argv[i], argv[i+1]);
6306 }
6307 */
6308
6309 ";
6310
6311   (* Generate a list of commands which are not tested anywhere. *)
6312   pr "static void no_test_warnings (void)\n";
6313   pr "{\n";
6314
6315   let hash : (string, bool) Hashtbl.t = Hashtbl.create 13 in
6316   List.iter (
6317     fun (_, _, _, _, tests, _, _) ->
6318       let tests = filter_map (
6319         function
6320         | (_, (Always|If _|Unless _), test) -> Some test
6321         | (_, Disabled, _) -> None
6322       ) tests in
6323       let seq = List.concat (List.map seq_of_test tests) in
6324       let cmds_tested = List.map List.hd seq in
6325       List.iter (fun cmd -> Hashtbl.replace hash cmd true) cmds_tested
6326   ) all_functions;
6327
6328   List.iter (
6329     fun (name, _, _, _, _, _, _) ->
6330       if not (Hashtbl.mem hash name) then
6331         pr "  fprintf (stderr, \"warning: \\\"guestfs_%s\\\" has no tests\\n\");\n" name
6332   ) all_functions;
6333
6334   pr "}\n";
6335   pr "\n";
6336
6337   (* Generate the actual tests.  Note that we generate the tests
6338    * in reverse order, deliberately, so that (in general) the
6339    * newest tests run first.  This makes it quicker and easier to
6340    * debug them.
6341    *)
6342   let test_names =
6343     List.map (
6344       fun (name, _, _, flags, tests, _, _) ->
6345         mapi (generate_one_test name flags) tests
6346     ) (List.rev all_functions) in
6347   let test_names = List.concat test_names in
6348   let nr_tests = List.length test_names in
6349
6350   pr "\
6351 int main (int argc, char *argv[])
6352 {
6353   char c = 0;
6354   unsigned long int n_failed = 0;
6355   const char *filename;
6356   int fd;
6357   int nr_tests, test_num = 0;
6358
6359   setbuf (stdout, NULL);
6360
6361   no_test_warnings ();
6362
6363   g = guestfs_create ();
6364   if (g == NULL) {
6365     printf (\"guestfs_create FAILED\\n\");
6366     exit (EXIT_FAILURE);
6367   }
6368
6369   guestfs_set_error_handler (g, print_error, NULL);
6370
6371   guestfs_set_path (g, \"../appliance\");
6372
6373   filename = \"test1.img\";
6374   fd = open (filename, O_WRONLY|O_CREAT|O_NOCTTY|O_NONBLOCK|O_TRUNC, 0666);
6375   if (fd == -1) {
6376     perror (filename);
6377     exit (EXIT_FAILURE);
6378   }
6379   if (lseek (fd, %d, SEEK_SET) == -1) {
6380     perror (\"lseek\");
6381     close (fd);
6382     unlink (filename);
6383     exit (EXIT_FAILURE);
6384   }
6385   if (write (fd, &c, 1) == -1) {
6386     perror (\"write\");
6387     close (fd);
6388     unlink (filename);
6389     exit (EXIT_FAILURE);
6390   }
6391   if (close (fd) == -1) {
6392     perror (filename);
6393     unlink (filename);
6394     exit (EXIT_FAILURE);
6395   }
6396   if (guestfs_add_drive (g, filename) == -1) {
6397     printf (\"guestfs_add_drive %%s FAILED\\n\", filename);
6398     exit (EXIT_FAILURE);
6399   }
6400
6401   filename = \"test2.img\";
6402   fd = open (filename, O_WRONLY|O_CREAT|O_NOCTTY|O_NONBLOCK|O_TRUNC, 0666);
6403   if (fd == -1) {
6404     perror (filename);
6405     exit (EXIT_FAILURE);
6406   }
6407   if (lseek (fd, %d, SEEK_SET) == -1) {
6408     perror (\"lseek\");
6409     close (fd);
6410     unlink (filename);
6411     exit (EXIT_FAILURE);
6412   }
6413   if (write (fd, &c, 1) == -1) {
6414     perror (\"write\");
6415     close (fd);
6416     unlink (filename);
6417     exit (EXIT_FAILURE);
6418   }
6419   if (close (fd) == -1) {
6420     perror (filename);
6421     unlink (filename);
6422     exit (EXIT_FAILURE);
6423   }
6424   if (guestfs_add_drive (g, filename) == -1) {
6425     printf (\"guestfs_add_drive %%s FAILED\\n\", filename);
6426     exit (EXIT_FAILURE);
6427   }
6428
6429   filename = \"test3.img\";
6430   fd = open (filename, O_WRONLY|O_CREAT|O_NOCTTY|O_NONBLOCK|O_TRUNC, 0666);
6431   if (fd == -1) {
6432     perror (filename);
6433     exit (EXIT_FAILURE);
6434   }
6435   if (lseek (fd, %d, SEEK_SET) == -1) {
6436     perror (\"lseek\");
6437     close (fd);
6438     unlink (filename);
6439     exit (EXIT_FAILURE);
6440   }
6441   if (write (fd, &c, 1) == -1) {
6442     perror (\"write\");
6443     close (fd);
6444     unlink (filename);
6445     exit (EXIT_FAILURE);
6446   }
6447   if (close (fd) == -1) {
6448     perror (filename);
6449     unlink (filename);
6450     exit (EXIT_FAILURE);
6451   }
6452   if (guestfs_add_drive (g, filename) == -1) {
6453     printf (\"guestfs_add_drive %%s FAILED\\n\", filename);
6454     exit (EXIT_FAILURE);
6455   }
6456
6457   if (guestfs_add_drive_ro (g, \"../images/test.iso\") == -1) {
6458     printf (\"guestfs_add_drive_ro ../images/test.iso FAILED\\n\");
6459     exit (EXIT_FAILURE);
6460   }
6461
6462   /* Set a timeout in case qemu hangs during launch (RHBZ#505329). */
6463   alarm (600);
6464
6465   if (guestfs_launch (g) == -1) {
6466     printf (\"guestfs_launch FAILED\\n\");
6467     exit (EXIT_FAILURE);
6468   }
6469
6470   /* Cancel previous alarm. */
6471   alarm (0);
6472
6473   nr_tests = %d;
6474
6475 " (500 * 1024 * 1024) (50 * 1024 * 1024) (10 * 1024 * 1024) nr_tests;
6476
6477   iteri (
6478     fun i test_name ->
6479       pr "  test_num++;\n";
6480       pr "  printf (\"%%3d/%%3d %s\\n\", test_num, nr_tests);\n" test_name;
6481       pr "  if (%s () == -1) {\n" test_name;
6482       pr "    printf (\"%s FAILED\\n\");\n" test_name;
6483       pr "    n_failed++;\n";
6484       pr "  }\n";
6485   ) test_names;
6486   pr "\n";
6487
6488   pr "  guestfs_close (g);\n";
6489   pr "  unlink (\"test1.img\");\n";
6490   pr "  unlink (\"test2.img\");\n";
6491   pr "  unlink (\"test3.img\");\n";
6492   pr "\n";
6493
6494   pr "  if (n_failed > 0) {\n";
6495   pr "    printf (\"***** %%lu / %%d tests FAILED *****\\n\", n_failed, nr_tests);\n";
6496   pr "    exit (EXIT_FAILURE);\n";
6497   pr "  }\n";
6498   pr "\n";
6499
6500   pr "  exit (EXIT_SUCCESS);\n";
6501   pr "}\n"
6502
6503 and generate_one_test name flags i (init, prereq, test) =
6504   let test_name = sprintf "test_%s_%d" name i in
6505
6506   pr "\
6507 static int %s_skip (void)
6508 {
6509   const char *str;
6510
6511   str = getenv (\"TEST_ONLY\");
6512   if (str)
6513     return strstr (str, \"%s\") == NULL;
6514   str = getenv (\"SKIP_%s\");
6515   if (str && STREQ (str, \"1\")) return 1;
6516   str = getenv (\"SKIP_TEST_%s\");
6517   if (str && STREQ (str, \"1\")) return 1;
6518   return 0;
6519 }
6520
6521 " test_name name (String.uppercase test_name) (String.uppercase name);
6522
6523   (match prereq with
6524    | Disabled | Always -> ()
6525    | If code | Unless code ->
6526        pr "static int %s_prereq (void)\n" test_name;
6527        pr "{\n";
6528        pr "  %s\n" code;
6529        pr "}\n";
6530        pr "\n";
6531   );
6532
6533   pr "\
6534 static int %s (void)
6535 {
6536   if (%s_skip ()) {
6537     printf (\"        %%s skipped (reason: environment variable set)\\n\", \"%s\");
6538     return 0;
6539   }
6540
6541 " test_name test_name test_name;
6542
6543   (* Optional functions should only be tested if the relevant
6544    * support is available in the daemon.
6545    *)
6546   List.iter (
6547     function
6548     | Optional group ->
6549         pr "  {\n";
6550         pr "    const char *groups[] = { \"%s\", NULL };\n" group;
6551         pr "    int r;\n";
6552         pr "    suppress_error = 1;\n";
6553         pr "    r = guestfs_available (g, (char **) groups);\n";
6554         pr "    suppress_error = 0;\n";
6555         pr "    if (r == -1) {\n";
6556         pr "      printf (\"        %%s skipped (reason: group %%s not available in daemon)\\n\", \"%s\", groups[0]);\n" test_name;
6557         pr "      return 0;\n";
6558         pr "    }\n";
6559         pr "  }\n";
6560     | _ -> ()
6561   ) flags;
6562
6563   (match prereq with
6564    | Disabled ->
6565        pr "  printf (\"        %%s skipped (reason: test disabled in generator)\\n\", \"%s\");\n" test_name
6566    | If _ ->
6567        pr "  if (! %s_prereq ()) {\n" test_name;
6568        pr "    printf (\"        %%s skipped (reason: test prerequisite)\\n\", \"%s\");\n" test_name;
6569        pr "    return 0;\n";
6570        pr "  }\n";
6571        pr "\n";
6572        generate_one_test_body name i test_name init test;
6573    | Unless _ ->
6574        pr "  if (%s_prereq ()) {\n" test_name;
6575        pr "    printf (\"        %%s skipped (reason: test prerequisite)\\n\", \"%s\");\n" test_name;
6576        pr "    return 0;\n";
6577        pr "  }\n";
6578        pr "\n";
6579        generate_one_test_body name i test_name init test;
6580    | Always ->
6581        generate_one_test_body name i test_name init test
6582   );
6583
6584   pr "  return 0;\n";
6585   pr "}\n";
6586   pr "\n";
6587   test_name
6588
6589 and generate_one_test_body name i test_name init test =
6590   (match init with
6591    | InitNone (* XXX at some point, InitNone and InitEmpty became
6592                * folded together as the same thing.  Really we should
6593                * make InitNone do nothing at all, but the tests may
6594                * need to be checked to make sure this is OK.
6595                *)
6596    | InitEmpty ->
6597        pr "  /* InitNone|InitEmpty for %s */\n" test_name;
6598        List.iter (generate_test_command_call test_name)
6599          [["blockdev_setrw"; "/dev/sda"];
6600           ["umount_all"];
6601           ["lvm_remove_all"]]
6602    | InitPartition ->
6603        pr "  /* InitPartition for %s: create /dev/sda1 */\n" test_name;
6604        List.iter (generate_test_command_call test_name)
6605          [["blockdev_setrw"; "/dev/sda"];
6606           ["umount_all"];
6607           ["lvm_remove_all"];
6608           ["part_disk"; "/dev/sda"; "mbr"]]
6609    | InitBasicFS ->
6610        pr "  /* InitBasicFS for %s: create ext2 on /dev/sda1 */\n" test_name;
6611        List.iter (generate_test_command_call test_name)
6612          [["blockdev_setrw"; "/dev/sda"];
6613           ["umount_all"];
6614           ["lvm_remove_all"];
6615           ["part_disk"; "/dev/sda"; "mbr"];
6616           ["mkfs"; "ext2"; "/dev/sda1"];
6617           ["mount_options"; ""; "/dev/sda1"; "/"]]
6618    | InitBasicFSonLVM ->
6619        pr "  /* InitBasicFSonLVM for %s: create ext2 on /dev/VG/LV */\n"
6620          test_name;
6621        List.iter (generate_test_command_call test_name)
6622          [["blockdev_setrw"; "/dev/sda"];
6623           ["umount_all"];
6624           ["lvm_remove_all"];
6625           ["part_disk"; "/dev/sda"; "mbr"];
6626           ["pvcreate"; "/dev/sda1"];
6627           ["vgcreate"; "VG"; "/dev/sda1"];
6628           ["lvcreate"; "LV"; "VG"; "8"];
6629           ["mkfs"; "ext2"; "/dev/VG/LV"];
6630           ["mount_options"; ""; "/dev/VG/LV"; "/"]]
6631    | InitISOFS ->
6632        pr "  /* InitISOFS for %s */\n" test_name;
6633        List.iter (generate_test_command_call test_name)
6634          [["blockdev_setrw"; "/dev/sda"];
6635           ["umount_all"];
6636           ["lvm_remove_all"];
6637           ["mount_ro"; "/dev/sdd"; "/"]]
6638   );
6639
6640   let get_seq_last = function
6641     | [] ->
6642         failwithf "%s: you cannot use [] (empty list) when expecting a command"
6643           test_name
6644     | seq ->
6645         let seq = List.rev seq in
6646         List.rev (List.tl seq), List.hd seq
6647   in
6648
6649   match test with
6650   | TestRun seq ->
6651       pr "  /* TestRun for %s (%d) */\n" name i;
6652       List.iter (generate_test_command_call test_name) seq
6653   | TestOutput (seq, expected) ->
6654       pr "  /* TestOutput for %s (%d) */\n" name i;
6655       pr "  const char *expected = \"%s\";\n" (c_quote expected);
6656       let seq, last = get_seq_last seq in
6657       let test () =
6658         pr "    if (STRNEQ (r, expected)) {\n";
6659         pr "      fprintf (stderr, \"%s: expected \\\"%%s\\\" but got \\\"%%s\\\"\\n\", expected, r);\n" test_name;
6660         pr "      return -1;\n";
6661         pr "    }\n"
6662       in
6663       List.iter (generate_test_command_call test_name) seq;
6664       generate_test_command_call ~test test_name last
6665   | TestOutputList (seq, expected) ->
6666       pr "  /* TestOutputList for %s (%d) */\n" name i;
6667       let seq, last = get_seq_last seq in
6668       let test () =
6669         iteri (
6670           fun i str ->
6671             pr "    if (!r[%d]) {\n" i;
6672             pr "      fprintf (stderr, \"%s: short list returned from command\\n\");\n" test_name;
6673             pr "      print_strings (r);\n";
6674             pr "      return -1;\n";
6675             pr "    }\n";
6676             pr "    {\n";
6677             pr "      const char *expected = \"%s\";\n" (c_quote str);
6678             pr "      if (STRNEQ (r[%d], expected)) {\n" i;
6679             pr "        fprintf (stderr, \"%s: expected \\\"%%s\\\" but got \\\"%%s\\\"\\n\", expected, r[%d]);\n" test_name i;
6680             pr "        return -1;\n";
6681             pr "      }\n";
6682             pr "    }\n"
6683         ) expected;
6684         pr "    if (r[%d] != NULL) {\n" (List.length expected);
6685         pr "      fprintf (stderr, \"%s: extra elements returned from command\\n\");\n"
6686           test_name;
6687         pr "      print_strings (r);\n";
6688         pr "      return -1;\n";
6689         pr "    }\n"
6690       in
6691       List.iter (generate_test_command_call test_name) seq;
6692       generate_test_command_call ~test test_name last
6693   | TestOutputListOfDevices (seq, expected) ->
6694       pr "  /* TestOutputListOfDevices for %s (%d) */\n" name i;
6695       let seq, last = get_seq_last seq in
6696       let test () =
6697         iteri (
6698           fun i str ->
6699             pr "    if (!r[%d]) {\n" i;
6700             pr "      fprintf (stderr, \"%s: short list returned from command\\n\");\n" test_name;
6701             pr "      print_strings (r);\n";
6702             pr "      return -1;\n";
6703             pr "    }\n";
6704             pr "    {\n";
6705             pr "      const char *expected = \"%s\";\n" (c_quote str);
6706             pr "      r[%d][5] = 's';\n" i;
6707             pr "      if (STRNEQ (r[%d], expected)) {\n" i;
6708             pr "        fprintf (stderr, \"%s: expected \\\"%%s\\\" but got \\\"%%s\\\"\\n\", expected, r[%d]);\n" test_name i;
6709             pr "        return -1;\n";
6710             pr "      }\n";
6711             pr "    }\n"
6712         ) expected;
6713         pr "    if (r[%d] != NULL) {\n" (List.length expected);
6714         pr "      fprintf (stderr, \"%s: extra elements returned from command\\n\");\n"
6715           test_name;
6716         pr "      print_strings (r);\n";
6717         pr "      return -1;\n";
6718         pr "    }\n"
6719       in
6720       List.iter (generate_test_command_call test_name) seq;
6721       generate_test_command_call ~test test_name last
6722   | TestOutputInt (seq, expected) ->
6723       pr "  /* TestOutputInt for %s (%d) */\n" name i;
6724       let seq, last = get_seq_last seq in
6725       let test () =
6726         pr "    if (r != %d) {\n" expected;
6727         pr "      fprintf (stderr, \"%s: expected %d but got %%d\\n\","
6728           test_name expected;
6729         pr "               (int) r);\n";
6730         pr "      return -1;\n";
6731         pr "    }\n"
6732       in
6733       List.iter (generate_test_command_call test_name) seq;
6734       generate_test_command_call ~test test_name last
6735   | TestOutputIntOp (seq, op, expected) ->
6736       pr "  /* TestOutputIntOp for %s (%d) */\n" name i;
6737       let seq, last = get_seq_last seq in
6738       let test () =
6739         pr "    if (! (r %s %d)) {\n" op expected;
6740         pr "      fprintf (stderr, \"%s: expected %s %d but got %%d\\n\","
6741           test_name op expected;
6742         pr "               (int) r);\n";
6743         pr "      return -1;\n";
6744         pr "    }\n"
6745       in
6746       List.iter (generate_test_command_call test_name) seq;
6747       generate_test_command_call ~test test_name last
6748   | TestOutputTrue seq ->
6749       pr "  /* TestOutputTrue for %s (%d) */\n" name i;
6750       let seq, last = get_seq_last seq in
6751       let test () =
6752         pr "    if (!r) {\n";
6753         pr "      fprintf (stderr, \"%s: expected true, got false\\n\");\n"
6754           test_name;
6755         pr "      return -1;\n";
6756         pr "    }\n"
6757       in
6758       List.iter (generate_test_command_call test_name) seq;
6759       generate_test_command_call ~test test_name last
6760   | TestOutputFalse seq ->
6761       pr "  /* TestOutputFalse for %s (%d) */\n" name i;
6762       let seq, last = get_seq_last seq in
6763       let test () =
6764         pr "    if (r) {\n";
6765         pr "      fprintf (stderr, \"%s: expected false, got true\\n\");\n"
6766           test_name;
6767         pr "      return -1;\n";
6768         pr "    }\n"
6769       in
6770       List.iter (generate_test_command_call test_name) seq;
6771       generate_test_command_call ~test test_name last
6772   | TestOutputLength (seq, expected) ->
6773       pr "  /* TestOutputLength for %s (%d) */\n" name i;
6774       let seq, last = get_seq_last seq in
6775       let test () =
6776         pr "    int j;\n";
6777         pr "    for (j = 0; j < %d; ++j)\n" expected;
6778         pr "      if (r[j] == NULL) {\n";
6779         pr "        fprintf (stderr, \"%s: short list returned\\n\");\n"
6780           test_name;
6781         pr "        print_strings (r);\n";
6782         pr "        return -1;\n";
6783         pr "      }\n";
6784         pr "    if (r[j] != NULL) {\n";
6785         pr "      fprintf (stderr, \"%s: long list returned\\n\");\n"
6786           test_name;
6787         pr "      print_strings (r);\n";
6788         pr "      return -1;\n";
6789         pr "    }\n"
6790       in
6791       List.iter (generate_test_command_call test_name) seq;
6792       generate_test_command_call ~test test_name last
6793   | TestOutputBuffer (seq, expected) ->
6794       pr "  /* TestOutputBuffer for %s (%d) */\n" name i;
6795       pr "  const char *expected = \"%s\";\n" (c_quote expected);
6796       let seq, last = get_seq_last seq in
6797       let len = String.length expected in
6798       let test () =
6799         pr "    if (size != %d) {\n" len;
6800         pr "      fprintf (stderr, \"%s: returned size of buffer wrong, expected %d but got %%zu\\n\", size);\n" test_name len;
6801         pr "      return -1;\n";
6802         pr "    }\n";
6803         pr "    if (STRNEQLEN (r, expected, size)) {\n";
6804         pr "      fprintf (stderr, \"%s: expected \\\"%%s\\\" but got \\\"%%s\\\"\\n\", expected, r);\n" test_name;
6805         pr "      return -1;\n";
6806         pr "    }\n"
6807       in
6808       List.iter (generate_test_command_call test_name) seq;
6809       generate_test_command_call ~test test_name last
6810   | TestOutputStruct (seq, checks) ->
6811       pr "  /* TestOutputStruct for %s (%d) */\n" name i;
6812       let seq, last = get_seq_last seq in
6813       let test () =
6814         List.iter (
6815           function
6816           | CompareWithInt (field, expected) ->
6817               pr "    if (r->%s != %d) {\n" field expected;
6818               pr "      fprintf (stderr, \"%s: %s was %%d, expected %d\\n\",\n"
6819                 test_name field expected;
6820               pr "               (int) r->%s);\n" field;
6821               pr "      return -1;\n";
6822               pr "    }\n"
6823           | CompareWithIntOp (field, op, expected) ->
6824               pr "    if (!(r->%s %s %d)) {\n" field op expected;
6825               pr "      fprintf (stderr, \"%s: %s was %%d, expected %s %d\\n\",\n"
6826                 test_name field op expected;
6827               pr "               (int) r->%s);\n" field;
6828               pr "      return -1;\n";
6829               pr "    }\n"
6830           | CompareWithString (field, expected) ->
6831               pr "    if (STRNEQ (r->%s, \"%s\")) {\n" field expected;
6832               pr "      fprintf (stderr, \"%s: %s was \"%%s\", expected \"%s\"\\n\",\n"
6833                 test_name field expected;
6834               pr "               r->%s);\n" field;
6835               pr "      return -1;\n";
6836               pr "    }\n"
6837           | CompareFieldsIntEq (field1, field2) ->
6838               pr "    if (r->%s != r->%s) {\n" field1 field2;
6839               pr "      fprintf (stderr, \"%s: %s (%%d) <> %s (%%d)\\n\",\n"
6840                 test_name field1 field2;
6841               pr "               (int) r->%s, (int) r->%s);\n" field1 field2;
6842               pr "      return -1;\n";
6843               pr "    }\n"
6844           | CompareFieldsStrEq (field1, field2) ->
6845               pr "    if (STRNEQ (r->%s, r->%s)) {\n" field1 field2;
6846               pr "      fprintf (stderr, \"%s: %s (\"%%s\") <> %s (\"%%s\")\\n\",\n"
6847                 test_name field1 field2;
6848               pr "               r->%s, r->%s);\n" field1 field2;
6849               pr "      return -1;\n";
6850               pr "    }\n"
6851         ) checks
6852       in
6853       List.iter (generate_test_command_call test_name) seq;
6854       generate_test_command_call ~test test_name last
6855   | TestLastFail seq ->
6856       pr "  /* TestLastFail for %s (%d) */\n" name i;
6857       let seq, last = get_seq_last seq in
6858       List.iter (generate_test_command_call test_name) seq;
6859       generate_test_command_call test_name ~expect_error:true last
6860
6861 (* Generate the code to run a command, leaving the result in 'r'.
6862  * If you expect to get an error then you should set expect_error:true.
6863  *)
6864 and generate_test_command_call ?(expect_error = false) ?test test_name cmd =
6865   match cmd with
6866   | [] -> assert false
6867   | name :: args ->
6868       (* Look up the command to find out what args/ret it has. *)
6869       let style =
6870         try
6871           let _, style, _, _, _, _, _ =
6872             List.find (fun (n, _, _, _, _, _, _) -> n = name) all_functions in
6873           style
6874         with Not_found ->
6875           failwithf "%s: in test, command %s was not found" test_name name in
6876
6877       if List.length (snd style) <> List.length args then
6878         failwithf "%s: in test, wrong number of args given to %s"
6879           test_name name;
6880
6881       pr "  {\n";
6882
6883       List.iter (
6884         function
6885         | OptString n, "NULL" -> ()
6886         | Pathname n, arg
6887         | Device n, arg
6888         | Dev_or_Path n, arg
6889         | String n, arg
6890         | OptString n, arg ->
6891             pr "    const char *%s = \"%s\";\n" n (c_quote arg);
6892         | Int _, _
6893         | Int64 _, _
6894         | Bool _, _
6895         | FileIn _, _ | FileOut _, _ -> ()
6896         | StringList n, "" | DeviceList n, "" ->
6897             pr "    const char *const %s[1] = { NULL };\n" n
6898         | StringList n, arg | DeviceList n, arg ->
6899             let strs = string_split " " arg in
6900             iteri (
6901               fun i str ->
6902                 pr "    const char *%s_%d = \"%s\";\n" n i (c_quote str);
6903             ) strs;
6904             pr "    const char *const %s[] = {\n" n;
6905             iteri (
6906               fun i _ -> pr "      %s_%d,\n" n i
6907             ) strs;
6908             pr "      NULL\n";
6909             pr "    };\n";
6910       ) (List.combine (snd style) args);
6911
6912       let error_code =
6913         match fst style with
6914         | RErr | RInt _ | RBool _ -> pr "    int r;\n"; "-1"
6915         | RInt64 _ -> pr "    int64_t r;\n"; "-1"
6916         | RConstString _ | RConstOptString _ ->
6917             pr "    const char *r;\n"; "NULL"
6918         | RString _ -> pr "    char *r;\n"; "NULL"
6919         | RStringList _ | RHashtable _ ->
6920             pr "    char **r;\n";
6921             pr "    int i;\n";
6922             "NULL"
6923         | RStruct (_, typ) ->
6924             pr "    struct guestfs_%s *r;\n" typ; "NULL"
6925         | RStructList (_, typ) ->
6926             pr "    struct guestfs_%s_list *r;\n" typ; "NULL"
6927         | RBufferOut _ ->
6928             pr "    char *r;\n";
6929             pr "    size_t size;\n";
6930             "NULL" in
6931
6932       pr "    suppress_error = %d;\n" (if expect_error then 1 else 0);
6933       pr "    r = guestfs_%s (g" name;
6934
6935       (* Generate the parameters. *)
6936       List.iter (
6937         function
6938         | OptString _, "NULL" -> pr ", NULL"
6939         | Pathname n, _
6940         | Device n, _ | Dev_or_Path n, _
6941         | String n, _
6942         | OptString n, _ ->
6943             pr ", %s" n
6944         | FileIn _, arg | FileOut _, arg ->
6945             pr ", \"%s\"" (c_quote arg)
6946         | StringList n, _ | DeviceList n, _ ->
6947             pr ", (char **) %s" n
6948         | Int _, arg ->
6949             let i =
6950               try int_of_string arg
6951               with Failure "int_of_string" ->
6952                 failwithf "%s: expecting an int, but got '%s'" test_name arg in
6953             pr ", %d" i
6954         | Int64 _, arg ->
6955             let i =
6956               try Int64.of_string arg
6957               with Failure "int_of_string" ->
6958                 failwithf "%s: expecting an int64, but got '%s'" test_name arg in
6959             pr ", %Ld" i
6960         | Bool _, arg ->
6961             let b = bool_of_string arg in pr ", %d" (if b then 1 else 0)
6962       ) (List.combine (snd style) args);
6963
6964       (match fst style with
6965        | RBufferOut _ -> pr ", &size"
6966        | _ -> ()
6967       );
6968
6969       pr ");\n";
6970
6971       if not expect_error then
6972         pr "    if (r == %s)\n" error_code
6973       else
6974         pr "    if (r != %s)\n" error_code;
6975       pr "      return -1;\n";
6976
6977       (* Insert the test code. *)
6978       (match test with
6979        | None -> ()
6980        | Some f -> f ()
6981       );
6982
6983       (match fst style with
6984        | RErr | RInt _ | RInt64 _ | RBool _
6985        | RConstString _ | RConstOptString _ -> ()
6986        | RString _ | RBufferOut _ -> pr "    free (r);\n"
6987        | RStringList _ | RHashtable _ ->
6988            pr "    for (i = 0; r[i] != NULL; ++i)\n";
6989            pr "      free (r[i]);\n";
6990            pr "    free (r);\n"
6991        | RStruct (_, typ) ->
6992            pr "    guestfs_free_%s (r);\n" typ
6993        | RStructList (_, typ) ->
6994            pr "    guestfs_free_%s_list (r);\n" typ
6995       );
6996
6997       pr "  }\n"
6998
6999 and c_quote str =
7000   let str = replace_str str "\r" "\\r" in
7001   let str = replace_str str "\n" "\\n" in
7002   let str = replace_str str "\t" "\\t" in
7003   let str = replace_str str "\000" "\\0" in
7004   str
7005
7006 (* Generate a lot of different functions for guestfish. *)
7007 and generate_fish_cmds () =
7008   generate_header CStyle GPLv2plus;
7009
7010   let all_functions =
7011     List.filter (
7012       fun (_, _, _, flags, _, _, _) -> not (List.mem NotInFish flags)
7013     ) all_functions in
7014   let all_functions_sorted =
7015     List.filter (
7016       fun (_, _, _, flags, _, _, _) -> not (List.mem NotInFish flags)
7017     ) all_functions_sorted in
7018
7019   pr "#include <config.h>\n";
7020   pr "\n";
7021   pr "#include <stdio.h>\n";
7022   pr "#include <stdlib.h>\n";
7023   pr "#include <string.h>\n";
7024   pr "#include <inttypes.h>\n";
7025   pr "\n";
7026   pr "#include <guestfs.h>\n";
7027   pr "#include \"c-ctype.h\"\n";
7028   pr "#include \"full-write.h\"\n";
7029   pr "#include \"xstrtol.h\"\n";
7030   pr "#include \"fish.h\"\n";
7031   pr "\n";
7032
7033   (* list_commands function, which implements guestfish -h *)
7034   pr "void list_commands (void)\n";
7035   pr "{\n";
7036   pr "  printf (\"    %%-16s     %%s\\n\", _(\"Command\"), _(\"Description\"));\n";
7037   pr "  list_builtin_commands ();\n";
7038   List.iter (
7039     fun (name, _, _, flags, _, shortdesc, _) ->
7040       let name = replace_char name '_' '-' in
7041       pr "  printf (\"%%-20s %%s\\n\", \"%s\", _(\"%s\"));\n"
7042         name shortdesc
7043   ) all_functions_sorted;
7044   pr "  printf (\"    %%s\\n\",";
7045   pr "          _(\"Use -h <cmd> / help <cmd> to show detailed help for a command.\"));\n";
7046   pr "}\n";
7047   pr "\n";
7048
7049   (* display_command function, which implements guestfish -h cmd *)
7050   pr "void display_command (const char *cmd)\n";
7051   pr "{\n";
7052   List.iter (
7053     fun (name, style, _, flags, _, shortdesc, longdesc) ->
7054       let name2 = replace_char name '_' '-' in
7055       let alias =
7056         try find_map (function FishAlias n -> Some n | _ -> None) flags
7057         with Not_found -> name in
7058       let longdesc = replace_str longdesc "C<guestfs_" "C<" in
7059       let synopsis =
7060         match snd style with
7061         | [] -> name2
7062         | args ->
7063             sprintf "%s %s"
7064               name2 (String.concat " " (List.map name_of_argt args)) in
7065
7066       let warnings =
7067         if List.mem ProtocolLimitWarning flags then
7068           ("\n\n" ^ protocol_limit_warning)
7069         else "" in
7070
7071       (* For DangerWillRobinson commands, we should probably have
7072        * guestfish prompt before allowing you to use them (especially
7073        * in interactive mode). XXX
7074        *)
7075       let warnings =
7076         warnings ^
7077           if List.mem DangerWillRobinson flags then
7078             ("\n\n" ^ danger_will_robinson)
7079           else "" in
7080
7081       let warnings =
7082         warnings ^
7083           match deprecation_notice flags with
7084           | None -> ""
7085           | Some txt -> "\n\n" ^ txt in
7086
7087       let describe_alias =
7088         if name <> alias then
7089           sprintf "\n\nYou can use '%s' as an alias for this command." alias
7090         else "" in
7091
7092       pr "  if (";
7093       pr "STRCASEEQ (cmd, \"%s\")" name;
7094       if name <> name2 then
7095         pr " || STRCASEEQ (cmd, \"%s\")" name2;
7096       if name <> alias then
7097         pr " || STRCASEEQ (cmd, \"%s\")" alias;
7098       pr ")\n";
7099       pr "    pod2text (\"%s\", _(\"%s\"), %S);\n"
7100         name2 shortdesc
7101         ("=head1 SYNOPSIS\n\n " ^ synopsis ^ "\n\n" ^
7102          "=head1 DESCRIPTION\n\n" ^
7103          longdesc ^ warnings ^ describe_alias);
7104       pr "  else\n"
7105   ) all_functions;
7106   pr "    display_builtin_command (cmd);\n";
7107   pr "}\n";
7108   pr "\n";
7109
7110   let emit_print_list_function typ =
7111     pr "static void print_%s_list (struct guestfs_%s_list *%ss)\n"
7112       typ typ typ;
7113     pr "{\n";
7114     pr "  unsigned int i;\n";
7115     pr "\n";
7116     pr "  for (i = 0; i < %ss->len; ++i) {\n" typ;
7117     pr "    printf (\"[%%d] = {\\n\", i);\n";
7118     pr "    print_%s_indent (&%ss->val[i], \"  \");\n" typ typ;
7119     pr "    printf (\"}\\n\");\n";
7120     pr "  }\n";
7121     pr "}\n";
7122     pr "\n";
7123   in
7124
7125   (* print_* functions *)
7126   List.iter (
7127     fun (typ, cols) ->
7128       let needs_i =
7129         List.exists (function (_, (FUUID|FBuffer)) -> true | _ -> false) cols in
7130
7131       pr "static void print_%s_indent (struct guestfs_%s *%s, const char *indent)\n" typ typ typ;
7132       pr "{\n";
7133       if needs_i then (
7134         pr "  unsigned int i;\n";
7135         pr "\n"
7136       );
7137       List.iter (
7138         function
7139         | name, FString ->
7140             pr "  printf (\"%%s%s: %%s\\n\", indent, %s->%s);\n" name typ name
7141         | name, FUUID ->
7142             pr "  printf (\"%%s%s: \", indent);\n" name;
7143             pr "  for (i = 0; i < 32; ++i)\n";
7144             pr "    printf (\"%%c\", %s->%s[i]);\n" typ name;
7145             pr "  printf (\"\\n\");\n"
7146         | name, FBuffer ->
7147             pr "  printf (\"%%s%s: \", indent);\n" name;
7148             pr "  for (i = 0; i < %s->%s_len; ++i)\n" typ name;
7149             pr "    if (c_isprint (%s->%s[i]))\n" typ name;
7150             pr "      printf (\"%%c\", %s->%s[i]);\n" typ name;
7151             pr "    else\n";
7152             pr "      printf (\"\\\\x%%02x\", %s->%s[i]);\n" typ name;
7153             pr "  printf (\"\\n\");\n"
7154         | name, (FUInt64|FBytes) ->
7155             pr "  printf (\"%%s%s: %%\" PRIu64 \"\\n\", indent, %s->%s);\n"
7156               name typ name
7157         | name, FInt64 ->
7158             pr "  printf (\"%%s%s: %%\" PRIi64 \"\\n\", indent, %s->%s);\n"
7159               name typ name
7160         | name, FUInt32 ->
7161             pr "  printf (\"%%s%s: %%\" PRIu32 \"\\n\", indent, %s->%s);\n"
7162               name typ name
7163         | name, FInt32 ->
7164             pr "  printf (\"%%s%s: %%\" PRIi32 \"\\n\", indent, %s->%s);\n"
7165               name typ name
7166         | name, FChar ->
7167             pr "  printf (\"%%s%s: %%c\\n\", indent, %s->%s);\n"
7168               name typ name
7169         | name, FOptPercent ->
7170             pr "  if (%s->%s >= 0) printf (\"%%s%s: %%g %%%%\\n\", indent, %s->%s);\n"
7171               typ name name typ name;
7172             pr "  else printf (\"%%s%s: \\n\", indent);\n" name
7173       ) cols;
7174       pr "}\n";
7175       pr "\n";
7176   ) structs;
7177
7178   (* Emit a print_TYPE_list function definition only if that function is used. *)
7179   List.iter (
7180     function
7181     | typ, (RStructListOnly | RStructAndList) ->
7182         (* generate the function for typ *)
7183         emit_print_list_function typ
7184     | typ, _ -> () (* empty *)
7185   ) (rstructs_used_by all_functions);
7186
7187   (* Emit a print_TYPE function definition only if that function is used. *)
7188   List.iter (
7189     function
7190     | typ, (RStructOnly | RStructAndList) ->
7191         pr "static void print_%s (struct guestfs_%s *%s)\n" typ typ typ;
7192         pr "{\n";
7193         pr "  print_%s_indent (%s, \"\");\n" typ typ;
7194         pr "}\n";
7195         pr "\n";
7196     | typ, _ -> () (* empty *)
7197   ) (rstructs_used_by all_functions);
7198
7199   (* run_<action> actions *)
7200   List.iter (
7201     fun (name, style, _, flags, _, _, _) ->
7202       pr "static int run_%s (const char *cmd, int argc, char *argv[])\n" name;
7203       pr "{\n";
7204       (match fst style with
7205        | RErr
7206        | RInt _
7207        | RBool _ -> pr "  int r;\n"
7208        | RInt64 _ -> pr "  int64_t r;\n"
7209        | RConstString _ | RConstOptString _ -> pr "  const char *r;\n"
7210        | RString _ -> pr "  char *r;\n"
7211        | RStringList _ | RHashtable _ -> pr "  char **r;\n"
7212        | RStruct (_, typ) -> pr "  struct guestfs_%s *r;\n" typ
7213        | RStructList (_, typ) -> pr "  struct guestfs_%s_list *r;\n" typ
7214        | RBufferOut _ ->
7215            pr "  char *r;\n";
7216            pr "  size_t size;\n";
7217       );
7218       List.iter (
7219         function
7220         | Device n
7221         | String n
7222         | OptString n
7223         | FileIn n
7224         | FileOut n -> pr "  const char *%s;\n" n
7225         | Pathname n
7226         | Dev_or_Path n -> pr "  char *%s;\n" n
7227         | StringList n | DeviceList n -> pr "  char **%s;\n" n
7228         | Bool n -> pr "  int %s;\n" n
7229         | Int n -> pr "  int %s;\n" n
7230         | Int64 n -> pr "  int64_t %s;\n" n
7231       ) (snd style);
7232
7233       (* Check and convert parameters. *)
7234       let argc_expected = List.length (snd style) in
7235       pr "  if (argc != %d) {\n" argc_expected;
7236       pr "    fprintf (stderr, _(\"%%s should have %%d parameter(s)\\n\"), cmd, %d);\n"
7237         argc_expected;
7238       pr "    fprintf (stderr, _(\"type 'help %%s' for help on %%s\\n\"), cmd, cmd);\n";
7239       pr "    return -1;\n";
7240       pr "  }\n";
7241
7242       let parse_integer fn fntyp rtyp range name i =
7243         pr "  {\n";
7244         pr "    strtol_error xerr;\n";
7245         pr "    %s r;\n" fntyp;
7246         pr "\n";
7247         pr "    xerr = %s (argv[%d], NULL, 0, &r, \"\");\n" fn i;
7248         pr "    if (xerr != LONGINT_OK) {\n";
7249         pr "      fprintf (stderr,\n";
7250         pr "               _(\"%%s: %%s: invalid integer parameter (%%s returned %%d)\\n\"),\n";
7251         pr "               cmd, \"%s\", \"%s\", xerr);\n" name fn;
7252         pr "      return -1;\n";
7253         pr "    }\n";
7254         (match range with
7255          | None -> ()
7256          | Some (min, max, comment) ->
7257              pr "    /* %s */\n" comment;
7258              pr "    if (r < %s || r > %s) {\n" min max;
7259              pr "      fprintf (stderr, _(\"%%s: %%s: integer out of range\\n\"), cmd, \"%s\");\n"
7260                name;
7261              pr "      return -1;\n";
7262              pr "    }\n";
7263              pr "    /* The check above should ensure this assignment does not overflow. */\n";
7264         );
7265         pr "    %s = r;\n" name;
7266         pr "  }\n";
7267       in
7268
7269       iteri (
7270         fun i ->
7271           function
7272           | Device name
7273           | String name ->
7274               pr "  %s = argv[%d];\n" name i
7275           | Pathname name
7276           | Dev_or_Path name ->
7277               pr "  %s = resolve_win_path (argv[%d]);\n" name i;
7278               pr "  if (%s == NULL) return -1;\n" name
7279           | OptString name ->
7280               pr "  %s = STRNEQ (argv[%d], \"\") ? argv[%d] : NULL;\n"
7281                 name i i
7282           | FileIn name ->
7283               pr "  %s = STRNEQ (argv[%d], \"-\") ? argv[%d] : \"/dev/stdin\";\n"
7284                 name i i
7285           | FileOut name ->
7286               pr "  %s = STRNEQ (argv[%d], \"-\") ? argv[%d] : \"/dev/stdout\";\n"
7287                 name i i
7288           | StringList name | DeviceList name ->
7289               pr "  %s = parse_string_list (argv[%d]);\n" name i;
7290               pr "  if (%s == NULL) return -1;\n" name;
7291           | Bool name ->
7292               pr "  %s = is_true (argv[%d]) ? 1 : 0;\n" name i
7293           | Int name ->
7294               let range =
7295                 let min = "(-(2LL<<30))"
7296                 and max = "((2LL<<30)-1)"
7297                 and comment =
7298                   "The Int type in the generator is a signed 31 bit int." in
7299                 Some (min, max, comment) in
7300               parse_integer "xstrtoll" "long long" "int" range name i
7301           | Int64 name ->
7302               parse_integer "xstrtoll" "long long" "int64_t" None name i
7303       ) (snd style);
7304
7305       (* Call C API function. *)
7306       let fn =
7307         try find_map (function FishAction n -> Some n | _ -> None) flags
7308         with Not_found -> sprintf "guestfs_%s" name in
7309       pr "  r = %s " fn;
7310       generate_c_call_args ~handle:"g" style;
7311       pr ";\n";
7312
7313       List.iter (
7314         function
7315         | Device name | String name
7316         | OptString name | FileIn name | FileOut name | Bool name
7317         | Int name | Int64 name -> ()
7318         | Pathname name | Dev_or_Path name ->
7319             pr "  free (%s);\n" name
7320         | StringList name | DeviceList name ->
7321             pr "  free_strings (%s);\n" name
7322       ) (snd style);
7323
7324       (* Check return value for errors and display command results. *)
7325       (match fst style with
7326        | RErr -> pr "  return r;\n"
7327        | RInt _ ->
7328            pr "  if (r == -1) return -1;\n";
7329            pr "  printf (\"%%d\\n\", r);\n";
7330            pr "  return 0;\n"
7331        | RInt64 _ ->
7332            pr "  if (r == -1) return -1;\n";
7333            pr "  printf (\"%%\" PRIi64 \"\\n\", r);\n";
7334            pr "  return 0;\n"
7335        | RBool _ ->
7336            pr "  if (r == -1) return -1;\n";
7337            pr "  if (r) printf (\"true\\n\"); else printf (\"false\\n\");\n";
7338            pr "  return 0;\n"
7339        | RConstString _ ->
7340            pr "  if (r == NULL) return -1;\n";
7341            pr "  printf (\"%%s\\n\", r);\n";
7342            pr "  return 0;\n"
7343        | RConstOptString _ ->
7344            pr "  printf (\"%%s\\n\", r ? : \"(null)\");\n";
7345            pr "  return 0;\n"
7346        | RString _ ->
7347            pr "  if (r == NULL) return -1;\n";
7348            pr "  printf (\"%%s\\n\", r);\n";
7349            pr "  free (r);\n";
7350            pr "  return 0;\n"
7351        | RStringList _ ->
7352            pr "  if (r == NULL) return -1;\n";
7353            pr "  print_strings (r);\n";
7354            pr "  free_strings (r);\n";
7355            pr "  return 0;\n"
7356        | RStruct (_, typ) ->
7357            pr "  if (r == NULL) return -1;\n";
7358            pr "  print_%s (r);\n" typ;
7359            pr "  guestfs_free_%s (r);\n" typ;
7360            pr "  return 0;\n"
7361        | RStructList (_, typ) ->
7362            pr "  if (r == NULL) return -1;\n";
7363            pr "  print_%s_list (r);\n" typ;
7364            pr "  guestfs_free_%s_list (r);\n" typ;
7365            pr "  return 0;\n"
7366        | RHashtable _ ->
7367            pr "  if (r == NULL) return -1;\n";
7368            pr "  print_table (r);\n";
7369            pr "  free_strings (r);\n";
7370            pr "  return 0;\n"
7371        | RBufferOut _ ->
7372            pr "  if (r == NULL) return -1;\n";
7373            pr "  if (full_write (1, r, size) != size) {\n";
7374            pr "    perror (\"write\");\n";
7375            pr "    free (r);\n";
7376            pr "    return -1;\n";
7377            pr "  }\n";
7378            pr "  free (r);\n";
7379            pr "  return 0;\n"
7380       );
7381       pr "}\n";
7382       pr "\n"
7383   ) all_functions;
7384
7385   (* run_action function *)
7386   pr "int run_action (const char *cmd, int argc, char *argv[])\n";
7387   pr "{\n";
7388   List.iter (
7389     fun (name, _, _, flags, _, _, _) ->
7390       let name2 = replace_char name '_' '-' in
7391       let alias =
7392         try find_map (function FishAlias n -> Some n | _ -> None) flags
7393         with Not_found -> name in
7394       pr "  if (";
7395       pr "STRCASEEQ (cmd, \"%s\")" name;
7396       if name <> name2 then
7397         pr " || STRCASEEQ (cmd, \"%s\")" name2;
7398       if name <> alias then
7399         pr " || STRCASEEQ (cmd, \"%s\")" alias;
7400       pr ")\n";
7401       pr "    return run_%s (cmd, argc, argv);\n" name;
7402       pr "  else\n";
7403   ) all_functions;
7404   pr "    {\n";
7405   pr "      fprintf (stderr, _(\"%%s: unknown command\\n\"), cmd);\n";
7406   pr "      if (command_num == 1)\n";
7407   pr "        extended_help_message ();\n";
7408   pr "      return -1;\n";
7409   pr "    }\n";
7410   pr "  return 0;\n";
7411   pr "}\n";
7412   pr "\n"
7413
7414 (* Readline completion for guestfish. *)
7415 and generate_fish_completion () =
7416   generate_header CStyle GPLv2plus;
7417
7418   let all_functions =
7419     List.filter (
7420       fun (_, _, _, flags, _, _, _) -> not (List.mem NotInFish flags)
7421     ) all_functions in
7422
7423   pr "\
7424 #include <config.h>
7425
7426 #include <stdio.h>
7427 #include <stdlib.h>
7428 #include <string.h>
7429
7430 #ifdef HAVE_LIBREADLINE
7431 #include <readline/readline.h>
7432 #endif
7433
7434 #include \"fish.h\"
7435
7436 #ifdef HAVE_LIBREADLINE
7437
7438 static const char *const commands[] = {
7439   BUILTIN_COMMANDS_FOR_COMPLETION,
7440 ";
7441
7442   (* Get the commands, including the aliases.  They don't need to be
7443    * sorted - the generator() function just does a dumb linear search.
7444    *)
7445   let commands =
7446     List.map (
7447       fun (name, _, _, flags, _, _, _) ->
7448         let name2 = replace_char name '_' '-' in
7449         let alias =
7450           try find_map (function FishAlias n -> Some n | _ -> None) flags
7451           with Not_found -> name in
7452
7453         if name <> alias then [name2; alias] else [name2]
7454     ) all_functions in
7455   let commands = List.flatten commands in
7456
7457   List.iter (pr "  \"%s\",\n") commands;
7458
7459   pr "  NULL
7460 };
7461
7462 static char *
7463 generator (const char *text, int state)
7464 {
7465   static int index, len;
7466   const char *name;
7467
7468   if (!state) {
7469     index = 0;
7470     len = strlen (text);
7471   }
7472
7473   rl_attempted_completion_over = 1;
7474
7475   while ((name = commands[index]) != NULL) {
7476     index++;
7477     if (STRCASEEQLEN (name, text, len))
7478       return strdup (name);
7479   }
7480
7481   return NULL;
7482 }
7483
7484 #endif /* HAVE_LIBREADLINE */
7485
7486 #ifdef HAVE_RL_COMPLETION_MATCHES
7487 #define RL_COMPLETION_MATCHES rl_completion_matches
7488 #else
7489 #ifdef HAVE_COMPLETION_MATCHES
7490 #define RL_COMPLETION_MATCHES completion_matches
7491 #endif
7492 #endif /* else just fail if we don't have either symbol */
7493
7494 char **
7495 do_completion (const char *text, int start, int end)
7496 {
7497   char **matches = NULL;
7498
7499 #ifdef HAVE_LIBREADLINE
7500   rl_completion_append_character = ' ';
7501
7502   if (start == 0)
7503     matches = RL_COMPLETION_MATCHES (text, generator);
7504   else if (complete_dest_paths)
7505     matches = RL_COMPLETION_MATCHES (text, complete_dest_paths_generator);
7506 #endif
7507
7508   return matches;
7509 }
7510 ";
7511
7512 (* Generate the POD documentation for guestfish. *)
7513 and generate_fish_actions_pod () =
7514   let all_functions_sorted =
7515     List.filter (
7516       fun (_, _, _, flags, _, _, _) ->
7517         not (List.mem NotInFish flags || List.mem NotInDocs flags)
7518     ) all_functions_sorted in
7519
7520   let rex = Str.regexp "C<guestfs_\\([^>]+\\)>" in
7521
7522   List.iter (
7523     fun (name, style, _, flags, _, _, longdesc) ->
7524       let longdesc =
7525         Str.global_substitute rex (
7526           fun s ->
7527             let sub =
7528               try Str.matched_group 1 s
7529               with Not_found ->
7530                 failwithf "error substituting C<guestfs_...> in longdesc of function %s" name in
7531             "C<" ^ replace_char sub '_' '-' ^ ">"
7532         ) longdesc in
7533       let name = replace_char name '_' '-' in
7534       let alias =
7535         try find_map (function FishAlias n -> Some n | _ -> None) flags
7536         with Not_found -> name in
7537
7538       pr "=head2 %s" name;
7539       if name <> alias then
7540         pr " | %s" alias;
7541       pr "\n";
7542       pr "\n";
7543       pr " %s" name;
7544       List.iter (
7545         function
7546         | Pathname n | Device n | Dev_or_Path n | String n -> pr " %s" n
7547         | OptString n -> pr " %s" n
7548         | StringList n | DeviceList n -> pr " '%s ...'" n
7549         | Bool _ -> pr " true|false"
7550         | Int n -> pr " %s" n
7551         | Int64 n -> pr " %s" n
7552         | FileIn n | FileOut n -> pr " (%s|-)" n
7553       ) (snd style);
7554       pr "\n";
7555       pr "\n";
7556       pr "%s\n\n" longdesc;
7557
7558       if List.exists (function FileIn _ | FileOut _ -> true
7559                       | _ -> false) (snd style) then
7560         pr "Use C<-> instead of a filename to read/write from stdin/stdout.\n\n";
7561
7562       if List.mem ProtocolLimitWarning flags then
7563         pr "%s\n\n" protocol_limit_warning;
7564
7565       if List.mem DangerWillRobinson flags then
7566         pr "%s\n\n" danger_will_robinson;
7567
7568       match deprecation_notice flags with
7569       | None -> ()
7570       | Some txt -> pr "%s\n\n" txt
7571   ) all_functions_sorted
7572
7573 (* Generate a C function prototype. *)
7574 and generate_prototype ?(extern = true) ?(static = false) ?(semicolon = true)
7575     ?(single_line = false) ?(newline = false) ?(in_daemon = false)
7576     ?(prefix = "")
7577     ?handle name style =
7578   if extern then pr "extern ";
7579   if static then pr "static ";
7580   (match fst style with
7581    | RErr -> pr "int "
7582    | RInt _ -> pr "int "
7583    | RInt64 _ -> pr "int64_t "
7584    | RBool _ -> pr "int "
7585    | RConstString _ | RConstOptString _ -> pr "const char *"
7586    | RString _ | RBufferOut _ -> pr "char *"
7587    | RStringList _ | RHashtable _ -> pr "char **"
7588    | RStruct (_, typ) ->
7589        if not in_daemon then pr "struct guestfs_%s *" typ
7590        else pr "guestfs_int_%s *" typ
7591    | RStructList (_, typ) ->
7592        if not in_daemon then pr "struct guestfs_%s_list *" typ
7593        else pr "guestfs_int_%s_list *" typ
7594   );
7595   let is_RBufferOut = match fst style with RBufferOut _ -> true | _ -> false in
7596   pr "%s%s (" prefix name;
7597   if handle = None && List.length (snd style) = 0 && not is_RBufferOut then
7598     pr "void"
7599   else (
7600     let comma = ref false in
7601     (match handle with
7602      | None -> ()
7603      | Some handle -> pr "guestfs_h *%s" handle; comma := true
7604     );
7605     let next () =
7606       if !comma then (
7607         if single_line then pr ", " else pr ",\n\t\t"
7608       );
7609       comma := true
7610     in
7611     List.iter (
7612       function
7613       | Pathname n
7614       | Device n | Dev_or_Path n
7615       | String n
7616       | OptString n ->
7617           next ();
7618           pr "const char *%s" n
7619       | StringList n | DeviceList n ->
7620           next ();
7621           pr "char *const *%s" n
7622       | Bool n -> next (); pr "int %s" n
7623       | Int n -> next (); pr "int %s" n
7624       | Int64 n -> next (); pr "int64_t %s" n
7625       | FileIn n
7626       | FileOut n ->
7627           if not in_daemon then (next (); pr "const char *%s" n)
7628     ) (snd style);
7629     if is_RBufferOut then (next (); pr "size_t *size_r");
7630   );
7631   pr ")";
7632   if semicolon then pr ";";
7633   if newline then pr "\n"
7634
7635 (* Generate C call arguments, eg "(handle, foo, bar)" *)
7636 and generate_c_call_args ?handle ?(decl = false) style =
7637   pr "(";
7638   let comma = ref false in
7639   let next () =
7640     if !comma then pr ", ";
7641     comma := true
7642   in
7643   (match handle with
7644    | None -> ()
7645    | Some handle -> pr "%s" handle; comma := true
7646   );
7647   List.iter (
7648     fun arg ->
7649       next ();
7650       pr "%s" (name_of_argt arg)
7651   ) (snd style);
7652   (* For RBufferOut calls, add implicit &size parameter. *)
7653   if not decl then (
7654     match fst style with
7655     | RBufferOut _ ->
7656         next ();
7657         pr "&size"
7658     | _ -> ()
7659   );
7660   pr ")"
7661
7662 (* Generate the OCaml bindings interface. *)
7663 and generate_ocaml_mli () =
7664   generate_header OCamlStyle LGPLv2plus;
7665
7666   pr "\
7667 (** For API documentation you should refer to the C API
7668     in the guestfs(3) manual page.  The OCaml API uses almost
7669     exactly the same calls. *)
7670
7671 type t
7672 (** A [guestfs_h] handle. *)
7673
7674 exception Error of string
7675 (** This exception is raised when there is an error. *)
7676
7677 exception Handle_closed of string
7678 (** This exception is raised if you use a {!Guestfs.t} handle
7679     after calling {!close} on it.  The string is the name of
7680     the function. *)
7681
7682 val create : unit -> t
7683 (** Create a {!Guestfs.t} handle. *)
7684
7685 val close : t -> unit
7686 (** Close the {!Guestfs.t} handle and free up all resources used
7687     by it immediately.
7688
7689     Handles are closed by the garbage collector when they become
7690     unreferenced, but callers can call this in order to provide
7691     predictable cleanup. *)
7692
7693 ";
7694   generate_ocaml_structure_decls ();
7695
7696   (* The actions. *)
7697   List.iter (
7698     fun (name, style, _, _, _, shortdesc, _) ->
7699       generate_ocaml_prototype name style;
7700       pr "(** %s *)\n" shortdesc;
7701       pr "\n"
7702   ) all_functions_sorted
7703
7704 (* Generate the OCaml bindings implementation. *)
7705 and generate_ocaml_ml () =
7706   generate_header OCamlStyle LGPLv2plus;
7707
7708   pr "\
7709 type t
7710
7711 exception Error of string
7712 exception Handle_closed of string
7713
7714 external create : unit -> t = \"ocaml_guestfs_create\"
7715 external close : t -> unit = \"ocaml_guestfs_close\"
7716
7717 (* Give the exceptions names, so they can be raised from the C code. *)
7718 let () =
7719   Callback.register_exception \"ocaml_guestfs_error\" (Error \"\");
7720   Callback.register_exception \"ocaml_guestfs_closed\" (Handle_closed \"\")
7721
7722 ";
7723
7724   generate_ocaml_structure_decls ();
7725
7726   (* The actions. *)
7727   List.iter (
7728     fun (name, style, _, _, _, shortdesc, _) ->
7729       generate_ocaml_prototype ~is_external:true name style;
7730   ) all_functions_sorted
7731
7732 (* Generate the OCaml bindings C implementation. *)
7733 and generate_ocaml_c () =
7734   generate_header CStyle LGPLv2plus;
7735
7736   pr "\
7737 #include <stdio.h>
7738 #include <stdlib.h>
7739 #include <string.h>
7740
7741 #include <caml/config.h>
7742 #include <caml/alloc.h>
7743 #include <caml/callback.h>
7744 #include <caml/fail.h>
7745 #include <caml/memory.h>
7746 #include <caml/mlvalues.h>
7747 #include <caml/signals.h>
7748
7749 #include <guestfs.h>
7750
7751 #include \"guestfs_c.h\"
7752
7753 /* Copy a hashtable of string pairs into an assoc-list.  We return
7754  * the list in reverse order, but hashtables aren't supposed to be
7755  * ordered anyway.
7756  */
7757 static CAMLprim value
7758 copy_table (char * const * argv)
7759 {
7760   CAMLparam0 ();
7761   CAMLlocal5 (rv, pairv, kv, vv, cons);
7762   int i;
7763
7764   rv = Val_int (0);
7765   for (i = 0; argv[i] != NULL; i += 2) {
7766     kv = caml_copy_string (argv[i]);
7767     vv = caml_copy_string (argv[i+1]);
7768     pairv = caml_alloc (2, 0);
7769     Store_field (pairv, 0, kv);
7770     Store_field (pairv, 1, vv);
7771     cons = caml_alloc (2, 0);
7772     Store_field (cons, 1, rv);
7773     rv = cons;
7774     Store_field (cons, 0, pairv);
7775   }
7776
7777   CAMLreturn (rv);
7778 }
7779
7780 ";
7781
7782   (* Struct copy functions. *)
7783
7784   let emit_ocaml_copy_list_function typ =
7785     pr "static CAMLprim value\n";
7786     pr "copy_%s_list (const struct guestfs_%s_list *%ss)\n" typ typ typ;
7787     pr "{\n";
7788     pr "  CAMLparam0 ();\n";
7789     pr "  CAMLlocal2 (rv, v);\n";
7790     pr "  unsigned int i;\n";
7791     pr "\n";
7792     pr "  if (%ss->len == 0)\n" typ;
7793     pr "    CAMLreturn (Atom (0));\n";
7794     pr "  else {\n";
7795     pr "    rv = caml_alloc (%ss->len, 0);\n" typ;
7796     pr "    for (i = 0; i < %ss->len; ++i) {\n" typ;
7797     pr "      v = copy_%s (&%ss->val[i]);\n" typ typ;
7798     pr "      caml_modify (&Field (rv, i), v);\n";
7799     pr "    }\n";
7800     pr "    CAMLreturn (rv);\n";
7801     pr "  }\n";
7802     pr "}\n";
7803     pr "\n";
7804   in
7805
7806   List.iter (
7807     fun (typ, cols) ->
7808       let has_optpercent_col =
7809         List.exists (function (_, FOptPercent) -> true | _ -> false) cols in
7810
7811       pr "static CAMLprim value\n";
7812       pr "copy_%s (const struct guestfs_%s *%s)\n" typ typ typ;
7813       pr "{\n";
7814       pr "  CAMLparam0 ();\n";
7815       if has_optpercent_col then
7816         pr "  CAMLlocal3 (rv, v, v2);\n"
7817       else
7818         pr "  CAMLlocal2 (rv, v);\n";
7819       pr "\n";
7820       pr "  rv = caml_alloc (%d, 0);\n" (List.length cols);
7821       iteri (
7822         fun i col ->
7823           (match col with
7824            | name, FString ->
7825                pr "  v = caml_copy_string (%s->%s);\n" typ name
7826            | name, FBuffer ->
7827                pr "  v = caml_alloc_string (%s->%s_len);\n" typ name;
7828                pr "  memcpy (String_val (v), %s->%s, %s->%s_len);\n"
7829                  typ name typ name
7830            | name, FUUID ->
7831                pr "  v = caml_alloc_string (32);\n";
7832                pr "  memcpy (String_val (v), %s->%s, 32);\n" typ name
7833            | name, (FBytes|FInt64|FUInt64) ->
7834                pr "  v = caml_copy_int64 (%s->%s);\n" typ name
7835            | name, (FInt32|FUInt32) ->
7836                pr "  v = caml_copy_int32 (%s->%s);\n" typ name
7837            | name, FOptPercent ->
7838                pr "  if (%s->%s >= 0) { /* Some %s */\n" typ name name;
7839                pr "    v2 = caml_copy_double (%s->%s);\n" typ name;
7840                pr "    v = caml_alloc (1, 0);\n";
7841                pr "    Store_field (v, 0, v2);\n";
7842                pr "  } else /* None */\n";
7843                pr "    v = Val_int (0);\n";
7844            | name, FChar ->
7845                pr "  v = Val_int (%s->%s);\n" typ name
7846           );
7847           pr "  Store_field (rv, %d, v);\n" i
7848       ) cols;
7849       pr "  CAMLreturn (rv);\n";
7850       pr "}\n";
7851       pr "\n";
7852   ) structs;
7853
7854   (* Emit a copy_TYPE_list function definition only if that function is used. *)
7855   List.iter (
7856     function
7857     | typ, (RStructListOnly | RStructAndList) ->
7858         (* generate the function for typ *)
7859         emit_ocaml_copy_list_function typ
7860     | typ, _ -> () (* empty *)
7861   ) (rstructs_used_by all_functions);
7862
7863   (* The wrappers. *)
7864   List.iter (
7865     fun (name, style, _, _, _, _, _) ->
7866       pr "/* Automatically generated wrapper for function\n";
7867       pr " * ";
7868       generate_ocaml_prototype name style;
7869       pr " */\n";
7870       pr "\n";
7871
7872       let params =
7873         "gv" :: List.map (fun arg -> name_of_argt arg ^ "v") (snd style) in
7874
7875       let needs_extra_vs =
7876         match fst style with RConstOptString _ -> true | _ -> false in
7877
7878       pr "/* Emit prototype to appease gcc's -Wmissing-prototypes. */\n";
7879       pr "CAMLprim value ocaml_guestfs_%s (value %s" name (List.hd params);
7880       List.iter (pr ", value %s") (List.tl params); pr ");\n";
7881       pr "\n";
7882
7883       pr "CAMLprim value\n";
7884       pr "ocaml_guestfs_%s (value %s" name (List.hd params);
7885       List.iter (pr ", value %s") (List.tl params);
7886       pr ")\n";
7887       pr "{\n";
7888
7889       (match params with
7890        | [p1; p2; p3; p4; p5] ->
7891            pr "  CAMLparam5 (%s);\n" (String.concat ", " params)
7892        | p1 :: p2 :: p3 :: p4 :: p5 :: rest ->
7893            pr "  CAMLparam5 (%s);\n" (String.concat ", " [p1; p2; p3; p4; p5]);
7894            pr "  CAMLxparam%d (%s);\n"
7895              (List.length rest) (String.concat ", " rest)
7896        | ps ->
7897            pr "  CAMLparam%d (%s);\n" (List.length ps) (String.concat ", " ps)
7898       );
7899       if not needs_extra_vs then
7900         pr "  CAMLlocal1 (rv);\n"
7901       else
7902         pr "  CAMLlocal3 (rv, v, v2);\n";
7903       pr "\n";
7904
7905       pr "  guestfs_h *g = Guestfs_val (gv);\n";
7906       pr "  if (g == NULL)\n";
7907       pr "    ocaml_guestfs_raise_closed (\"%s\");\n" name;
7908       pr "\n";
7909
7910       List.iter (
7911         function
7912         | Pathname n
7913         | Device n | Dev_or_Path n
7914         | String n
7915         | FileIn n
7916         | FileOut n ->
7917             pr "  const char *%s = String_val (%sv);\n" n n
7918         | OptString n ->
7919             pr "  const char *%s =\n" n;
7920             pr "    %sv != Val_int (0) ? String_val (Field (%sv, 0)) : NULL;\n"
7921               n n
7922         | StringList n | DeviceList n ->
7923             pr "  char **%s = ocaml_guestfs_strings_val (g, %sv);\n" n n
7924         | Bool n ->
7925             pr "  int %s = Bool_val (%sv);\n" n n
7926         | Int n ->
7927             pr "  int %s = Int_val (%sv);\n" n n
7928         | Int64 n ->
7929             pr "  int64_t %s = Int64_val (%sv);\n" n n
7930       ) (snd style);
7931       let error_code =
7932         match fst style with
7933         | RErr -> pr "  int r;\n"; "-1"
7934         | RInt _ -> pr "  int r;\n"; "-1"
7935         | RInt64 _ -> pr "  int64_t r;\n"; "-1"
7936         | RBool _ -> pr "  int r;\n"; "-1"
7937         | RConstString _ | RConstOptString _ ->
7938             pr "  const char *r;\n"; "NULL"
7939         | RString _ -> pr "  char *r;\n"; "NULL"
7940         | RStringList _ ->
7941             pr "  int i;\n";
7942             pr "  char **r;\n";
7943             "NULL"
7944         | RStruct (_, typ) ->
7945             pr "  struct guestfs_%s *r;\n" typ; "NULL"
7946         | RStructList (_, typ) ->
7947             pr "  struct guestfs_%s_list *r;\n" typ; "NULL"
7948         | RHashtable _ ->
7949             pr "  int i;\n";
7950             pr "  char **r;\n";
7951             "NULL"
7952         | RBufferOut _ ->
7953             pr "  char *r;\n";
7954             pr "  size_t size;\n";
7955             "NULL" in
7956       pr "\n";
7957
7958       pr "  caml_enter_blocking_section ();\n";
7959       pr "  r = guestfs_%s " name;
7960       generate_c_call_args ~handle:"g" style;
7961       pr ";\n";
7962       pr "  caml_leave_blocking_section ();\n";
7963
7964       List.iter (
7965         function
7966         | StringList n | DeviceList n ->
7967             pr "  ocaml_guestfs_free_strings (%s);\n" n;
7968         | Pathname _ | Device _ | Dev_or_Path _ | String _ | OptString _
7969         | Bool _ | Int _ | Int64 _
7970         | FileIn _ | FileOut _ -> ()
7971       ) (snd style);
7972
7973       pr "  if (r == %s)\n" error_code;
7974       pr "    ocaml_guestfs_raise_error (g, \"%s\");\n" name;
7975       pr "\n";
7976
7977       (match fst style with
7978        | RErr -> pr "  rv = Val_unit;\n"
7979        | RInt _ -> pr "  rv = Val_int (r);\n"
7980        | RInt64 _ ->
7981            pr "  rv = caml_copy_int64 (r);\n"
7982        | RBool _ -> pr "  rv = Val_bool (r);\n"
7983        | RConstString _ ->
7984            pr "  rv = caml_copy_string (r);\n"
7985        | RConstOptString _ ->
7986            pr "  if (r) { /* Some string */\n";
7987            pr "    v = caml_alloc (1, 0);\n";
7988            pr "    v2 = caml_copy_string (r);\n";
7989            pr "    Store_field (v, 0, v2);\n";
7990            pr "  } else /* None */\n";
7991            pr "    v = Val_int (0);\n";
7992        | RString _ ->
7993            pr "  rv = caml_copy_string (r);\n";
7994            pr "  free (r);\n"
7995        | RStringList _ ->
7996            pr "  rv = caml_copy_string_array ((const char **) r);\n";
7997            pr "  for (i = 0; r[i] != NULL; ++i) free (r[i]);\n";
7998            pr "  free (r);\n"
7999        | RStruct (_, typ) ->
8000            pr "  rv = copy_%s (r);\n" typ;
8001            pr "  guestfs_free_%s (r);\n" typ;
8002        | RStructList (_, typ) ->
8003            pr "  rv = copy_%s_list (r);\n" typ;
8004            pr "  guestfs_free_%s_list (r);\n" typ;
8005        | RHashtable _ ->
8006            pr "  rv = copy_table (r);\n";
8007            pr "  for (i = 0; r[i] != NULL; ++i) free (r[i]);\n";
8008            pr "  free (r);\n";
8009        | RBufferOut _ ->
8010            pr "  rv = caml_alloc_string (size);\n";
8011            pr "  memcpy (String_val (rv), r, size);\n";
8012       );
8013
8014       pr "  CAMLreturn (rv);\n";
8015       pr "}\n";
8016       pr "\n";
8017
8018       if List.length params > 5 then (
8019         pr "/* Emit prototype to appease gcc's -Wmissing-prototypes. */\n";
8020         pr "CAMLprim value ";
8021         pr "ocaml_guestfs_%s_byte (value *argv, int argn);\n" name;
8022         pr "CAMLprim value\n";
8023         pr "ocaml_guestfs_%s_byte (value *argv, int argn)\n" name;
8024         pr "{\n";
8025         pr "  return ocaml_guestfs_%s (argv[0]" name;
8026         iteri (fun i _ -> pr ", argv[%d]" i) (List.tl params);
8027         pr ");\n";
8028         pr "}\n";
8029         pr "\n"
8030       )
8031   ) all_functions_sorted
8032
8033 and generate_ocaml_structure_decls () =
8034   List.iter (
8035     fun (typ, cols) ->
8036       pr "type %s = {\n" typ;
8037       List.iter (
8038         function
8039         | name, FString -> pr "  %s : string;\n" name
8040         | name, FBuffer -> pr "  %s : string;\n" name
8041         | name, FUUID -> pr "  %s : string;\n" name
8042         | name, (FBytes|FInt64|FUInt64) -> pr "  %s : int64;\n" name
8043         | name, (FInt32|FUInt32) -> pr "  %s : int32;\n" name
8044         | name, FChar -> pr "  %s : char;\n" name
8045         | name, FOptPercent -> pr "  %s : float option;\n" name
8046       ) cols;
8047       pr "}\n";
8048       pr "\n"
8049   ) structs
8050
8051 and generate_ocaml_prototype ?(is_external = false) name style =
8052   if is_external then pr "external " else pr "val ";
8053   pr "%s : t -> " name;
8054   List.iter (
8055     function
8056     | Pathname _ | Device _ | Dev_or_Path _ | String _ | FileIn _ | FileOut _ -> pr "string -> "
8057     | OptString _ -> pr "string option -> "
8058     | StringList _ | DeviceList _ -> pr "string array -> "
8059     | Bool _ -> pr "bool -> "
8060     | Int _ -> pr "int -> "
8061     | Int64 _ -> pr "int64 -> "
8062   ) (snd style);
8063   (match fst style with
8064    | RErr -> pr "unit" (* all errors are turned into exceptions *)
8065    | RInt _ -> pr "int"
8066    | RInt64 _ -> pr "int64"
8067    | RBool _ -> pr "bool"
8068    | RConstString _ -> pr "string"
8069    | RConstOptString _ -> pr "string option"
8070    | RString _ | RBufferOut _ -> pr "string"
8071    | RStringList _ -> pr "string array"
8072    | RStruct (_, typ) -> pr "%s" typ
8073    | RStructList (_, typ) -> pr "%s array" typ
8074    | RHashtable _ -> pr "(string * string) list"
8075   );
8076   if is_external then (
8077     pr " = ";
8078     if List.length (snd style) + 1 > 5 then
8079       pr "\"ocaml_guestfs_%s_byte\" " name;
8080     pr "\"ocaml_guestfs_%s\"" name
8081   );
8082   pr "\n"
8083
8084 (* Generate Perl xs code, a sort of crazy variation of C with macros. *)
8085 and generate_perl_xs () =
8086   generate_header CStyle LGPLv2plus;
8087
8088   pr "\
8089 #include \"EXTERN.h\"
8090 #include \"perl.h\"
8091 #include \"XSUB.h\"
8092
8093 #include <guestfs.h>
8094
8095 #ifndef PRId64
8096 #define PRId64 \"lld\"
8097 #endif
8098
8099 static SV *
8100 my_newSVll(long long val) {
8101 #ifdef USE_64_BIT_ALL
8102   return newSViv(val);
8103 #else
8104   char buf[100];
8105   int len;
8106   len = snprintf(buf, 100, \"%%\" PRId64, val);
8107   return newSVpv(buf, len);
8108 #endif
8109 }
8110
8111 #ifndef PRIu64
8112 #define PRIu64 \"llu\"
8113 #endif
8114
8115 static SV *
8116 my_newSVull(unsigned long long val) {
8117 #ifdef USE_64_BIT_ALL
8118   return newSVuv(val);
8119 #else
8120   char buf[100];
8121   int len;
8122   len = snprintf(buf, 100, \"%%\" PRIu64, val);
8123   return newSVpv(buf, len);
8124 #endif
8125 }
8126
8127 /* http://www.perlmonks.org/?node_id=680842 */
8128 static char **
8129 XS_unpack_charPtrPtr (SV *arg) {
8130   char **ret;
8131   AV *av;
8132   I32 i;
8133
8134   if (!arg || !SvOK (arg) || !SvROK (arg) || SvTYPE (SvRV (arg)) != SVt_PVAV)
8135     croak (\"array reference expected\");
8136
8137   av = (AV *)SvRV (arg);
8138   ret = malloc ((av_len (av) + 1 + 1) * sizeof (char *));
8139   if (!ret)
8140     croak (\"malloc failed\");
8141
8142   for (i = 0; i <= av_len (av); i++) {
8143     SV **elem = av_fetch (av, i, 0);
8144
8145     if (!elem || !*elem)
8146       croak (\"missing element in list\");
8147
8148     ret[i] = SvPV_nolen (*elem);
8149   }
8150
8151   ret[i] = NULL;
8152
8153   return ret;
8154 }
8155
8156 MODULE = Sys::Guestfs  PACKAGE = Sys::Guestfs
8157
8158 PROTOTYPES: ENABLE
8159
8160 guestfs_h *
8161 _create ()
8162    CODE:
8163       RETVAL = guestfs_create ();
8164       if (!RETVAL)
8165         croak (\"could not create guestfs handle\");
8166       guestfs_set_error_handler (RETVAL, NULL, NULL);
8167  OUTPUT:
8168       RETVAL
8169
8170 void
8171 DESTROY (g)
8172       guestfs_h *g;
8173  PPCODE:
8174       guestfs_close (g);
8175
8176 ";
8177
8178   List.iter (
8179     fun (name, style, _, _, _, _, _) ->
8180       (match fst style with
8181        | RErr -> pr "void\n"
8182        | RInt _ -> pr "SV *\n"
8183        | RInt64 _ -> pr "SV *\n"
8184        | RBool _ -> pr "SV *\n"
8185        | RConstString _ -> pr "SV *\n"
8186        | RConstOptString _ -> pr "SV *\n"
8187        | RString _ -> pr "SV *\n"
8188        | RBufferOut _ -> pr "SV *\n"
8189        | RStringList _
8190        | RStruct _ | RStructList _
8191        | RHashtable _ ->
8192            pr "void\n" (* all lists returned implictly on the stack *)
8193       );
8194       (* Call and arguments. *)
8195       pr "%s " name;
8196       generate_c_call_args ~handle:"g" ~decl:true style;
8197       pr "\n";
8198       pr "      guestfs_h *g;\n";
8199       iteri (
8200         fun i ->
8201           function
8202           | Pathname n | Device n | Dev_or_Path n | String n | FileIn n | FileOut n ->
8203               pr "      char *%s;\n" n
8204           | OptString n ->
8205               (* http://www.perlmonks.org/?node_id=554277
8206                * Note that the implicit handle argument means we have
8207                * to add 1 to the ST(x) operator.
8208                *)
8209               pr "      char *%s = SvOK(ST(%d)) ? SvPV_nolen(ST(%d)) : NULL;\n" n (i+1) (i+1)
8210           | StringList n | DeviceList n -> pr "      char **%s;\n" n
8211           | Bool n -> pr "      int %s;\n" n
8212           | Int n -> pr "      int %s;\n" n
8213           | Int64 n -> pr "      int64_t %s;\n" n
8214       ) (snd style);
8215
8216       let do_cleanups () =
8217         List.iter (
8218           function
8219           | Pathname _ | Device _ | Dev_or_Path _ | String _ | OptString _
8220           | Bool _ | Int _ | Int64 _
8221           | FileIn _ | FileOut _ -> ()
8222           | StringList n | DeviceList n -> pr "      free (%s);\n" n
8223         ) (snd style)
8224       in
8225
8226       (* Code. *)
8227       (match fst style with
8228        | RErr ->
8229            pr "PREINIT:\n";
8230            pr "      int r;\n";
8231            pr " PPCODE:\n";
8232            pr "      r = guestfs_%s " name;
8233            generate_c_call_args ~handle:"g" style;
8234            pr ";\n";
8235            do_cleanups ();
8236            pr "      if (r == -1)\n";
8237            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8238        | RInt n
8239        | RBool n ->
8240            pr "PREINIT:\n";
8241            pr "      int %s;\n" n;
8242            pr "   CODE:\n";
8243            pr "      %s = guestfs_%s " n name;
8244            generate_c_call_args ~handle:"g" style;
8245            pr ";\n";
8246            do_cleanups ();
8247            pr "      if (%s == -1)\n" n;
8248            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8249            pr "      RETVAL = newSViv (%s);\n" n;
8250            pr " OUTPUT:\n";
8251            pr "      RETVAL\n"
8252        | RInt64 n ->
8253            pr "PREINIT:\n";
8254            pr "      int64_t %s;\n" n;
8255            pr "   CODE:\n";
8256            pr "      %s = guestfs_%s " n name;
8257            generate_c_call_args ~handle:"g" style;
8258            pr ";\n";
8259            do_cleanups ();
8260            pr "      if (%s == -1)\n" n;
8261            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8262            pr "      RETVAL = my_newSVll (%s);\n" n;
8263            pr " OUTPUT:\n";
8264            pr "      RETVAL\n"
8265        | RConstString n ->
8266            pr "PREINIT:\n";
8267            pr "      const char *%s;\n" n;
8268            pr "   CODE:\n";
8269            pr "      %s = guestfs_%s " n name;
8270            generate_c_call_args ~handle:"g" style;
8271            pr ";\n";
8272            do_cleanups ();
8273            pr "      if (%s == NULL)\n" n;
8274            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8275            pr "      RETVAL = newSVpv (%s, 0);\n" n;
8276            pr " OUTPUT:\n";
8277            pr "      RETVAL\n"
8278        | RConstOptString n ->
8279            pr "PREINIT:\n";
8280            pr "      const char *%s;\n" n;
8281            pr "   CODE:\n";
8282            pr "      %s = guestfs_%s " n name;
8283            generate_c_call_args ~handle:"g" style;
8284            pr ";\n";
8285            do_cleanups ();
8286            pr "      if (%s == NULL)\n" n;
8287            pr "        RETVAL = &PL_sv_undef;\n";
8288            pr "      else\n";
8289            pr "        RETVAL = newSVpv (%s, 0);\n" n;
8290            pr " OUTPUT:\n";
8291            pr "      RETVAL\n"
8292        | RString n ->
8293            pr "PREINIT:\n";
8294            pr "      char *%s;\n" n;
8295            pr "   CODE:\n";
8296            pr "      %s = guestfs_%s " n name;
8297            generate_c_call_args ~handle:"g" style;
8298            pr ";\n";
8299            do_cleanups ();
8300            pr "      if (%s == NULL)\n" n;
8301            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8302            pr "      RETVAL = newSVpv (%s, 0);\n" n;
8303            pr "      free (%s);\n" n;
8304            pr " OUTPUT:\n";
8305            pr "      RETVAL\n"
8306        | RStringList n | RHashtable n ->
8307            pr "PREINIT:\n";
8308            pr "      char **%s;\n" n;
8309            pr "      int i, n;\n";
8310            pr " PPCODE:\n";
8311            pr "      %s = guestfs_%s " n name;
8312            generate_c_call_args ~handle:"g" style;
8313            pr ";\n";
8314            do_cleanups ();
8315            pr "      if (%s == NULL)\n" n;
8316            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8317            pr "      for (n = 0; %s[n] != NULL; ++n) /**/;\n" n;
8318            pr "      EXTEND (SP, n);\n";
8319            pr "      for (i = 0; i < n; ++i) {\n";
8320            pr "        PUSHs (sv_2mortal (newSVpv (%s[i], 0)));\n" n;
8321            pr "        free (%s[i]);\n" n;
8322            pr "      }\n";
8323            pr "      free (%s);\n" n;
8324        | RStruct (n, typ) ->
8325            let cols = cols_of_struct typ in
8326            generate_perl_struct_code typ cols name style n do_cleanups
8327        | RStructList (n, typ) ->
8328            let cols = cols_of_struct typ in
8329            generate_perl_struct_list_code typ cols name style n do_cleanups
8330        | RBufferOut n ->
8331            pr "PREINIT:\n";
8332            pr "      char *%s;\n" n;
8333            pr "      size_t size;\n";
8334            pr "   CODE:\n";
8335            pr "      %s = guestfs_%s " n name;
8336            generate_c_call_args ~handle:"g" style;
8337            pr ";\n";
8338            do_cleanups ();
8339            pr "      if (%s == NULL)\n" n;
8340            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8341            pr "      RETVAL = newSVpvn (%s, size);\n" n;
8342            pr "      free (%s);\n" n;
8343            pr " OUTPUT:\n";
8344            pr "      RETVAL\n"
8345       );
8346
8347       pr "\n"
8348   ) all_functions
8349
8350 and generate_perl_struct_list_code typ cols name style n do_cleanups =
8351   pr "PREINIT:\n";
8352   pr "      struct guestfs_%s_list *%s;\n" typ n;
8353   pr "      int i;\n";
8354   pr "      HV *hv;\n";
8355   pr " PPCODE:\n";
8356   pr "      %s = guestfs_%s " n name;
8357   generate_c_call_args ~handle:"g" style;
8358   pr ";\n";
8359   do_cleanups ();
8360   pr "      if (%s == NULL)\n" n;
8361   pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8362   pr "      EXTEND (SP, %s->len);\n" n;
8363   pr "      for (i = 0; i < %s->len; ++i) {\n" n;
8364   pr "        hv = newHV ();\n";
8365   List.iter (
8366     function
8367     | name, FString ->
8368         pr "        (void) hv_store (hv, \"%s\", %d, newSVpv (%s->val[i].%s, 0), 0);\n"
8369           name (String.length name) n name
8370     | name, FUUID ->
8371         pr "        (void) hv_store (hv, \"%s\", %d, newSVpv (%s->val[i].%s, 32), 0);\n"
8372           name (String.length name) n name
8373     | name, FBuffer ->
8374         pr "        (void) hv_store (hv, \"%s\", %d, newSVpvn (%s->val[i].%s, %s->val[i].%s_len), 0);\n"
8375           name (String.length name) n name n name
8376     | name, (FBytes|FUInt64) ->
8377         pr "        (void) hv_store (hv, \"%s\", %d, my_newSVull (%s->val[i].%s), 0);\n"
8378           name (String.length name) n name
8379     | name, FInt64 ->
8380         pr "        (void) hv_store (hv, \"%s\", %d, my_newSVll (%s->val[i].%s), 0);\n"
8381           name (String.length name) n name
8382     | name, (FInt32|FUInt32) ->
8383         pr "        (void) hv_store (hv, \"%s\", %d, newSVnv (%s->val[i].%s), 0);\n"
8384           name (String.length name) n name
8385     | name, FChar ->
8386         pr "        (void) hv_store (hv, \"%s\", %d, newSVpv (&%s->val[i].%s, 1), 0);\n"
8387           name (String.length name) n name
8388     | name, FOptPercent ->
8389         pr "        (void) hv_store (hv, \"%s\", %d, newSVnv (%s->val[i].%s), 0);\n"
8390           name (String.length name) n name
8391   ) cols;
8392   pr "        PUSHs (sv_2mortal (newRV ((SV *) hv)));\n";
8393   pr "      }\n";
8394   pr "      guestfs_free_%s_list (%s);\n" typ n
8395
8396 and generate_perl_struct_code typ cols name style n do_cleanups =
8397   pr "PREINIT:\n";
8398   pr "      struct guestfs_%s *%s;\n" typ n;
8399   pr " PPCODE:\n";
8400   pr "      %s = guestfs_%s " n name;
8401   generate_c_call_args ~handle:"g" style;
8402   pr ";\n";
8403   do_cleanups ();
8404   pr "      if (%s == NULL)\n" n;
8405   pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8406   pr "      EXTEND (SP, 2 * %d);\n" (List.length cols);
8407   List.iter (
8408     fun ((name, _) as col) ->
8409       pr "      PUSHs (sv_2mortal (newSVpv (\"%s\", 0)));\n" name;
8410
8411       match col with
8412       | name, FString ->
8413           pr "      PUSHs (sv_2mortal (newSVpv (%s->%s, 0)));\n"
8414             n name
8415       | name, FBuffer ->
8416           pr "      PUSHs (sv_2mortal (newSVpvn (%s->%s, %s->%s_len)));\n"
8417             n name n name
8418       | name, FUUID ->
8419           pr "      PUSHs (sv_2mortal (newSVpv (%s->%s, 32)));\n"
8420             n name
8421       | name, (FBytes|FUInt64) ->
8422           pr "      PUSHs (sv_2mortal (my_newSVull (%s->%s)));\n"
8423             n name
8424       | name, FInt64 ->
8425           pr "      PUSHs (sv_2mortal (my_newSVll (%s->%s)));\n"
8426             n name
8427       | name, (FInt32|FUInt32) ->
8428           pr "      PUSHs (sv_2mortal (newSVnv (%s->%s)));\n"
8429             n name
8430       | name, FChar ->
8431           pr "      PUSHs (sv_2mortal (newSVpv (&%s->%s, 1)));\n"
8432             n name
8433       | name, FOptPercent ->
8434           pr "      PUSHs (sv_2mortal (newSVnv (%s->%s)));\n"
8435             n name
8436   ) cols;
8437   pr "      free (%s);\n" n
8438
8439 (* Generate Sys/Guestfs.pm. *)
8440 and generate_perl_pm () =
8441   generate_header HashStyle LGPLv2plus;
8442
8443   pr "\
8444 =pod
8445
8446 =head1 NAME
8447
8448 Sys::Guestfs - Perl bindings for libguestfs
8449
8450 =head1 SYNOPSIS
8451
8452  use Sys::Guestfs;
8453
8454  my $h = Sys::Guestfs->new ();
8455  $h->add_drive ('guest.img');
8456  $h->launch ();
8457  $h->mount ('/dev/sda1', '/');
8458  $h->touch ('/hello');
8459  $h->sync ();
8460
8461 =head1 DESCRIPTION
8462
8463 The C<Sys::Guestfs> module provides a Perl XS binding to the
8464 libguestfs API for examining and modifying virtual machine
8465 disk images.
8466
8467 Amongst the things this is good for: making batch configuration
8468 changes to guests, getting disk used/free statistics (see also:
8469 virt-df), migrating between virtualization systems (see also:
8470 virt-p2v), performing partial backups, performing partial guest
8471 clones, cloning guests and changing registry/UUID/hostname info, and
8472 much else besides.
8473
8474 Libguestfs uses Linux kernel and qemu code, and can access any type of
8475 guest filesystem that Linux and qemu can, including but not limited
8476 to: ext2/3/4, btrfs, FAT and NTFS, LVM, many different disk partition
8477 schemes, qcow, qcow2, vmdk.
8478
8479 Libguestfs provides ways to enumerate guest storage (eg. partitions,
8480 LVs, what filesystem is in each LV, etc.).  It can also run commands
8481 in the context of the guest.  Also you can access filesystems over
8482 FUSE.
8483
8484 See also L<Sys::Guestfs::Lib(3)> for a set of useful library
8485 functions for using libguestfs from Perl, including integration
8486 with libvirt.
8487
8488 =head1 ERRORS
8489
8490 All errors turn into calls to C<croak> (see L<Carp(3)>).
8491
8492 =head1 METHODS
8493
8494 =over 4
8495
8496 =cut
8497
8498 package Sys::Guestfs;
8499
8500 use strict;
8501 use warnings;
8502
8503 require XSLoader;
8504 XSLoader::load ('Sys::Guestfs');
8505
8506 =item $h = Sys::Guestfs->new ();
8507
8508 Create a new guestfs handle.
8509
8510 =cut
8511
8512 sub new {
8513   my $proto = shift;
8514   my $class = ref ($proto) || $proto;
8515
8516   my $self = Sys::Guestfs::_create ();
8517   bless $self, $class;
8518   return $self;
8519 }
8520
8521 ";
8522
8523   (* Actions.  We only need to print documentation for these as
8524    * they are pulled in from the XS code automatically.
8525    *)
8526   List.iter (
8527     fun (name, style, _, flags, _, _, longdesc) ->
8528       if not (List.mem NotInDocs flags) then (
8529         let longdesc = replace_str longdesc "C<guestfs_" "C<$h-E<gt>" in
8530         pr "=item ";
8531         generate_perl_prototype name style;
8532         pr "\n\n";
8533         pr "%s\n\n" longdesc;
8534         if List.mem ProtocolLimitWarning flags then
8535           pr "%s\n\n" protocol_limit_warning;
8536         if List.mem DangerWillRobinson flags then
8537           pr "%s\n\n" danger_will_robinson;
8538         match deprecation_notice flags with
8539         | None -> ()
8540         | Some txt -> pr "%s\n\n" txt
8541       )
8542   ) all_functions_sorted;
8543
8544   (* End of file. *)
8545   pr "\
8546 =cut
8547
8548 1;
8549
8550 =back
8551
8552 =head1 COPYRIGHT
8553
8554 Copyright (C) %s Red Hat Inc.
8555
8556 =head1 LICENSE
8557
8558 Please see the file COPYING.LIB for the full license.
8559
8560 =head1 SEE ALSO
8561
8562 L<guestfs(3)>,
8563 L<guestfish(1)>,
8564 L<http://libguestfs.org>,
8565 L<Sys::Guestfs::Lib(3)>.
8566
8567 =cut
8568 " copyright_years
8569
8570 and generate_perl_prototype name style =
8571   (match fst style with
8572    | RErr -> ()
8573    | RBool n
8574    | RInt n
8575    | RInt64 n
8576    | RConstString n
8577    | RConstOptString n
8578    | RString n
8579    | RBufferOut n -> pr "$%s = " n
8580    | RStruct (n,_)
8581    | RHashtable n -> pr "%%%s = " n
8582    | RStringList n
8583    | RStructList (n,_) -> pr "@%s = " n
8584   );
8585   pr "$h->%s (" name;
8586   let comma = ref false in
8587   List.iter (
8588     fun arg ->
8589       if !comma then pr ", ";
8590       comma := true;
8591       match arg with
8592       | Pathname n | Device n | Dev_or_Path n | String n
8593       | OptString n | Bool n | Int n | Int64 n | FileIn n | FileOut n ->
8594           pr "$%s" n
8595       | StringList n | DeviceList n ->
8596           pr "\\@%s" n
8597   ) (snd style);
8598   pr ");"
8599
8600 (* Generate Python C module. *)
8601 and generate_python_c () =
8602   generate_header CStyle LGPLv2plus;
8603
8604   pr "\
8605 #include <Python.h>
8606
8607 #include <stdio.h>
8608 #include <stdlib.h>
8609 #include <assert.h>
8610
8611 #include \"guestfs.h\"
8612
8613 typedef struct {
8614   PyObject_HEAD
8615   guestfs_h *g;
8616 } Pyguestfs_Object;
8617
8618 static guestfs_h *
8619 get_handle (PyObject *obj)
8620 {
8621   assert (obj);
8622   assert (obj != Py_None);
8623   return ((Pyguestfs_Object *) obj)->g;
8624 }
8625
8626 static PyObject *
8627 put_handle (guestfs_h *g)
8628 {
8629   assert (g);
8630   return
8631     PyCObject_FromVoidPtrAndDesc ((void *) g, (char *) \"guestfs_h\", NULL);
8632 }
8633
8634 /* This list should be freed (but not the strings) after use. */
8635 static char **
8636 get_string_list (PyObject *obj)
8637 {
8638   int i, len;
8639   char **r;
8640
8641   assert (obj);
8642
8643   if (!PyList_Check (obj)) {
8644     PyErr_SetString (PyExc_RuntimeError, \"expecting a list parameter\");
8645     return NULL;
8646   }
8647
8648   len = PyList_Size (obj);
8649   r = malloc (sizeof (char *) * (len+1));
8650   if (r == NULL) {
8651     PyErr_SetString (PyExc_RuntimeError, \"get_string_list: out of memory\");
8652     return NULL;
8653   }
8654
8655   for (i = 0; i < len; ++i)
8656     r[i] = PyString_AsString (PyList_GetItem (obj, i));
8657   r[len] = NULL;
8658
8659   return r;
8660 }
8661
8662 static PyObject *
8663 put_string_list (char * const * const argv)
8664 {
8665   PyObject *list;
8666   int argc, i;
8667
8668   for (argc = 0; argv[argc] != NULL; ++argc)
8669     ;
8670
8671   list = PyList_New (argc);
8672   for (i = 0; i < argc; ++i)
8673     PyList_SetItem (list, i, PyString_FromString (argv[i]));
8674
8675   return list;
8676 }
8677
8678 static PyObject *
8679 put_table (char * const * const argv)
8680 {
8681   PyObject *list, *item;
8682   int argc, i;
8683
8684   for (argc = 0; argv[argc] != NULL; ++argc)
8685     ;
8686
8687   list = PyList_New (argc >> 1);
8688   for (i = 0; i < argc; i += 2) {
8689     item = PyTuple_New (2);
8690     PyTuple_SetItem (item, 0, PyString_FromString (argv[i]));
8691     PyTuple_SetItem (item, 1, PyString_FromString (argv[i+1]));
8692     PyList_SetItem (list, i >> 1, item);
8693   }
8694
8695   return list;
8696 }
8697
8698 static void
8699 free_strings (char **argv)
8700 {
8701   int argc;
8702
8703   for (argc = 0; argv[argc] != NULL; ++argc)
8704     free (argv[argc]);
8705   free (argv);
8706 }
8707
8708 static PyObject *
8709 py_guestfs_create (PyObject *self, PyObject *args)
8710 {
8711   guestfs_h *g;
8712
8713   g = guestfs_create ();
8714   if (g == NULL) {
8715     PyErr_SetString (PyExc_RuntimeError,
8716                      \"guestfs.create: failed to allocate handle\");
8717     return NULL;
8718   }
8719   guestfs_set_error_handler (g, NULL, NULL);
8720   return put_handle (g);
8721 }
8722
8723 static PyObject *
8724 py_guestfs_close (PyObject *self, PyObject *args)
8725 {
8726   PyObject *py_g;
8727   guestfs_h *g;
8728
8729   if (!PyArg_ParseTuple (args, (char *) \"O:guestfs_close\", &py_g))
8730     return NULL;
8731   g = get_handle (py_g);
8732
8733   guestfs_close (g);
8734
8735   Py_INCREF (Py_None);
8736   return Py_None;
8737 }
8738
8739 ";
8740
8741   let emit_put_list_function typ =
8742     pr "static PyObject *\n";
8743     pr "put_%s_list (struct guestfs_%s_list *%ss)\n" typ typ typ;
8744     pr "{\n";
8745     pr "  PyObject *list;\n";
8746     pr "  int i;\n";
8747     pr "\n";
8748     pr "  list = PyList_New (%ss->len);\n" typ;
8749     pr "  for (i = 0; i < %ss->len; ++i)\n" typ;
8750     pr "    PyList_SetItem (list, i, put_%s (&%ss->val[i]));\n" typ typ;
8751     pr "  return list;\n";
8752     pr "};\n";
8753     pr "\n"
8754   in
8755
8756   (* Structures, turned into Python dictionaries. *)
8757   List.iter (
8758     fun (typ, cols) ->
8759       pr "static PyObject *\n";
8760       pr "put_%s (struct guestfs_%s *%s)\n" typ typ typ;
8761       pr "{\n";
8762       pr "  PyObject *dict;\n";
8763       pr "\n";
8764       pr "  dict = PyDict_New ();\n";
8765       List.iter (
8766         function
8767         | name, FString ->
8768             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8769             pr "                        PyString_FromString (%s->%s));\n"
8770               typ name
8771         | name, FBuffer ->
8772             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8773             pr "                        PyString_FromStringAndSize (%s->%s, %s->%s_len));\n"
8774               typ name typ name
8775         | name, FUUID ->
8776             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8777             pr "                        PyString_FromStringAndSize (%s->%s, 32));\n"
8778               typ name
8779         | name, (FBytes|FUInt64) ->
8780             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8781             pr "                        PyLong_FromUnsignedLongLong (%s->%s));\n"
8782               typ name
8783         | name, FInt64 ->
8784             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8785             pr "                        PyLong_FromLongLong (%s->%s));\n"
8786               typ name
8787         | name, FUInt32 ->
8788             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8789             pr "                        PyLong_FromUnsignedLong (%s->%s));\n"
8790               typ name
8791         | name, FInt32 ->
8792             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8793             pr "                        PyLong_FromLong (%s->%s));\n"
8794               typ name
8795         | name, FOptPercent ->
8796             pr "  if (%s->%s >= 0)\n" typ name;
8797             pr "    PyDict_SetItemString (dict, \"%s\",\n" name;
8798             pr "                          PyFloat_FromDouble ((double) %s->%s));\n"
8799               typ name;
8800             pr "  else {\n";
8801             pr "    Py_INCREF (Py_None);\n";
8802             pr "    PyDict_SetItemString (dict, \"%s\", Py_None);\n" name;
8803             pr "  }\n"
8804         | name, FChar ->
8805             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8806             pr "                        PyString_FromStringAndSize (&dirent->%s, 1));\n" name
8807       ) cols;
8808       pr "  return dict;\n";
8809       pr "};\n";
8810       pr "\n";
8811
8812   ) structs;
8813
8814   (* Emit a put_TYPE_list function definition only if that function is used. *)
8815   List.iter (
8816     function
8817     | typ, (RStructListOnly | RStructAndList) ->
8818         (* generate the function for typ *)
8819         emit_put_list_function typ
8820     | typ, _ -> () (* empty *)
8821   ) (rstructs_used_by all_functions);
8822
8823   (* Python wrapper functions. *)
8824   List.iter (
8825     fun (name, style, _, _, _, _, _) ->
8826       pr "static PyObject *\n";
8827       pr "py_guestfs_%s (PyObject *self, PyObject *args)\n" name;
8828       pr "{\n";
8829
8830       pr "  PyObject *py_g;\n";
8831       pr "  guestfs_h *g;\n";
8832       pr "  PyObject *py_r;\n";
8833
8834       let error_code =
8835         match fst style with
8836         | RErr | RInt _ | RBool _ -> pr "  int r;\n"; "-1"
8837         | RInt64 _ -> pr "  int64_t r;\n"; "-1"
8838         | RConstString _ | RConstOptString _ ->
8839             pr "  const char *r;\n"; "NULL"
8840         | RString _ -> pr "  char *r;\n"; "NULL"
8841         | RStringList _ | RHashtable _ -> pr "  char **r;\n"; "NULL"
8842         | RStruct (_, typ) -> pr "  struct guestfs_%s *r;\n" typ; "NULL"
8843         | RStructList (_, typ) ->
8844             pr "  struct guestfs_%s_list *r;\n" typ; "NULL"
8845         | RBufferOut _ ->
8846             pr "  char *r;\n";
8847             pr "  size_t size;\n";
8848             "NULL" in
8849
8850       List.iter (
8851         function
8852         | Pathname n | Device n | Dev_or_Path n | String n | FileIn n | FileOut n ->
8853             pr "  const char *%s;\n" n
8854         | OptString n -> pr "  const char *%s;\n" n
8855         | StringList n | DeviceList n ->
8856             pr "  PyObject *py_%s;\n" n;
8857             pr "  char **%s;\n" n
8858         | Bool n -> pr "  int %s;\n" n
8859         | Int n -> pr "  int %s;\n" n
8860         | Int64 n -> pr "  long long %s;\n" n
8861       ) (snd style);
8862
8863       pr "\n";
8864
8865       (* Convert the parameters. *)
8866       pr "  if (!PyArg_ParseTuple (args, (char *) \"O";
8867       List.iter (
8868         function
8869         | Pathname _ | Device _ | Dev_or_Path _ | String _ | FileIn _ | FileOut _ -> pr "s"
8870         | OptString _ -> pr "z"
8871         | StringList _ | DeviceList _ -> pr "O"
8872         | Bool _ -> pr "i" (* XXX Python has booleans? *)
8873         | Int _ -> pr "i"
8874         | Int64 _ -> pr "L" (* XXX Whoever thought it was a good idea to
8875                              * emulate C's int/long/long long in Python?
8876                              *)
8877       ) (snd style);
8878       pr ":guestfs_%s\",\n" name;
8879       pr "                         &py_g";
8880       List.iter (
8881         function
8882         | Pathname n | Device n | Dev_or_Path n | String n | FileIn n | FileOut n -> pr ", &%s" n
8883         | OptString n -> pr ", &%s" n
8884         | StringList n | DeviceList n -> pr ", &py_%s" n
8885         | Bool n -> pr ", &%s" n
8886         | Int n -> pr ", &%s" n
8887         | Int64 n -> pr ", &%s" n
8888       ) (snd style);
8889
8890       pr "))\n";
8891       pr "    return NULL;\n";
8892
8893       pr "  g = get_handle (py_g);\n";
8894       List.iter (
8895         function
8896         | Pathname _ | Device _ | Dev_or_Path _ | String _
8897         | FileIn _ | FileOut _ | OptString _ | Bool _ | Int _ | Int64 _ -> ()
8898         | StringList n | DeviceList n ->
8899             pr "  %s = get_string_list (py_%s);\n" n n;
8900             pr "  if (!%s) return NULL;\n" n
8901       ) (snd style);
8902
8903       pr "\n";
8904
8905       pr "  r = guestfs_%s " name;
8906       generate_c_call_args ~handle:"g" style;
8907       pr ";\n";
8908
8909       List.iter (
8910         function
8911         | Pathname _ | Device _ | Dev_or_Path _ | String _
8912         | FileIn _ | FileOut _ | OptString _ | Bool _ | Int _ | Int64 _ -> ()
8913         | StringList n | DeviceList n ->
8914             pr "  free (%s);\n" n
8915       ) (snd style);
8916
8917       pr "  if (r == %s) {\n" error_code;
8918       pr "    PyErr_SetString (PyExc_RuntimeError, guestfs_last_error (g));\n";
8919       pr "    return NULL;\n";
8920       pr "  }\n";
8921       pr "\n";
8922
8923       (match fst style with
8924        | RErr ->
8925            pr "  Py_INCREF (Py_None);\n";
8926            pr "  py_r = Py_None;\n"
8927        | RInt _
8928        | RBool _ -> pr "  py_r = PyInt_FromLong ((long) r);\n"
8929        | RInt64 _ -> pr "  py_r = PyLong_FromLongLong (r);\n"
8930        | RConstString _ -> pr "  py_r = PyString_FromString (r);\n"
8931        | RConstOptString _ ->
8932            pr "  if (r)\n";
8933            pr "    py_r = PyString_FromString (r);\n";
8934            pr "  else {\n";
8935            pr "    Py_INCREF (Py_None);\n";
8936            pr "    py_r = Py_None;\n";
8937            pr "  }\n"
8938        | RString _ ->
8939            pr "  py_r = PyString_FromString (r);\n";
8940            pr "  free (r);\n"
8941        | RStringList _ ->
8942            pr "  py_r = put_string_list (r);\n";
8943            pr "  free_strings (r);\n"
8944        | RStruct (_, typ) ->
8945            pr "  py_r = put_%s (r);\n" typ;
8946            pr "  guestfs_free_%s (r);\n" typ
8947        | RStructList (_, typ) ->
8948            pr "  py_r = put_%s_list (r);\n" typ;
8949            pr "  guestfs_free_%s_list (r);\n" typ
8950        | RHashtable n ->
8951            pr "  py_r = put_table (r);\n";
8952            pr "  free_strings (r);\n"
8953        | RBufferOut _ ->
8954            pr "  py_r = PyString_FromStringAndSize (r, size);\n";
8955            pr "  free (r);\n"
8956       );
8957
8958       pr "  return py_r;\n";
8959       pr "}\n";
8960       pr "\n"
8961   ) all_functions;
8962
8963   (* Table of functions. *)
8964   pr "static PyMethodDef methods[] = {\n";
8965   pr "  { (char *) \"create\", py_guestfs_create, METH_VARARGS, NULL },\n";
8966   pr "  { (char *) \"close\", py_guestfs_close, METH_VARARGS, NULL },\n";
8967   List.iter (
8968     fun (name, _, _, _, _, _, _) ->
8969       pr "  { (char *) \"%s\", py_guestfs_%s, METH_VARARGS, NULL },\n"
8970         name name
8971   ) all_functions;
8972   pr "  { NULL, NULL, 0, NULL }\n";
8973   pr "};\n";
8974   pr "\n";
8975
8976   (* Init function. *)
8977   pr "\
8978 void
8979 initlibguestfsmod (void)
8980 {
8981   static int initialized = 0;
8982
8983   if (initialized) return;
8984   Py_InitModule ((char *) \"libguestfsmod\", methods);
8985   initialized = 1;
8986 }
8987 "
8988
8989 (* Generate Python module. *)
8990 and generate_python_py () =
8991   generate_header HashStyle LGPLv2plus;
8992
8993   pr "\
8994 u\"\"\"Python bindings for libguestfs
8995
8996 import guestfs
8997 g = guestfs.GuestFS ()
8998 g.add_drive (\"guest.img\")
8999 g.launch ()
9000 parts = g.list_partitions ()
9001
9002 The guestfs module provides a Python binding to the libguestfs API
9003 for examining and modifying virtual machine disk images.
9004
9005 Amongst the things this is good for: making batch configuration
9006 changes to guests, getting disk used/free statistics (see also:
9007 virt-df), migrating between virtualization systems (see also:
9008 virt-p2v), performing partial backups, performing partial guest
9009 clones, cloning guests and changing registry/UUID/hostname info, and
9010 much else besides.
9011
9012 Libguestfs uses Linux kernel and qemu code, and can access any type of
9013 guest filesystem that Linux and qemu can, including but not limited
9014 to: ext2/3/4, btrfs, FAT and NTFS, LVM, many different disk partition
9015 schemes, qcow, qcow2, vmdk.
9016
9017 Libguestfs provides ways to enumerate guest storage (eg. partitions,
9018 LVs, what filesystem is in each LV, etc.).  It can also run commands
9019 in the context of the guest.  Also you can access filesystems over
9020 FUSE.
9021
9022 Errors which happen while using the API are turned into Python
9023 RuntimeError exceptions.
9024
9025 To create a guestfs handle you usually have to perform the following
9026 sequence of calls:
9027
9028 # Create the handle, call add_drive at least once, and possibly
9029 # several times if the guest has multiple block devices:
9030 g = guestfs.GuestFS ()
9031 g.add_drive (\"guest.img\")
9032
9033 # Launch the qemu subprocess and wait for it to become ready:
9034 g.launch ()
9035
9036 # Now you can issue commands, for example:
9037 logvols = g.lvs ()
9038
9039 \"\"\"
9040
9041 import libguestfsmod
9042
9043 class GuestFS:
9044     \"\"\"Instances of this class are libguestfs API handles.\"\"\"
9045
9046     def __init__ (self):
9047         \"\"\"Create a new libguestfs handle.\"\"\"
9048         self._o = libguestfsmod.create ()
9049
9050     def __del__ (self):
9051         libguestfsmod.close (self._o)
9052
9053 ";
9054
9055   List.iter (
9056     fun (name, style, _, flags, _, _, longdesc) ->
9057       pr "    def %s " name;
9058       generate_py_call_args ~handle:"self" (snd style);
9059       pr ":\n";
9060
9061       if not (List.mem NotInDocs flags) then (
9062         let doc = replace_str longdesc "C<guestfs_" "C<g." in
9063         let doc =
9064           match fst style with
9065           | RErr | RInt _ | RInt64 _ | RBool _
9066           | RConstOptString _ | RConstString _
9067           | RString _ | RBufferOut _ -> doc
9068           | RStringList _ ->
9069               doc ^ "\n\nThis function returns a list of strings."
9070           | RStruct (_, typ) ->
9071               doc ^ sprintf "\n\nThis function returns a dictionary, with keys matching the various fields in the guestfs_%s structure." typ
9072           | RStructList (_, typ) ->
9073               doc ^ sprintf "\n\nThis function returns a list of %ss.  Each %s is represented as a dictionary." typ typ
9074           | RHashtable _ ->
9075               doc ^ "\n\nThis function returns a dictionary." in
9076         let doc =
9077           if List.mem ProtocolLimitWarning flags then
9078             doc ^ "\n\n" ^ protocol_limit_warning
9079           else doc in
9080         let doc =
9081           if List.mem DangerWillRobinson flags then
9082             doc ^ "\n\n" ^ danger_will_robinson
9083           else doc in
9084         let doc =
9085           match deprecation_notice flags with
9086           | None -> doc
9087           | Some txt -> doc ^ "\n\n" ^ txt in
9088         let doc = pod2text ~width:60 name doc in
9089         let doc = List.map (fun line -> replace_str line "\\" "\\\\") doc in
9090         let doc = String.concat "\n        " doc in
9091         pr "        u\"\"\"%s\"\"\"\n" doc;
9092       );
9093       pr "        return libguestfsmod.%s " name;
9094       generate_py_call_args ~handle:"self._o" (snd style);
9095       pr "\n";
9096       pr "\n";
9097   ) all_functions
9098
9099 (* Generate Python call arguments, eg "(handle, foo, bar)" *)
9100 and generate_py_call_args ~handle args =
9101   pr "(%s" handle;
9102   List.iter (fun arg -> pr ", %s" (name_of_argt arg)) args;
9103   pr ")"
9104
9105 (* Useful if you need the longdesc POD text as plain text.  Returns a
9106  * list of lines.
9107  *
9108  * Because this is very slow (the slowest part of autogeneration),
9109  * we memoize the results.
9110  *)
9111 and pod2text ~width name longdesc =
9112   let key = width, name, longdesc in
9113   try Hashtbl.find pod2text_memo key
9114   with Not_found ->
9115     let filename, chan = Filename.open_temp_file "gen" ".tmp" in
9116     fprintf chan "=head1 %s\n\n%s\n" name longdesc;
9117     close_out chan;
9118     let cmd = sprintf "pod2text -w %d %s" width (Filename.quote filename) in
9119     let chan = open_process_in cmd in
9120     let lines = ref [] in
9121     let rec loop i =
9122       let line = input_line chan in
9123       if i = 1 then             (* discard the first line of output *)
9124         loop (i+1)
9125       else (
9126         let line = triml line in
9127         lines := line :: !lines;
9128         loop (i+1)
9129       ) in
9130     let lines = try loop 1 with End_of_file -> List.rev !lines in
9131     unlink filename;
9132     (match close_process_in chan with
9133      | WEXITED 0 -> ()
9134      | WEXITED i ->
9135          failwithf "pod2text: process exited with non-zero status (%d)" i
9136      | WSIGNALED i | WSTOPPED i ->
9137          failwithf "pod2text: process signalled or stopped by signal %d" i
9138     );
9139     Hashtbl.add pod2text_memo key lines;
9140     pod2text_memo_updated ();
9141     lines
9142
9143 (* Generate ruby bindings. *)
9144 and generate_ruby_c () =
9145   generate_header CStyle LGPLv2plus;
9146
9147   pr "\
9148 #include <stdio.h>
9149 #include <stdlib.h>
9150
9151 #include <ruby.h>
9152
9153 #include \"guestfs.h\"
9154
9155 #include \"extconf.h\"
9156
9157 /* For Ruby < 1.9 */
9158 #ifndef RARRAY_LEN
9159 #define RARRAY_LEN(r) (RARRAY((r))->len)
9160 #endif
9161
9162 static VALUE m_guestfs;                 /* guestfs module */
9163 static VALUE c_guestfs;                 /* guestfs_h handle */
9164 static VALUE e_Error;                   /* used for all errors */
9165
9166 static void ruby_guestfs_free (void *p)
9167 {
9168   if (!p) return;
9169   guestfs_close ((guestfs_h *) p);
9170 }
9171
9172 static VALUE ruby_guestfs_create (VALUE m)
9173 {
9174   guestfs_h *g;
9175
9176   g = guestfs_create ();
9177   if (!g)
9178     rb_raise (e_Error, \"failed to create guestfs handle\");
9179
9180   /* Don't print error messages to stderr by default. */
9181   guestfs_set_error_handler (g, NULL, NULL);
9182
9183   /* Wrap it, and make sure the close function is called when the
9184    * handle goes away.
9185    */
9186   return Data_Wrap_Struct (c_guestfs, NULL, ruby_guestfs_free, g);
9187 }
9188
9189 static VALUE ruby_guestfs_close (VALUE gv)
9190 {
9191   guestfs_h *g;
9192   Data_Get_Struct (gv, guestfs_h, g);
9193
9194   ruby_guestfs_free (g);
9195   DATA_PTR (gv) = NULL;
9196
9197   return Qnil;
9198 }
9199
9200 ";
9201
9202   List.iter (
9203     fun (name, style, _, _, _, _, _) ->
9204       pr "static VALUE ruby_guestfs_%s (VALUE gv" name;
9205       List.iter (fun arg -> pr ", VALUE %sv" (name_of_argt arg)) (snd style);
9206       pr ")\n";
9207       pr "{\n";
9208       pr "  guestfs_h *g;\n";
9209       pr "  Data_Get_Struct (gv, guestfs_h, g);\n";
9210       pr "  if (!g)\n";
9211       pr "    rb_raise (rb_eArgError, \"%%s: used handle after closing it\", \"%s\");\n"
9212         name;
9213       pr "\n";
9214
9215       List.iter (
9216         function
9217         | Pathname n | Device n | Dev_or_Path n | String n | FileIn n | FileOut n ->
9218             pr "  Check_Type (%sv, T_STRING);\n" n;
9219             pr "  const char *%s = StringValueCStr (%sv);\n" n n;
9220             pr "  if (!%s)\n" n;
9221             pr "    rb_raise (rb_eTypeError, \"expected string for parameter %%s of %%s\",\n";
9222             pr "              \"%s\", \"%s\");\n" n name
9223         | OptString n ->
9224             pr "  const char *%s = !NIL_P (%sv) ? StringValueCStr (%sv) : NULL;\n" n n n
9225         | StringList n | DeviceList n ->
9226             pr "  char **%s;\n" n;
9227             pr "  Check_Type (%sv, T_ARRAY);\n" n;
9228             pr "  {\n";
9229             pr "    int i, len;\n";
9230             pr "    len = RARRAY_LEN (%sv);\n" n;
9231             pr "    %s = guestfs_safe_malloc (g, sizeof (char *) * (len+1));\n"
9232               n;
9233             pr "    for (i = 0; i < len; ++i) {\n";
9234             pr "      VALUE v = rb_ary_entry (%sv, i);\n" n;
9235             pr "      %s[i] = StringValueCStr (v);\n" n;
9236             pr "    }\n";
9237             pr "    %s[len] = NULL;\n" n;
9238             pr "  }\n";
9239         | Bool n ->
9240             pr "  int %s = RTEST (%sv);\n" n n
9241         | Int n ->
9242             pr "  int %s = NUM2INT (%sv);\n" n n
9243         | Int64 n ->
9244             pr "  long long %s = NUM2LL (%sv);\n" n n
9245       ) (snd style);
9246       pr "\n";
9247
9248       let error_code =
9249         match fst style with
9250         | RErr | RInt _ | RBool _ -> pr "  int r;\n"; "-1"
9251         | RInt64 _ -> pr "  int64_t r;\n"; "-1"
9252         | RConstString _ | RConstOptString _ ->
9253             pr "  const char *r;\n"; "NULL"
9254         | RString _ -> pr "  char *r;\n"; "NULL"
9255         | RStringList _ | RHashtable _ -> pr "  char **r;\n"; "NULL"
9256         | RStruct (_, typ) -> pr "  struct guestfs_%s *r;\n" typ; "NULL"
9257         | RStructList (_, typ) ->
9258             pr "  struct guestfs_%s_list *r;\n" typ; "NULL"
9259         | RBufferOut _ ->
9260             pr "  char *r;\n";
9261             pr "  size_t size;\n";
9262             "NULL" in
9263       pr "\n";
9264
9265       pr "  r = guestfs_%s " name;
9266       generate_c_call_args ~handle:"g" style;
9267       pr ";\n";
9268
9269       List.iter (
9270         function
9271         | Pathname _ | Device _ | Dev_or_Path _ | String _
9272         | FileIn _ | FileOut _ | OptString _ | Bool _ | Int _ | Int64 _ -> ()
9273         | StringList n | DeviceList n ->
9274             pr "  free (%s);\n" n
9275       ) (snd style);
9276
9277       pr "  if (r == %s)\n" error_code;
9278       pr "    rb_raise (e_Error, \"%%s\", guestfs_last_error (g));\n";
9279       pr "\n";
9280
9281       (match fst style with
9282        | RErr ->
9283            pr "  return Qnil;\n"
9284        | RInt _ | RBool _ ->
9285            pr "  return INT2NUM (r);\n"
9286        | RInt64 _ ->
9287            pr "  return ULL2NUM (r);\n"
9288        | RConstString _ ->
9289            pr "  return rb_str_new2 (r);\n";
9290        | RConstOptString _ ->
9291            pr "  if (r)\n";
9292            pr "    return rb_str_new2 (r);\n";
9293            pr "  else\n";
9294            pr "    return Qnil;\n";
9295        | RString _ ->
9296            pr "  VALUE rv = rb_str_new2 (r);\n";
9297            pr "  free (r);\n";
9298            pr "  return rv;\n";
9299        | RStringList _ ->
9300            pr "  int i, len = 0;\n";
9301            pr "  for (i = 0; r[i] != NULL; ++i) len++;\n";
9302            pr "  VALUE rv = rb_ary_new2 (len);\n";
9303            pr "  for (i = 0; r[i] != NULL; ++i) {\n";
9304            pr "    rb_ary_push (rv, rb_str_new2 (r[i]));\n";
9305            pr "    free (r[i]);\n";
9306            pr "  }\n";
9307            pr "  free (r);\n";
9308            pr "  return rv;\n"
9309        | RStruct (_, typ) ->
9310            let cols = cols_of_struct typ in
9311            generate_ruby_struct_code typ cols
9312        | RStructList (_, typ) ->
9313            let cols = cols_of_struct typ in
9314            generate_ruby_struct_list_code typ cols
9315        | RHashtable _ ->
9316            pr "  VALUE rv = rb_hash_new ();\n";
9317            pr "  int i;\n";
9318            pr "  for (i = 0; r[i] != NULL; i+=2) {\n";
9319            pr "    rb_hash_aset (rv, rb_str_new2 (r[i]), rb_str_new2 (r[i+1]));\n";
9320            pr "    free (r[i]);\n";
9321            pr "    free (r[i+1]);\n";
9322            pr "  }\n";
9323            pr "  free (r);\n";
9324            pr "  return rv;\n"
9325        | RBufferOut _ ->
9326            pr "  VALUE rv = rb_str_new (r, size);\n";
9327            pr "  free (r);\n";
9328            pr "  return rv;\n";
9329       );
9330
9331       pr "}\n";
9332       pr "\n"
9333   ) all_functions;
9334
9335   pr "\
9336 /* Initialize the module. */
9337 void Init__guestfs ()
9338 {
9339   m_guestfs = rb_define_module (\"Guestfs\");
9340   c_guestfs = rb_define_class_under (m_guestfs, \"Guestfs\", rb_cObject);
9341   e_Error = rb_define_class_under (m_guestfs, \"Error\", rb_eStandardError);
9342
9343   rb_define_module_function (m_guestfs, \"create\", ruby_guestfs_create, 0);
9344   rb_define_method (c_guestfs, \"close\", ruby_guestfs_close, 0);
9345
9346 ";
9347   (* Define the rest of the methods. *)
9348   List.iter (
9349     fun (name, style, _, _, _, _, _) ->
9350       pr "  rb_define_method (c_guestfs, \"%s\",\n" name;
9351       pr "        ruby_guestfs_%s, %d);\n" name (List.length (snd style))
9352   ) all_functions;
9353
9354   pr "}\n"
9355
9356 (* Ruby code to return a struct. *)
9357 and generate_ruby_struct_code typ cols =
9358   pr "  VALUE rv = rb_hash_new ();\n";
9359   List.iter (
9360     function
9361     | name, FString ->
9362         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), rb_str_new2 (r->%s));\n" name name
9363     | name, FBuffer ->
9364         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), rb_str_new (r->%s, r->%s_len));\n" name name name
9365     | name, FUUID ->
9366         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), rb_str_new (r->%s, 32));\n" name name
9367     | name, (FBytes|FUInt64) ->
9368         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), ULL2NUM (r->%s));\n" name name
9369     | name, FInt64 ->
9370         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), LL2NUM (r->%s));\n" name name
9371     | name, FUInt32 ->
9372         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), UINT2NUM (r->%s));\n" name name
9373     | name, FInt32 ->
9374         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), INT2NUM (r->%s));\n" name name
9375     | name, FOptPercent ->
9376         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), rb_dbl2big (r->%s));\n" name name
9377     | name, FChar -> (* XXX wrong? *)
9378         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), ULL2NUM (r->%s));\n" name name
9379   ) cols;
9380   pr "  guestfs_free_%s (r);\n" typ;
9381   pr "  return rv;\n"
9382
9383 (* Ruby code to return a struct list. *)
9384 and generate_ruby_struct_list_code typ cols =
9385   pr "  VALUE rv = rb_ary_new2 (r->len);\n";
9386   pr "  int i;\n";
9387   pr "  for (i = 0; i < r->len; ++i) {\n";
9388   pr "    VALUE hv = rb_hash_new ();\n";
9389   List.iter (
9390     function
9391     | name, FString ->
9392         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), rb_str_new2 (r->val[i].%s));\n" name name
9393     | name, FBuffer ->
9394         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
9395     | name, FUUID ->
9396         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), rb_str_new (r->val[i].%s, 32));\n" name name
9397     | name, (FBytes|FUInt64) ->
9398         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), ULL2NUM (r->val[i].%s));\n" name name
9399     | name, FInt64 ->
9400         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), LL2NUM (r->val[i].%s));\n" name name
9401     | name, FUInt32 ->
9402         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), UINT2NUM (r->val[i].%s));\n" name name
9403     | name, FInt32 ->
9404         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), INT2NUM (r->val[i].%s));\n" name name
9405     | name, FOptPercent ->
9406         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), rb_dbl2big (r->val[i].%s));\n" name name
9407     | name, FChar -> (* XXX wrong? *)
9408         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), ULL2NUM (r->val[i].%s));\n" name name
9409   ) cols;
9410   pr "    rb_ary_push (rv, hv);\n";
9411   pr "  }\n";
9412   pr "  guestfs_free_%s_list (r);\n" typ;
9413   pr "  return rv;\n"
9414
9415 (* Generate Java bindings GuestFS.java file. *)
9416 and generate_java_java () =
9417   generate_header CStyle LGPLv2plus;
9418
9419   pr "\
9420 package com.redhat.et.libguestfs;
9421
9422 import java.util.HashMap;
9423 import com.redhat.et.libguestfs.LibGuestFSException;
9424 import com.redhat.et.libguestfs.PV;
9425 import com.redhat.et.libguestfs.VG;
9426 import com.redhat.et.libguestfs.LV;
9427 import com.redhat.et.libguestfs.Stat;
9428 import com.redhat.et.libguestfs.StatVFS;
9429 import com.redhat.et.libguestfs.IntBool;
9430 import com.redhat.et.libguestfs.Dirent;
9431
9432 /**
9433  * The GuestFS object is a libguestfs handle.
9434  *
9435  * @author rjones
9436  */
9437 public class GuestFS {
9438   // Load the native code.
9439   static {
9440     System.loadLibrary (\"guestfs_jni\");
9441   }
9442
9443   /**
9444    * The native guestfs_h pointer.
9445    */
9446   long g;
9447
9448   /**
9449    * Create a libguestfs handle.
9450    *
9451    * @throws LibGuestFSException
9452    */
9453   public GuestFS () throws LibGuestFSException
9454   {
9455     g = _create ();
9456   }
9457   private native long _create () throws LibGuestFSException;
9458
9459   /**
9460    * Close a libguestfs handle.
9461    *
9462    * You can also leave handles to be collected by the garbage
9463    * collector, but this method ensures that the resources used
9464    * by the handle are freed up immediately.  If you call any
9465    * other methods after closing the handle, you will get an
9466    * exception.
9467    *
9468    * @throws LibGuestFSException
9469    */
9470   public void close () throws LibGuestFSException
9471   {
9472     if (g != 0)
9473       _close (g);
9474     g = 0;
9475   }
9476   private native void _close (long g) throws LibGuestFSException;
9477
9478   public void finalize () throws LibGuestFSException
9479   {
9480     close ();
9481   }
9482
9483 ";
9484
9485   List.iter (
9486     fun (name, style, _, flags, _, shortdesc, longdesc) ->
9487       if not (List.mem NotInDocs flags); then (
9488         let doc = replace_str longdesc "C<guestfs_" "C<g." in
9489         let doc =
9490           if List.mem ProtocolLimitWarning flags then
9491             doc ^ "\n\n" ^ protocol_limit_warning
9492           else doc in
9493         let doc =
9494           if List.mem DangerWillRobinson flags then
9495             doc ^ "\n\n" ^ danger_will_robinson
9496           else doc in
9497         let doc =
9498           match deprecation_notice flags with
9499           | None -> doc
9500           | Some txt -> doc ^ "\n\n" ^ txt in
9501         let doc = pod2text ~width:60 name doc in
9502         let doc = List.map (            (* RHBZ#501883 *)
9503           function
9504           | "" -> "<p>"
9505           | nonempty -> nonempty
9506         ) doc in
9507         let doc = String.concat "\n   * " doc in
9508
9509         pr "  /**\n";
9510         pr "   * %s\n" shortdesc;
9511         pr "   * <p>\n";
9512         pr "   * %s\n" doc;
9513         pr "   * @throws LibGuestFSException\n";
9514         pr "   */\n";
9515         pr "  ";
9516       );
9517       generate_java_prototype ~public:true ~semicolon:false name style;
9518       pr "\n";
9519       pr "  {\n";
9520       pr "    if (g == 0)\n";
9521       pr "      throw new LibGuestFSException (\"%s: handle is closed\");\n"
9522         name;
9523       pr "    ";
9524       if fst style <> RErr then pr "return ";
9525       pr "_%s " name;
9526       generate_java_call_args ~handle:"g" (snd style);
9527       pr ";\n";
9528       pr "  }\n";
9529       pr "  ";
9530       generate_java_prototype ~privat:true ~native:true name style;
9531       pr "\n";
9532       pr "\n";
9533   ) all_functions;
9534
9535   pr "}\n"
9536
9537 (* Generate Java call arguments, eg "(handle, foo, bar)" *)
9538 and generate_java_call_args ~handle args =
9539   pr "(%s" handle;
9540   List.iter (fun arg -> pr ", %s" (name_of_argt arg)) args;
9541   pr ")"
9542
9543 and generate_java_prototype ?(public=false) ?(privat=false) ?(native=false)
9544     ?(semicolon=true) name style =
9545   if privat then pr "private ";
9546   if public then pr "public ";
9547   if native then pr "native ";
9548
9549   (* return type *)
9550   (match fst style with
9551    | RErr -> pr "void ";
9552    | RInt _ -> pr "int ";
9553    | RInt64 _ -> pr "long ";
9554    | RBool _ -> pr "boolean ";
9555    | RConstString _ | RConstOptString _ | RString _
9556    | RBufferOut _ -> pr "String ";
9557    | RStringList _ -> pr "String[] ";
9558    | RStruct (_, typ) ->
9559        let name = java_name_of_struct typ in
9560        pr "%s " name;
9561    | RStructList (_, typ) ->
9562        let name = java_name_of_struct typ in
9563        pr "%s[] " name;
9564    | RHashtable _ -> pr "HashMap<String,String> ";
9565   );
9566
9567   if native then pr "_%s " name else pr "%s " name;
9568   pr "(";
9569   let needs_comma = ref false in
9570   if native then (
9571     pr "long g";
9572     needs_comma := true
9573   );
9574
9575   (* args *)
9576   List.iter (
9577     fun arg ->
9578       if !needs_comma then pr ", ";
9579       needs_comma := true;
9580
9581       match arg with
9582       | Pathname n
9583       | Device n | Dev_or_Path n
9584       | String n
9585       | OptString n
9586       | FileIn n
9587       | FileOut n ->
9588           pr "String %s" n
9589       | StringList n | DeviceList n ->
9590           pr "String[] %s" n
9591       | Bool n ->
9592           pr "boolean %s" n
9593       | Int n ->
9594           pr "int %s" n
9595       | Int64 n ->
9596           pr "long %s" n
9597   ) (snd style);
9598
9599   pr ")\n";
9600   pr "    throws LibGuestFSException";
9601   if semicolon then pr ";"
9602
9603 and generate_java_struct jtyp cols () =
9604   generate_header CStyle LGPLv2plus;
9605
9606   pr "\
9607 package com.redhat.et.libguestfs;
9608
9609 /**
9610  * Libguestfs %s structure.
9611  *
9612  * @author rjones
9613  * @see GuestFS
9614  */
9615 public class %s {
9616 " jtyp jtyp;
9617
9618   List.iter (
9619     function
9620     | name, FString
9621     | name, FUUID
9622     | name, FBuffer -> pr "  public String %s;\n" name
9623     | name, (FBytes|FUInt64|FInt64) -> pr "  public long %s;\n" name
9624     | name, (FUInt32|FInt32) -> pr "  public int %s;\n" name
9625     | name, FChar -> pr "  public char %s;\n" name
9626     | name, FOptPercent ->
9627         pr "  /* The next field is [0..100] or -1 meaning 'not present': */\n";
9628         pr "  public float %s;\n" name
9629   ) cols;
9630
9631   pr "}\n"
9632
9633 and generate_java_c () =
9634   generate_header CStyle LGPLv2plus;
9635
9636   pr "\
9637 #include <stdio.h>
9638 #include <stdlib.h>
9639 #include <string.h>
9640
9641 #include \"com_redhat_et_libguestfs_GuestFS.h\"
9642 #include \"guestfs.h\"
9643
9644 /* Note that this function returns.  The exception is not thrown
9645  * until after the wrapper function returns.
9646  */
9647 static void
9648 throw_exception (JNIEnv *env, const char *msg)
9649 {
9650   jclass cl;
9651   cl = (*env)->FindClass (env,
9652                           \"com/redhat/et/libguestfs/LibGuestFSException\");
9653   (*env)->ThrowNew (env, cl, msg);
9654 }
9655
9656 JNIEXPORT jlong JNICALL
9657 Java_com_redhat_et_libguestfs_GuestFS__1create
9658   (JNIEnv *env, jobject obj)
9659 {
9660   guestfs_h *g;
9661
9662   g = guestfs_create ();
9663   if (g == NULL) {
9664     throw_exception (env, \"GuestFS.create: failed to allocate handle\");
9665     return 0;
9666   }
9667   guestfs_set_error_handler (g, NULL, NULL);
9668   return (jlong) (long) g;
9669 }
9670
9671 JNIEXPORT void JNICALL
9672 Java_com_redhat_et_libguestfs_GuestFS__1close
9673   (JNIEnv *env, jobject obj, jlong jg)
9674 {
9675   guestfs_h *g = (guestfs_h *) (long) jg;
9676   guestfs_close (g);
9677 }
9678
9679 ";
9680
9681   List.iter (
9682     fun (name, style, _, _, _, _, _) ->
9683       pr "JNIEXPORT ";
9684       (match fst style with
9685        | RErr -> pr "void ";
9686        | RInt _ -> pr "jint ";
9687        | RInt64 _ -> pr "jlong ";
9688        | RBool _ -> pr "jboolean ";
9689        | RConstString _ | RConstOptString _ | RString _
9690        | RBufferOut _ -> pr "jstring ";
9691        | RStruct _ | RHashtable _ ->
9692            pr "jobject ";
9693        | RStringList _ | RStructList _ ->
9694            pr "jobjectArray ";
9695       );
9696       pr "JNICALL\n";
9697       pr "Java_com_redhat_et_libguestfs_GuestFS_";
9698       pr "%s" (replace_str ("_" ^ name) "_" "_1");
9699       pr "\n";
9700       pr "  (JNIEnv *env, jobject obj, jlong jg";
9701       List.iter (
9702         function
9703         | Pathname n
9704         | Device n | Dev_or_Path n
9705         | String n
9706         | OptString n
9707         | FileIn n
9708         | FileOut n ->
9709             pr ", jstring j%s" n
9710         | StringList n | DeviceList n ->
9711             pr ", jobjectArray j%s" n
9712         | Bool n ->
9713             pr ", jboolean j%s" n
9714         | Int n ->
9715             pr ", jint j%s" n
9716         | Int64 n ->
9717             pr ", jlong j%s" n
9718       ) (snd style);
9719       pr ")\n";
9720       pr "{\n";
9721       pr "  guestfs_h *g = (guestfs_h *) (long) jg;\n";
9722       let error_code, no_ret =
9723         match fst style with
9724         | RErr -> pr "  int r;\n"; "-1", ""
9725         | RBool _
9726         | RInt _ -> pr "  int r;\n"; "-1", "0"
9727         | RInt64 _ -> pr "  int64_t r;\n"; "-1", "0"
9728         | RConstString _ -> pr "  const char *r;\n"; "NULL", "NULL"
9729         | RConstOptString _ -> pr "  const char *r;\n"; "NULL", "NULL"
9730         | RString _ ->
9731             pr "  jstring jr;\n";
9732             pr "  char *r;\n"; "NULL", "NULL"
9733         | RStringList _ ->
9734             pr "  jobjectArray jr;\n";
9735             pr "  int r_len;\n";
9736             pr "  jclass cl;\n";
9737             pr "  jstring jstr;\n";
9738             pr "  char **r;\n"; "NULL", "NULL"
9739         | RStruct (_, typ) ->
9740             pr "  jobject jr;\n";
9741             pr "  jclass cl;\n";
9742             pr "  jfieldID fl;\n";
9743             pr "  struct guestfs_%s *r;\n" typ; "NULL", "NULL"
9744         | RStructList (_, typ) ->
9745             pr "  jobjectArray jr;\n";
9746             pr "  jclass cl;\n";
9747             pr "  jfieldID fl;\n";
9748             pr "  jobject jfl;\n";
9749             pr "  struct guestfs_%s_list *r;\n" typ; "NULL", "NULL"
9750         | RHashtable _ -> pr "  char **r;\n"; "NULL", "NULL"
9751         | RBufferOut _ ->
9752             pr "  jstring jr;\n";
9753             pr "  char *r;\n";
9754             pr "  size_t size;\n";
9755             "NULL", "NULL" in
9756       List.iter (
9757         function
9758         | Pathname n
9759         | Device n | Dev_or_Path n
9760         | String n
9761         | OptString n
9762         | FileIn n
9763         | FileOut n ->
9764             pr "  const char *%s;\n" n
9765         | StringList n | DeviceList n ->
9766             pr "  int %s_len;\n" n;
9767             pr "  const char **%s;\n" n
9768         | Bool n
9769         | Int n ->
9770             pr "  int %s;\n" n
9771         | Int64 n ->
9772             pr "  int64_t %s;\n" n
9773       ) (snd style);
9774
9775       let needs_i =
9776         (match fst style with
9777          | RStringList _ | RStructList _ -> true
9778          | RErr | RBool _ | RInt _ | RInt64 _ | RConstString _
9779          | RConstOptString _
9780          | RString _ | RBufferOut _ | RStruct _ | RHashtable _ -> false) ||
9781           List.exists (function
9782                        | StringList _ -> true
9783                        | DeviceList _ -> true
9784                        | _ -> false) (snd style) in
9785       if needs_i then
9786         pr "  int i;\n";
9787
9788       pr "\n";
9789
9790       (* Get the parameters. *)
9791       List.iter (
9792         function
9793         | Pathname n
9794         | Device n | Dev_or_Path n
9795         | String n
9796         | FileIn n
9797         | FileOut n ->
9798             pr "  %s = (*env)->GetStringUTFChars (env, j%s, NULL);\n" n n
9799         | OptString n ->
9800             (* This is completely undocumented, but Java null becomes
9801              * a NULL parameter.
9802              *)
9803             pr "  %s = j%s ? (*env)->GetStringUTFChars (env, j%s, NULL) : NULL;\n" n n n
9804         | StringList n | DeviceList n ->
9805             pr "  %s_len = (*env)->GetArrayLength (env, j%s);\n" n n;
9806             pr "  %s = guestfs_safe_malloc (g, sizeof (char *) * (%s_len+1));\n" n n;
9807             pr "  for (i = 0; i < %s_len; ++i) {\n" n;
9808             pr "    jobject o = (*env)->GetObjectArrayElement (env, j%s, i);\n"
9809               n;
9810             pr "    %s[i] = (*env)->GetStringUTFChars (env, o, NULL);\n" n;
9811             pr "  }\n";
9812             pr "  %s[%s_len] = NULL;\n" n n;
9813         | Bool n
9814         | Int n
9815         | Int64 n ->
9816             pr "  %s = j%s;\n" n n
9817       ) (snd style);
9818
9819       (* Make the call. *)
9820       pr "  r = guestfs_%s " name;
9821       generate_c_call_args ~handle:"g" style;
9822       pr ";\n";
9823
9824       (* Release the parameters. *)
9825       List.iter (
9826         function
9827         | Pathname n
9828         | Device n | Dev_or_Path n
9829         | String n
9830         | FileIn n
9831         | FileOut n ->
9832             pr "  (*env)->ReleaseStringUTFChars (env, j%s, %s);\n" n n
9833         | OptString n ->
9834             pr "  if (j%s)\n" n;
9835             pr "    (*env)->ReleaseStringUTFChars (env, j%s, %s);\n" n n
9836         | StringList n | DeviceList n ->
9837             pr "  for (i = 0; i < %s_len; ++i) {\n" n;
9838             pr "    jobject o = (*env)->GetObjectArrayElement (env, j%s, i);\n"
9839               n;
9840             pr "    (*env)->ReleaseStringUTFChars (env, o, %s[i]);\n" n;
9841             pr "  }\n";
9842             pr "  free (%s);\n" n
9843         | Bool n
9844         | Int n
9845         | Int64 n -> ()
9846       ) (snd style);
9847
9848       (* Check for errors. *)
9849       pr "  if (r == %s) {\n" error_code;
9850       pr "    throw_exception (env, guestfs_last_error (g));\n";
9851       pr "    return %s;\n" no_ret;
9852       pr "  }\n";
9853
9854       (* Return value. *)
9855       (match fst style with
9856        | RErr -> ()
9857        | RInt _ -> pr "  return (jint) r;\n"
9858        | RBool _ -> pr "  return (jboolean) r;\n"
9859        | RInt64 _ -> pr "  return (jlong) r;\n"
9860        | RConstString _ -> pr "  return (*env)->NewStringUTF (env, r);\n"
9861        | RConstOptString _ ->
9862            pr "  return (*env)->NewStringUTF (env, r); /* XXX r NULL? */\n"
9863        | RString _ ->
9864            pr "  jr = (*env)->NewStringUTF (env, r);\n";
9865            pr "  free (r);\n";
9866            pr "  return jr;\n"
9867        | RStringList _ ->
9868            pr "  for (r_len = 0; r[r_len] != NULL; ++r_len) ;\n";
9869            pr "  cl = (*env)->FindClass (env, \"java/lang/String\");\n";
9870            pr "  jstr = (*env)->NewStringUTF (env, \"\");\n";
9871            pr "  jr = (*env)->NewObjectArray (env, r_len, cl, jstr);\n";
9872            pr "  for (i = 0; i < r_len; ++i) {\n";
9873            pr "    jstr = (*env)->NewStringUTF (env, r[i]);\n";
9874            pr "    (*env)->SetObjectArrayElement (env, jr, i, jstr);\n";
9875            pr "    free (r[i]);\n";
9876            pr "  }\n";
9877            pr "  free (r);\n";
9878            pr "  return jr;\n"
9879        | RStruct (_, typ) ->
9880            let jtyp = java_name_of_struct typ in
9881            let cols = cols_of_struct typ in
9882            generate_java_struct_return typ jtyp cols
9883        | RStructList (_, typ) ->
9884            let jtyp = java_name_of_struct typ in
9885            let cols = cols_of_struct typ in
9886            generate_java_struct_list_return typ jtyp cols
9887        | RHashtable _ ->
9888            (* XXX *)
9889            pr "  throw_exception (env, \"%s: internal error: please let us know how to make a Java HashMap from JNI bindings!\");\n" name;
9890            pr "  return NULL;\n"
9891        | RBufferOut _ ->
9892            pr "  jr = (*env)->NewStringUTF (env, r); /* XXX size */\n";
9893            pr "  free (r);\n";
9894            pr "  return jr;\n"
9895       );
9896
9897       pr "}\n";
9898       pr "\n"
9899   ) all_functions
9900
9901 and generate_java_struct_return typ jtyp cols =
9902   pr "  cl = (*env)->FindClass (env, \"com/redhat/et/libguestfs/%s\");\n" jtyp;
9903   pr "  jr = (*env)->AllocObject (env, cl);\n";
9904   List.iter (
9905     function
9906     | name, FString ->
9907         pr "  fl = (*env)->GetFieldID (env, cl, \"%s\", \"Ljava/lang/String;\");\n" name;
9908         pr "  (*env)->SetObjectField (env, jr, fl, (*env)->NewStringUTF (env, r->%s));\n" name;
9909     | name, FUUID ->
9910         pr "  {\n";
9911         pr "    char s[33];\n";
9912         pr "    memcpy (s, r->%s, 32);\n" name;
9913         pr "    s[32] = 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, FBuffer ->
9918         pr "  {\n";
9919         pr "    int len = r->%s_len;\n" name;
9920         pr "    char s[len+1];\n";
9921         pr "    memcpy (s, r->%s, len);\n" name;
9922         pr "    s[len] = 0;\n";
9923         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"Ljava/lang/String;\");\n" name;
9924         pr "    (*env)->SetObjectField (env, jr, fl, (*env)->NewStringUTF (env, s));\n";
9925         pr "  }\n";
9926     | name, (FBytes|FUInt64|FInt64) ->
9927         pr "  fl = (*env)->GetFieldID (env, cl, \"%s\", \"J\");\n" name;
9928         pr "  (*env)->SetLongField (env, jr, fl, r->%s);\n" name;
9929     | name, (FUInt32|FInt32) ->
9930         pr "  fl = (*env)->GetFieldID (env, cl, \"%s\", \"I\");\n" name;
9931         pr "  (*env)->SetLongField (env, jr, fl, r->%s);\n" name;
9932     | name, FOptPercent ->
9933         pr "  fl = (*env)->GetFieldID (env, cl, \"%s\", \"F\");\n" name;
9934         pr "  (*env)->SetFloatField (env, jr, fl, r->%s);\n" name;
9935     | name, FChar ->
9936         pr "  fl = (*env)->GetFieldID (env, cl, \"%s\", \"C\");\n" name;
9937         pr "  (*env)->SetLongField (env, jr, fl, r->%s);\n" name;
9938   ) cols;
9939   pr "  free (r);\n";
9940   pr "  return jr;\n"
9941
9942 and generate_java_struct_list_return typ jtyp cols =
9943   pr "  cl = (*env)->FindClass (env, \"com/redhat/et/libguestfs/%s\");\n" jtyp;
9944   pr "  jr = (*env)->NewObjectArray (env, r->len, cl, NULL);\n";
9945   pr "  for (i = 0; i < r->len; ++i) {\n";
9946   pr "    jfl = (*env)->AllocObject (env, cl);\n";
9947   List.iter (
9948     function
9949     | name, FString ->
9950         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"Ljava/lang/String;\");\n" name;
9951         pr "    (*env)->SetObjectField (env, jfl, fl, (*env)->NewStringUTF (env, r->val[i].%s));\n" name;
9952     | name, FUUID ->
9953         pr "    {\n";
9954         pr "      char s[33];\n";
9955         pr "      memcpy (s, r->val[i].%s, 32);\n" name;
9956         pr "      s[32] = 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, FBuffer ->
9961         pr "    {\n";
9962         pr "      int len = r->val[i].%s_len;\n" name;
9963         pr "      char s[len+1];\n";
9964         pr "      memcpy (s, r->val[i].%s, len);\n" name;
9965         pr "      s[len] = 0;\n";
9966         pr "      fl = (*env)->GetFieldID (env, cl, \"%s\", \"Ljava/lang/String;\");\n" name;
9967         pr "      (*env)->SetObjectField (env, jfl, fl, (*env)->NewStringUTF (env, s));\n";
9968         pr "    }\n";
9969     | name, (FBytes|FUInt64|FInt64) ->
9970         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"J\");\n" name;
9971         pr "    (*env)->SetLongField (env, jfl, fl, r->val[i].%s);\n" name;
9972     | name, (FUInt32|FInt32) ->
9973         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"I\");\n" name;
9974         pr "    (*env)->SetLongField (env, jfl, fl, r->val[i].%s);\n" name;
9975     | name, FOptPercent ->
9976         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"F\");\n" name;
9977         pr "    (*env)->SetFloatField (env, jfl, fl, r->val[i].%s);\n" name;
9978     | name, FChar ->
9979         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"C\");\n" name;
9980         pr "    (*env)->SetLongField (env, jfl, fl, r->val[i].%s);\n" name;
9981   ) cols;
9982   pr "    (*env)->SetObjectArrayElement (env, jfl, i, jfl);\n";
9983   pr "  }\n";
9984   pr "  guestfs_free_%s_list (r);\n" typ;
9985   pr "  return jr;\n"
9986
9987 and generate_java_makefile_inc () =
9988   generate_header HashStyle GPLv2plus;
9989
9990   pr "java_built_sources = \\\n";
9991   List.iter (
9992     fun (typ, jtyp) ->
9993         pr "\tcom/redhat/et/libguestfs/%s.java \\\n" jtyp;
9994   ) java_structs;
9995   pr "\tcom/redhat/et/libguestfs/GuestFS.java\n"
9996
9997 and generate_haskell_hs () =
9998   generate_header HaskellStyle LGPLv2plus;
9999
10000   (* XXX We only know how to generate partial FFI for Haskell
10001    * at the moment.  Please help out!
10002    *)
10003   let can_generate style =
10004     match style with
10005     | RErr, _
10006     | RInt _, _
10007     | RInt64 _, _ -> true
10008     | RBool _, _
10009     | RConstString _, _
10010     | RConstOptString _, _
10011     | RString _, _
10012     | RStringList _, _
10013     | RStruct _, _
10014     | RStructList _, _
10015     | RHashtable _, _
10016     | RBufferOut _, _ -> false in
10017
10018   pr "\
10019 {-# INCLUDE <guestfs.h> #-}
10020 {-# LANGUAGE ForeignFunctionInterface #-}
10021
10022 module Guestfs (
10023   create";
10024
10025   (* List out the names of the actions we want to export. *)
10026   List.iter (
10027     fun (name, style, _, _, _, _, _) ->
10028       if can_generate style then pr ",\n  %s" name
10029   ) all_functions;
10030
10031   pr "
10032   ) where
10033
10034 -- Unfortunately some symbols duplicate ones already present
10035 -- in Prelude.  We don't know which, so we hard-code a list
10036 -- here.
10037 import Prelude hiding (truncate)
10038
10039 import Foreign
10040 import Foreign.C
10041 import Foreign.C.Types
10042 import IO
10043 import Control.Exception
10044 import Data.Typeable
10045
10046 data GuestfsS = GuestfsS            -- represents the opaque C struct
10047 type GuestfsP = Ptr GuestfsS        -- guestfs_h *
10048 type GuestfsH = ForeignPtr GuestfsS -- guestfs_h * with attached finalizer
10049
10050 -- XXX define properly later XXX
10051 data PV = PV
10052 data VG = VG
10053 data LV = LV
10054 data IntBool = IntBool
10055 data Stat = Stat
10056 data StatVFS = StatVFS
10057 data Hashtable = Hashtable
10058
10059 foreign import ccall unsafe \"guestfs_create\" c_create
10060   :: IO GuestfsP
10061 foreign import ccall unsafe \"&guestfs_close\" c_close
10062   :: FunPtr (GuestfsP -> IO ())
10063 foreign import ccall unsafe \"guestfs_set_error_handler\" c_set_error_handler
10064   :: GuestfsP -> Ptr CInt -> Ptr CInt -> IO ()
10065
10066 create :: IO GuestfsH
10067 create = do
10068   p <- c_create
10069   c_set_error_handler p nullPtr nullPtr
10070   h <- newForeignPtr c_close p
10071   return h
10072
10073 foreign import ccall unsafe \"guestfs_last_error\" c_last_error
10074   :: GuestfsP -> IO CString
10075
10076 -- last_error :: GuestfsH -> IO (Maybe String)
10077 -- last_error h = do
10078 --   str <- withForeignPtr h (\\p -> c_last_error p)
10079 --   maybePeek peekCString str
10080
10081 last_error :: GuestfsH -> IO (String)
10082 last_error h = do
10083   str <- withForeignPtr h (\\p -> c_last_error p)
10084   if (str == nullPtr)
10085     then return \"no error\"
10086     else peekCString str
10087
10088 ";
10089
10090   (* Generate wrappers for each foreign function. *)
10091   List.iter (
10092     fun (name, style, _, _, _, _, _) ->
10093       if can_generate style then (
10094         pr "foreign import ccall unsafe \"guestfs_%s\" c_%s\n" name name;
10095         pr "  :: ";
10096         generate_haskell_prototype ~handle:"GuestfsP" style;
10097         pr "\n";
10098         pr "\n";
10099         pr "%s :: " name;
10100         generate_haskell_prototype ~handle:"GuestfsH" ~hs:true style;
10101         pr "\n";
10102         pr "%s %s = do\n" name
10103           (String.concat " " ("h" :: List.map name_of_argt (snd style)));
10104         pr "  r <- ";
10105         (* Convert pointer arguments using with* functions. *)
10106         List.iter (
10107           function
10108           | FileIn n
10109           | FileOut n
10110           | Pathname n | Device n | Dev_or_Path n | String n -> pr "withCString %s $ \\%s -> " n n
10111           | OptString n -> pr "maybeWith withCString %s $ \\%s -> " n n
10112           | StringList n | DeviceList n -> pr "withMany withCString %s $ \\%s -> withArray0 nullPtr %s $ \\%s -> " n n n n
10113           | Bool _ | Int _ | Int64 _ -> ()
10114         ) (snd style);
10115         (* Convert integer arguments. *)
10116         let args =
10117           List.map (
10118             function
10119             | Bool n -> sprintf "(fromBool %s)" n
10120             | Int n -> sprintf "(fromIntegral %s)" n
10121             | Int64 n -> sprintf "(fromIntegral %s)" n
10122             | FileIn n | FileOut n
10123             | Pathname n | Device n | Dev_or_Path n | String n | OptString n | StringList n | DeviceList n -> n
10124           ) (snd style) in
10125         pr "withForeignPtr h (\\p -> c_%s %s)\n" name
10126           (String.concat " " ("p" :: args));
10127         (match fst style with
10128          | RErr | RInt _ | RInt64 _ | RBool _ ->
10129              pr "  if (r == -1)\n";
10130              pr "    then do\n";
10131              pr "      err <- last_error h\n";
10132              pr "      fail err\n";
10133          | RConstString _ | RConstOptString _ | RString _
10134          | RStringList _ | RStruct _
10135          | RStructList _ | RHashtable _ | RBufferOut _ ->
10136              pr "  if (r == nullPtr)\n";
10137              pr "    then do\n";
10138              pr "      err <- last_error h\n";
10139              pr "      fail err\n";
10140         );
10141         (match fst style with
10142          | RErr ->
10143              pr "    else return ()\n"
10144          | RInt _ ->
10145              pr "    else return (fromIntegral r)\n"
10146          | RInt64 _ ->
10147              pr "    else return (fromIntegral r)\n"
10148          | RBool _ ->
10149              pr "    else return (toBool r)\n"
10150          | RConstString _
10151          | RConstOptString _
10152          | RString _
10153          | RStringList _
10154          | RStruct _
10155          | RStructList _
10156          | RHashtable _
10157          | RBufferOut _ ->
10158              pr "    else return ()\n" (* XXXXXXXXXXXXXXXXXXXX *)
10159         );
10160         pr "\n";
10161       )
10162   ) all_functions
10163
10164 and generate_haskell_prototype ~handle ?(hs = false) style =
10165   pr "%s -> " handle;
10166   let string = if hs then "String" else "CString" in
10167   let int = if hs then "Int" else "CInt" in
10168   let bool = if hs then "Bool" else "CInt" in
10169   let int64 = if hs then "Integer" else "Int64" in
10170   List.iter (
10171     fun arg ->
10172       (match arg with
10173        | Pathname _ | Device _ | Dev_or_Path _ | String _ -> pr "%s" string
10174        | OptString _ -> if hs then pr "Maybe String" else pr "CString"
10175        | StringList _ | DeviceList _ -> if hs then pr "[String]" else pr "Ptr CString"
10176        | Bool _ -> pr "%s" bool
10177        | Int _ -> pr "%s" int
10178        | Int64 _ -> pr "%s" int
10179        | FileIn _ -> pr "%s" string
10180        | FileOut _ -> pr "%s" string
10181       );
10182       pr " -> ";
10183   ) (snd style);
10184   pr "IO (";
10185   (match fst style with
10186    | RErr -> if not hs then pr "CInt"
10187    | RInt _ -> pr "%s" int
10188    | RInt64 _ -> pr "%s" int64
10189    | RBool _ -> pr "%s" bool
10190    | RConstString _ -> pr "%s" string
10191    | RConstOptString _ -> pr "Maybe %s" string
10192    | RString _ -> pr "%s" string
10193    | RStringList _ -> pr "[%s]" string
10194    | RStruct (_, typ) ->
10195        let name = java_name_of_struct typ in
10196        pr "%s" name
10197    | RStructList (_, typ) ->
10198        let name = java_name_of_struct typ in
10199        pr "[%s]" name
10200    | RHashtable _ -> pr "Hashtable"
10201    | RBufferOut _ -> pr "%s" string
10202   );
10203   pr ")"
10204
10205 and generate_csharp () =
10206   generate_header CPlusPlusStyle LGPLv2plus;
10207
10208   (* XXX Make this configurable by the C# assembly users. *)
10209   let library = "libguestfs.so.0" in
10210
10211   pr "\
10212 // These C# bindings are highly experimental at present.
10213 //
10214 // Firstly they only work on Linux (ie. Mono).  In order to get them
10215 // to work on Windows (ie. .Net) you would need to port the library
10216 // itself to Windows first.
10217 //
10218 // The second issue is that some calls are known to be incorrect and
10219 // can cause Mono to segfault.  Particularly: calls which pass or
10220 // return string[], or return any structure value.  This is because
10221 // we haven't worked out the correct way to do this from C#.
10222 //
10223 // The third issue is that when compiling you get a lot of warnings.
10224 // We are not sure whether the warnings are important or not.
10225 //
10226 // Fourthly we do not routinely build or test these bindings as part
10227 // of the make && make check cycle, which means that regressions might
10228 // go unnoticed.
10229 //
10230 // Suggestions and patches are welcome.
10231
10232 // To compile:
10233 //
10234 // gmcs Libguestfs.cs
10235 // mono Libguestfs.exe
10236 //
10237 // (You'll probably want to add a Test class / static main function
10238 // otherwise this won't do anything useful).
10239
10240 using System;
10241 using System.IO;
10242 using System.Runtime.InteropServices;
10243 using System.Runtime.Serialization;
10244 using System.Collections;
10245
10246 namespace Guestfs
10247 {
10248   class Error : System.ApplicationException
10249   {
10250     public Error (string message) : base (message) {}
10251     protected Error (SerializationInfo info, StreamingContext context) {}
10252   }
10253
10254   class Guestfs
10255   {
10256     IntPtr _handle;
10257
10258     [DllImport (\"%s\")]
10259     static extern IntPtr guestfs_create ();
10260
10261     public Guestfs ()
10262     {
10263       _handle = guestfs_create ();
10264       if (_handle == IntPtr.Zero)
10265         throw new Error (\"could not create guestfs handle\");
10266     }
10267
10268     [DllImport (\"%s\")]
10269     static extern void guestfs_close (IntPtr h);
10270
10271     ~Guestfs ()
10272     {
10273       guestfs_close (_handle);
10274     }
10275
10276     [DllImport (\"%s\")]
10277     static extern string guestfs_last_error (IntPtr h);
10278
10279 " library library library;
10280
10281   (* Generate C# structure bindings.  We prefix struct names with
10282    * underscore because C# cannot have conflicting struct names and
10283    * method names (eg. "class stat" and "stat").
10284    *)
10285   List.iter (
10286     fun (typ, cols) ->
10287       pr "    [StructLayout (LayoutKind.Sequential)]\n";
10288       pr "    public class _%s {\n" typ;
10289       List.iter (
10290         function
10291         | name, FChar -> pr "      char %s;\n" name
10292         | name, FString -> pr "      string %s;\n" name
10293         | name, FBuffer ->
10294             pr "      uint %s_len;\n" name;
10295             pr "      string %s;\n" name
10296         | name, FUUID ->
10297             pr "      [MarshalAs (UnmanagedType.ByValTStr, SizeConst=16)]\n";
10298             pr "      string %s;\n" name
10299         | name, FUInt32 -> pr "      uint %s;\n" name
10300         | name, FInt32 -> pr "      int %s;\n" name
10301         | name, (FUInt64|FBytes) -> pr "      ulong %s;\n" name
10302         | name, FInt64 -> pr "      long %s;\n" name
10303         | name, FOptPercent -> pr "      float %s; /* [0..100] or -1 */\n" name
10304       ) cols;
10305       pr "    }\n";
10306       pr "\n"
10307   ) structs;
10308
10309   (* Generate C# function bindings. *)
10310   List.iter (
10311     fun (name, style, _, _, _, shortdesc, _) ->
10312       let rec csharp_return_type () =
10313         match fst style with
10314         | RErr -> "void"
10315         | RBool n -> "bool"
10316         | RInt n -> "int"
10317         | RInt64 n -> "long"
10318         | RConstString n
10319         | RConstOptString n
10320         | RString n
10321         | RBufferOut n -> "string"
10322         | RStruct (_,n) -> "_" ^ n
10323         | RHashtable n -> "Hashtable"
10324         | RStringList n -> "string[]"
10325         | RStructList (_,n) -> sprintf "_%s[]" n
10326
10327       and c_return_type () =
10328         match fst style with
10329         | RErr
10330         | RBool _
10331         | RInt _ -> "int"
10332         | RInt64 _ -> "long"
10333         | RConstString _
10334         | RConstOptString _
10335         | RString _
10336         | RBufferOut _ -> "string"
10337         | RStruct (_,n) -> "_" ^ n
10338         | RHashtable _
10339         | RStringList _ -> "string[]"
10340         | RStructList (_,n) -> sprintf "_%s[]" n
10341
10342       and c_error_comparison () =
10343         match fst style with
10344         | RErr
10345         | RBool _
10346         | RInt _
10347         | RInt64 _ -> "== -1"
10348         | RConstString _
10349         | RConstOptString _
10350         | RString _
10351         | RBufferOut _
10352         | RStruct (_,_)
10353         | RHashtable _
10354         | RStringList _
10355         | RStructList (_,_) -> "== null"
10356
10357       and generate_extern_prototype () =
10358         pr "    static extern %s guestfs_%s (IntPtr h"
10359           (c_return_type ()) name;
10360         List.iter (
10361           function
10362           | Pathname n | Device n | Dev_or_Path n | String n | OptString n
10363           | FileIn n | FileOut n ->
10364               pr ", [In] string %s" n
10365           | StringList n | DeviceList n ->
10366               pr ", [In] string[] %s" n
10367           | Bool n ->
10368               pr ", bool %s" n
10369           | Int n ->
10370               pr ", int %s" n
10371           | Int64 n ->
10372               pr ", long %s" n
10373         ) (snd style);
10374         pr ");\n"
10375
10376       and generate_public_prototype () =
10377         pr "    public %s %s (" (csharp_return_type ()) name;
10378         let comma = ref false in
10379         let next () =
10380           if !comma then pr ", ";
10381           comma := true
10382         in
10383         List.iter (
10384           function
10385           | Pathname n | Device n | Dev_or_Path n | String n | OptString n
10386           | FileIn n | FileOut n ->
10387               next (); pr "string %s" n
10388           | StringList n | DeviceList n ->
10389               next (); pr "string[] %s" n
10390           | Bool n ->
10391               next (); pr "bool %s" n
10392           | Int n ->
10393               next (); pr "int %s" n
10394           | Int64 n ->
10395               next (); pr "long %s" n
10396         ) (snd style);
10397         pr ")\n"
10398
10399       and generate_call () =
10400         pr "guestfs_%s (_handle" name;
10401         List.iter (fun arg -> pr ", %s" (name_of_argt arg)) (snd style);
10402         pr ");\n";
10403       in
10404
10405       pr "    [DllImport (\"%s\")]\n" library;
10406       generate_extern_prototype ();
10407       pr "\n";
10408       pr "    /// <summary>\n";
10409       pr "    /// %s\n" shortdesc;
10410       pr "    /// </summary>\n";
10411       generate_public_prototype ();
10412       pr "    {\n";
10413       pr "      %s r;\n" (c_return_type ());
10414       pr "      r = ";
10415       generate_call ();
10416       pr "      if (r %s)\n" (c_error_comparison ());
10417       pr "        throw new Error (guestfs_last_error (_handle));\n";
10418       (match fst style with
10419        | RErr -> ()
10420        | RBool _ ->
10421            pr "      return r != 0 ? true : false;\n"
10422        | RHashtable _ ->
10423            pr "      Hashtable rr = new Hashtable ();\n";
10424            pr "      for (int i = 0; i < r.Length; i += 2)\n";
10425            pr "        rr.Add (r[i], r[i+1]);\n";
10426            pr "      return rr;\n"
10427        | RInt _ | RInt64 _ | RConstString _ | RConstOptString _
10428        | RString _ | RBufferOut _ | RStruct _ | RStringList _
10429        | RStructList _ ->
10430            pr "      return r;\n"
10431       );
10432       pr "    }\n";
10433       pr "\n";
10434   ) all_functions_sorted;
10435
10436   pr "  }
10437 }
10438 "
10439
10440 and generate_bindtests () =
10441   generate_header CStyle LGPLv2plus;
10442
10443   pr "\
10444 #include <stdio.h>
10445 #include <stdlib.h>
10446 #include <inttypes.h>
10447 #include <string.h>
10448
10449 #include \"guestfs.h\"
10450 #include \"guestfs-internal.h\"
10451 #include \"guestfs-internal-actions.h\"
10452 #include \"guestfs_protocol.h\"
10453
10454 #define error guestfs_error
10455 #define safe_calloc guestfs_safe_calloc
10456 #define safe_malloc guestfs_safe_malloc
10457
10458 static void
10459 print_strings (char *const *argv)
10460 {
10461   int argc;
10462
10463   printf (\"[\");
10464   for (argc = 0; argv[argc] != NULL; ++argc) {
10465     if (argc > 0) printf (\", \");
10466     printf (\"\\\"%%s\\\"\", argv[argc]);
10467   }
10468   printf (\"]\\n\");
10469 }
10470
10471 /* The test0 function prints its parameters to stdout. */
10472 ";
10473
10474   let test0, tests =
10475     match test_functions with
10476     | [] -> assert false
10477     | test0 :: tests -> test0, tests in
10478
10479   let () =
10480     let (name, style, _, _, _, _, _) = test0 in
10481     generate_prototype ~extern:false ~semicolon:false ~newline:true
10482       ~handle:"g" ~prefix:"guestfs__" name style;
10483     pr "{\n";
10484     List.iter (
10485       function
10486       | Pathname n
10487       | Device n | Dev_or_Path n
10488       | String n
10489       | FileIn n
10490       | FileOut n -> pr "  printf (\"%%s\\n\", %s);\n" n
10491       | OptString n -> pr "  printf (\"%%s\\n\", %s ? %s : \"null\");\n" n n
10492       | StringList n | DeviceList n -> pr "  print_strings (%s);\n" n
10493       | Bool n -> pr "  printf (\"%%s\\n\", %s ? \"true\" : \"false\");\n" n
10494       | Int n -> pr "  printf (\"%%d\\n\", %s);\n" n
10495       | Int64 n -> pr "  printf (\"%%\" PRIi64 \"\\n\", %s);\n" n
10496     ) (snd style);
10497     pr "  /* Java changes stdout line buffering so we need this: */\n";
10498     pr "  fflush (stdout);\n";
10499     pr "  return 0;\n";
10500     pr "}\n";
10501     pr "\n" in
10502
10503   List.iter (
10504     fun (name, style, _, _, _, _, _) ->
10505       if String.sub name (String.length name - 3) 3 <> "err" then (
10506         pr "/* Test normal return. */\n";
10507         generate_prototype ~extern:false ~semicolon:false ~newline:true
10508           ~handle:"g" ~prefix:"guestfs__" name style;
10509         pr "{\n";
10510         (match fst style with
10511          | RErr ->
10512              pr "  return 0;\n"
10513          | RInt _ ->
10514              pr "  int r;\n";
10515              pr "  sscanf (val, \"%%d\", &r);\n";
10516              pr "  return r;\n"
10517          | RInt64 _ ->
10518              pr "  int64_t r;\n";
10519              pr "  sscanf (val, \"%%\" SCNi64, &r);\n";
10520              pr "  return r;\n"
10521          | RBool _ ->
10522              pr "  return STREQ (val, \"true\");\n"
10523          | RConstString _
10524          | RConstOptString _ ->
10525              (* Can't return the input string here.  Return a static
10526               * string so we ensure we get a segfault if the caller
10527               * tries to free it.
10528               *)
10529              pr "  return \"static string\";\n"
10530          | RString _ ->
10531              pr "  return strdup (val);\n"
10532          | RStringList _ ->
10533              pr "  char **strs;\n";
10534              pr "  int n, i;\n";
10535              pr "  sscanf (val, \"%%d\", &n);\n";
10536              pr "  strs = safe_malloc (g, (n+1) * sizeof (char *));\n";
10537              pr "  for (i = 0; i < n; ++i) {\n";
10538              pr "    strs[i] = safe_malloc (g, 16);\n";
10539              pr "    snprintf (strs[i], 16, \"%%d\", i);\n";
10540              pr "  }\n";
10541              pr "  strs[n] = NULL;\n";
10542              pr "  return strs;\n"
10543          | RStruct (_, typ) ->
10544              pr "  struct guestfs_%s *r;\n" typ;
10545              pr "  r = safe_calloc (g, sizeof *r, 1);\n";
10546              pr "  return r;\n"
10547          | RStructList (_, typ) ->
10548              pr "  struct guestfs_%s_list *r;\n" typ;
10549              pr "  r = safe_calloc (g, sizeof *r, 1);\n";
10550              pr "  sscanf (val, \"%%d\", &r->len);\n";
10551              pr "  r->val = safe_calloc (g, r->len, sizeof *r->val);\n";
10552              pr "  return r;\n"
10553          | RHashtable _ ->
10554              pr "  char **strs;\n";
10555              pr "  int n, i;\n";
10556              pr "  sscanf (val, \"%%d\", &n);\n";
10557              pr "  strs = safe_malloc (g, (n*2+1) * sizeof (*strs));\n";
10558              pr "  for (i = 0; i < n; ++i) {\n";
10559              pr "    strs[i*2] = safe_malloc (g, 16);\n";
10560              pr "    strs[i*2+1] = safe_malloc (g, 16);\n";
10561              pr "    snprintf (strs[i*2], 16, \"%%d\", i);\n";
10562              pr "    snprintf (strs[i*2+1], 16, \"%%d\", i);\n";
10563              pr "  }\n";
10564              pr "  strs[n*2] = NULL;\n";
10565              pr "  return strs;\n"
10566          | RBufferOut _ ->
10567              pr "  return strdup (val);\n"
10568         );
10569         pr "}\n";
10570         pr "\n"
10571       ) else (
10572         pr "/* Test error return. */\n";
10573         generate_prototype ~extern:false ~semicolon:false ~newline:true
10574           ~handle:"g" ~prefix:"guestfs__" name style;
10575         pr "{\n";
10576         pr "  error (g, \"error\");\n";
10577         (match fst style with
10578          | RErr | RInt _ | RInt64 _ | RBool _ ->
10579              pr "  return -1;\n"
10580          | RConstString _ | RConstOptString _
10581          | RString _ | RStringList _ | RStruct _
10582          | RStructList _
10583          | RHashtable _
10584          | RBufferOut _ ->
10585              pr "  return NULL;\n"
10586         );
10587         pr "}\n";
10588         pr "\n"
10589       )
10590   ) tests
10591
10592 and generate_ocaml_bindtests () =
10593   generate_header OCamlStyle GPLv2plus;
10594
10595   pr "\
10596 let () =
10597   let g = Guestfs.create () in
10598 ";
10599
10600   let mkargs args =
10601     String.concat " " (
10602       List.map (
10603         function
10604         | CallString s -> "\"" ^ s ^ "\""
10605         | CallOptString None -> "None"
10606         | CallOptString (Some s) -> sprintf "(Some \"%s\")" s
10607         | CallStringList xs ->
10608             "[|" ^ String.concat ";" (List.map (sprintf "\"%s\"") xs) ^ "|]"
10609         | CallInt i when i >= 0 -> string_of_int i
10610         | CallInt i (* when i < 0 *) -> "(" ^ string_of_int i ^ ")"
10611         | CallInt64 i when i >= 0L -> Int64.to_string i ^ "L"
10612         | CallInt64 i (* when i < 0L *) -> "(" ^ Int64.to_string i ^ "L)"
10613         | CallBool b -> string_of_bool b
10614       ) args
10615     )
10616   in
10617
10618   generate_lang_bindtests (
10619     fun f args -> pr "  Guestfs.%s g %s;\n" f (mkargs args)
10620   );
10621
10622   pr "print_endline \"EOF\"\n"
10623
10624 and generate_perl_bindtests () =
10625   pr "#!/usr/bin/perl -w\n";
10626   generate_header HashStyle GPLv2plus;
10627
10628   pr "\
10629 use strict;
10630
10631 use Sys::Guestfs;
10632
10633 my $g = Sys::Guestfs->new ();
10634 ";
10635
10636   let mkargs args =
10637     String.concat ", " (
10638       List.map (
10639         function
10640         | CallString s -> "\"" ^ s ^ "\""
10641         | CallOptString None -> "undef"
10642         | CallOptString (Some s) -> sprintf "\"%s\"" s
10643         | CallStringList xs ->
10644             "[" ^ String.concat "," (List.map (sprintf "\"%s\"") xs) ^ "]"
10645         | CallInt i -> string_of_int i
10646         | CallInt64 i -> Int64.to_string i
10647         | CallBool b -> if b then "1" else "0"
10648       ) args
10649     )
10650   in
10651
10652   generate_lang_bindtests (
10653     fun f args -> pr "$g->%s (%s);\n" f (mkargs args)
10654   );
10655
10656   pr "print \"EOF\\n\"\n"
10657
10658 and generate_python_bindtests () =
10659   generate_header HashStyle GPLv2plus;
10660
10661   pr "\
10662 import guestfs
10663
10664 g = guestfs.GuestFS ()
10665 ";
10666
10667   let mkargs args =
10668     String.concat ", " (
10669       List.map (
10670         function
10671         | CallString s -> "\"" ^ s ^ "\""
10672         | CallOptString None -> "None"
10673         | CallOptString (Some s) -> sprintf "\"%s\"" s
10674         | CallStringList xs ->
10675             "[" ^ String.concat "," (List.map (sprintf "\"%s\"") xs) ^ "]"
10676         | CallInt i -> string_of_int i
10677         | CallInt64 i -> Int64.to_string i
10678         | CallBool b -> if b then "1" else "0"
10679       ) args
10680     )
10681   in
10682
10683   generate_lang_bindtests (
10684     fun f args -> pr "g.%s (%s)\n" f (mkargs args)
10685   );
10686
10687   pr "print \"EOF\"\n"
10688
10689 and generate_ruby_bindtests () =
10690   generate_header HashStyle GPLv2plus;
10691
10692   pr "\
10693 require 'guestfs'
10694
10695 g = Guestfs::create()
10696 ";
10697
10698   let mkargs args =
10699     String.concat ", " (
10700       List.map (
10701         function
10702         | CallString s -> "\"" ^ s ^ "\""
10703         | CallOptString None -> "nil"
10704         | CallOptString (Some s) -> sprintf "\"%s\"" s
10705         | CallStringList xs ->
10706             "[" ^ String.concat "," (List.map (sprintf "\"%s\"") xs) ^ "]"
10707         | CallInt i -> string_of_int i
10708         | CallInt64 i -> Int64.to_string i
10709         | CallBool b -> string_of_bool b
10710       ) args
10711     )
10712   in
10713
10714   generate_lang_bindtests (
10715     fun f args -> pr "g.%s(%s)\n" f (mkargs args)
10716   );
10717
10718   pr "print \"EOF\\n\"\n"
10719
10720 and generate_java_bindtests () =
10721   generate_header CStyle GPLv2plus;
10722
10723   pr "\
10724 import com.redhat.et.libguestfs.*;
10725
10726 public class Bindtests {
10727     public static void main (String[] argv)
10728     {
10729         try {
10730             GuestFS g = new GuestFS ();
10731 ";
10732
10733   let mkargs args =
10734     String.concat ", " (
10735       List.map (
10736         function
10737         | CallString s -> "\"" ^ s ^ "\""
10738         | CallOptString None -> "null"
10739         | CallOptString (Some s) -> sprintf "\"%s\"" s
10740         | CallStringList xs ->
10741             "new String[]{" ^
10742               String.concat "," (List.map (sprintf "\"%s\"") xs) ^ "}"
10743         | CallInt i -> string_of_int i
10744         | CallInt64 i -> Int64.to_string i
10745         | CallBool b -> string_of_bool b
10746       ) args
10747     )
10748   in
10749
10750   generate_lang_bindtests (
10751     fun f args -> pr "            g.%s (%s);\n" f (mkargs args)
10752   );
10753
10754   pr "
10755             System.out.println (\"EOF\");
10756         }
10757         catch (Exception exn) {
10758             System.err.println (exn);
10759             System.exit (1);
10760         }
10761     }
10762 }
10763 "
10764
10765 and generate_haskell_bindtests () =
10766   generate_header HaskellStyle GPLv2plus;
10767
10768   pr "\
10769 module Bindtests where
10770 import qualified Guestfs
10771
10772 main = do
10773   g <- Guestfs.create
10774 ";
10775
10776   let mkargs args =
10777     String.concat " " (
10778       List.map (
10779         function
10780         | CallString s -> "\"" ^ s ^ "\""
10781         | CallOptString None -> "Nothing"
10782         | CallOptString (Some s) -> sprintf "(Just \"%s\")" s
10783         | CallStringList xs ->
10784             "[" ^ String.concat "," (List.map (sprintf "\"%s\"") xs) ^ "]"
10785         | CallInt i when i < 0 -> "(" ^ string_of_int i ^ ")"
10786         | CallInt i -> string_of_int i
10787         | CallInt64 i when i < 0L -> "(" ^ Int64.to_string i ^ ")"
10788         | CallInt64 i -> Int64.to_string i
10789         | CallBool true -> "True"
10790         | CallBool false -> "False"
10791       ) args
10792     )
10793   in
10794
10795   generate_lang_bindtests (
10796     fun f args -> pr "  Guestfs.%s g %s\n" f (mkargs args)
10797   );
10798
10799   pr "  putStrLn \"EOF\"\n"
10800
10801 (* Language-independent bindings tests - we do it this way to
10802  * ensure there is parity in testing bindings across all languages.
10803  *)
10804 and generate_lang_bindtests call =
10805   call "test0" [CallString "abc"; CallOptString (Some "def");
10806                 CallStringList []; CallBool false;
10807                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10808   call "test0" [CallString "abc"; CallOptString None;
10809                 CallStringList []; CallBool false;
10810                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10811   call "test0" [CallString ""; CallOptString (Some "def");
10812                 CallStringList []; CallBool false;
10813                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10814   call "test0" [CallString ""; CallOptString (Some "");
10815                 CallStringList []; CallBool false;
10816                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10817   call "test0" [CallString "abc"; CallOptString (Some "def");
10818                 CallStringList ["1"]; CallBool false;
10819                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10820   call "test0" [CallString "abc"; CallOptString (Some "def");
10821                 CallStringList ["1"; "2"]; CallBool false;
10822                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10823   call "test0" [CallString "abc"; CallOptString (Some "def");
10824                 CallStringList ["1"]; CallBool true;
10825                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10826   call "test0" [CallString "abc"; CallOptString (Some "def");
10827                 CallStringList ["1"]; CallBool false;
10828                 CallInt (-1); CallInt64 (-1L); CallString "123"; CallString "456"];
10829   call "test0" [CallString "abc"; CallOptString (Some "def");
10830                 CallStringList ["1"]; CallBool false;
10831                 CallInt (-2); CallInt64 (-2L); CallString "123"; CallString "456"];
10832   call "test0" [CallString "abc"; CallOptString (Some "def");
10833                 CallStringList ["1"]; CallBool false;
10834                 CallInt 1; CallInt64 1L; CallString "123"; CallString "456"];
10835   call "test0" [CallString "abc"; CallOptString (Some "def");
10836                 CallStringList ["1"]; CallBool false;
10837                 CallInt 2; CallInt64 2L; CallString "123"; CallString "456"];
10838   call "test0" [CallString "abc"; CallOptString (Some "def");
10839                 CallStringList ["1"]; CallBool false;
10840                 CallInt 4095; CallInt64 4095L; CallString "123"; CallString "456"];
10841   call "test0" [CallString "abc"; CallOptString (Some "def");
10842                 CallStringList ["1"]; CallBool false;
10843                 CallInt 0; CallInt64 0L; CallString ""; CallString ""]
10844
10845 (* XXX Add here tests of the return and error functions. *)
10846
10847 (* Code to generator bindings for virt-inspector.  Currently only
10848  * implemented for OCaml code (for virt-p2v 2.0).
10849  *)
10850 let rng_input = "inspector/virt-inspector.rng"
10851
10852 (* Read the input file and parse it into internal structures.  This is
10853  * by no means a complete RELAX NG parser, but is just enough to be
10854  * able to parse the specific input file.
10855  *)
10856 type rng =
10857   | Element of string * rng list        (* <element name=name/> *)
10858   | Attribute of string * rng list        (* <attribute name=name/> *)
10859   | Interleave of rng list                (* <interleave/> *)
10860   | ZeroOrMore of rng                        (* <zeroOrMore/> *)
10861   | OneOrMore of rng                        (* <oneOrMore/> *)
10862   | Optional of rng                        (* <optional/> *)
10863   | Choice of string list                (* <choice><value/>*</choice> *)
10864   | Value of string                        (* <value>str</value> *)
10865   | Text                                (* <text/> *)
10866
10867 let rec string_of_rng = function
10868   | Element (name, xs) ->
10869       "Element (\"" ^ name ^ "\", (" ^ string_of_rng_list xs ^ "))"
10870   | Attribute (name, xs) ->
10871       "Attribute (\"" ^ name ^ "\", (" ^ string_of_rng_list xs ^ "))"
10872   | Interleave xs -> "Interleave (" ^ string_of_rng_list xs ^ ")"
10873   | ZeroOrMore rng -> "ZeroOrMore (" ^ string_of_rng rng ^ ")"
10874   | OneOrMore rng -> "OneOrMore (" ^ string_of_rng rng ^ ")"
10875   | Optional rng -> "Optional (" ^ string_of_rng rng ^ ")"
10876   | Choice values -> "Choice [" ^ String.concat ", " values ^ "]"
10877   | Value value -> "Value \"" ^ value ^ "\""
10878   | Text -> "Text"
10879
10880 and string_of_rng_list xs =
10881   String.concat ", " (List.map string_of_rng xs)
10882
10883 let rec parse_rng ?defines context = function
10884   | [] -> []
10885   | Xml.Element ("element", ["name", name], children) :: rest ->
10886       Element (name, parse_rng ?defines context children)
10887       :: parse_rng ?defines context rest
10888   | Xml.Element ("attribute", ["name", name], children) :: rest ->
10889       Attribute (name, parse_rng ?defines context children)
10890       :: parse_rng ?defines context rest
10891   | Xml.Element ("interleave", [], children) :: rest ->
10892       Interleave (parse_rng ?defines context children)
10893       :: parse_rng ?defines context rest
10894   | Xml.Element ("zeroOrMore", [], [child]) :: rest ->
10895       let rng = parse_rng ?defines context [child] in
10896       (match rng with
10897        | [child] -> ZeroOrMore child :: parse_rng ?defines context rest
10898        | _ ->
10899            failwithf "%s: <zeroOrMore> contains more than one child element"
10900              context
10901       )
10902   | Xml.Element ("oneOrMore", [], [child]) :: rest ->
10903       let rng = parse_rng ?defines context [child] in
10904       (match rng with
10905        | [child] -> OneOrMore child :: parse_rng ?defines context rest
10906        | _ ->
10907            failwithf "%s: <oneOrMore> contains more than one child element"
10908              context
10909       )
10910   | Xml.Element ("optional", [], [child]) :: rest ->
10911       let rng = parse_rng ?defines context [child] in
10912       (match rng with
10913        | [child] -> Optional child :: parse_rng ?defines context rest
10914        | _ ->
10915            failwithf "%s: <optional> contains more than one child element"
10916              context
10917       )
10918   | Xml.Element ("choice", [], children) :: rest ->
10919       let values = List.map (
10920         function Xml.Element ("value", [], [Xml.PCData value]) -> value
10921         | _ ->
10922             failwithf "%s: can't handle anything except <value> in <choice>"
10923               context
10924       ) children in
10925       Choice values
10926       :: parse_rng ?defines context rest
10927   | Xml.Element ("value", [], [Xml.PCData value]) :: rest ->
10928       Value value :: parse_rng ?defines context rest
10929   | Xml.Element ("text", [], []) :: rest ->
10930       Text :: parse_rng ?defines context rest
10931   | Xml.Element ("ref", ["name", name], []) :: rest ->
10932       (* Look up the reference.  Because of limitations in this parser,
10933        * we can't handle arbitrarily nested <ref> yet.  You can only
10934        * use <ref> from inside <start>.
10935        *)
10936       (match defines with
10937        | None ->
10938            failwithf "%s: contains <ref>, but no refs are defined yet" context
10939        | Some map ->
10940            let rng = StringMap.find name map in
10941            rng @ parse_rng ?defines context rest
10942       )
10943   | x :: _ ->
10944       failwithf "%s: can't handle '%s' in schema" context (Xml.to_string x)
10945
10946 let grammar =
10947   let xml = Xml.parse_file rng_input in
10948   match xml with
10949   | Xml.Element ("grammar", _,
10950                  Xml.Element ("start", _, gram) :: defines) ->
10951       (* The <define/> elements are referenced in the <start> section,
10952        * so build a map of those first.
10953        *)
10954       let defines = List.fold_left (
10955         fun map ->
10956           function Xml.Element ("define", ["name", name], defn) ->
10957             StringMap.add name defn map
10958           | _ ->
10959               failwithf "%s: expected <define name=name/>" rng_input
10960       ) StringMap.empty defines in
10961       let defines = StringMap.mapi parse_rng defines in
10962
10963       (* Parse the <start> clause, passing the defines. *)
10964       parse_rng ~defines "<start>" gram
10965   | _ ->
10966       failwithf "%s: input is not <grammar><start/><define>*</grammar>"
10967         rng_input
10968
10969 let name_of_field = function
10970   | Element (name, _) | Attribute (name, _)
10971   | ZeroOrMore (Element (name, _))
10972   | OneOrMore (Element (name, _))
10973   | Optional (Element (name, _)) -> name
10974   | Optional (Attribute (name, _)) -> name
10975   | Text -> (* an unnamed field in an element *)
10976       "data"
10977   | rng ->
10978       failwithf "name_of_field failed at: %s" (string_of_rng rng)
10979
10980 (* At the moment this function only generates OCaml types.  However we
10981  * should parameterize it later so it can generate types/structs in a
10982  * variety of languages.
10983  *)
10984 let generate_types xs =
10985   (* A simple type is one that can be printed out directly, eg.
10986    * "string option".  A complex type is one which has a name and has
10987    * to be defined via another toplevel definition, eg. a struct.
10988    *
10989    * generate_type generates code for either simple or complex types.
10990    * In the simple case, it returns the string ("string option").  In
10991    * the complex case, it returns the name ("mountpoint").  In the
10992    * complex case it has to print out the definition before returning,
10993    * so it should only be called when we are at the beginning of a
10994    * new line (BOL context).
10995    *)
10996   let rec generate_type = function
10997     | Text ->                                (* string *)
10998         "string", true
10999     | Choice values ->                        (* [`val1|`val2|...] *)
11000         "[" ^ String.concat "|" (List.map ((^)"`") values) ^ "]", true
11001     | ZeroOrMore rng ->                        (* <rng> list *)
11002         let t, is_simple = generate_type rng in
11003         t ^ " list (* 0 or more *)", is_simple
11004     | OneOrMore rng ->                        (* <rng> list *)
11005         let t, is_simple = generate_type rng in
11006         t ^ " list (* 1 or more *)", is_simple
11007                                         (* virt-inspector hack: bool *)
11008     | Optional (Attribute (name, [Value "1"])) ->
11009         "bool", true
11010     | Optional rng ->                        (* <rng> list *)
11011         let t, is_simple = generate_type rng in
11012         t ^ " option", is_simple
11013                                         (* type name = { fields ... } *)
11014     | Element (name, fields) when is_attrs_interleave fields ->
11015         generate_type_struct name (get_attrs_interleave fields)
11016     | Element (name, [field])                (* type name = field *)
11017     | Attribute (name, [field]) ->
11018         let t, is_simple = generate_type field in
11019         if is_simple then (t, true)
11020         else (
11021           pr "type %s = %s\n" name t;
11022           name, false
11023         )
11024     | Element (name, fields) ->              (* type name = { fields ... } *)
11025         generate_type_struct name fields
11026     | rng ->
11027         failwithf "generate_type failed at: %s" (string_of_rng rng)
11028
11029   and is_attrs_interleave = function
11030     | [Interleave _] -> true
11031     | Attribute _ :: fields -> is_attrs_interleave fields
11032     | Optional (Attribute _) :: fields -> is_attrs_interleave fields
11033     | _ -> false
11034
11035   and get_attrs_interleave = function
11036     | [Interleave fields] -> fields
11037     | ((Attribute _) as field) :: fields
11038     | ((Optional (Attribute _)) as field) :: fields ->
11039         field :: get_attrs_interleave fields
11040     | _ -> assert false
11041
11042   and generate_types xs =
11043     List.iter (fun x -> ignore (generate_type x)) xs
11044
11045   and generate_type_struct name fields =
11046     (* Calculate the types of the fields first.  We have to do this
11047      * before printing anything so we are still in BOL context.
11048      *)
11049     let types = List.map fst (List.map generate_type fields) in
11050
11051     (* Special case of a struct containing just a string and another
11052      * field.  Turn it into an assoc list.
11053      *)
11054     match types with
11055     | ["string"; other] ->
11056         let fname1, fname2 =
11057           match fields with
11058           | [f1; f2] -> name_of_field f1, name_of_field f2
11059           | _ -> assert false in
11060         pr "type %s = string * %s (* %s -> %s *)\n" name other fname1 fname2;
11061         name, false
11062
11063     | types ->
11064         pr "type %s = {\n" name;
11065         List.iter (
11066           fun (field, ftype) ->
11067             let fname = name_of_field field in
11068             pr "  %s_%s : %s;\n" name fname ftype
11069         ) (List.combine fields types);
11070         pr "}\n";
11071         (* Return the name of this type, and
11072          * false because it's not a simple type.
11073          *)
11074         name, false
11075   in
11076
11077   generate_types xs
11078
11079 let generate_parsers xs =
11080   (* As for generate_type above, generate_parser makes a parser for
11081    * some type, and returns the name of the parser it has generated.
11082    * Because it (may) need to print something, it should always be
11083    * called in BOL context.
11084    *)
11085   let rec generate_parser = function
11086     | Text ->                                (* string *)
11087         "string_child_or_empty"
11088     | Choice values ->                        (* [`val1|`val2|...] *)
11089         sprintf "(fun x -> match Xml.pcdata (first_child x) with %s | str -> failwith (\"unexpected field value: \" ^ str))"
11090           (String.concat "|"
11091              (List.map (fun v -> sprintf "%S -> `%s" v v) values))
11092     | ZeroOrMore rng ->                        (* <rng> list *)
11093         let pa = generate_parser rng in
11094         sprintf "(fun x -> List.map %s (Xml.children x))" pa
11095     | OneOrMore rng ->                        (* <rng> list *)
11096         let pa = generate_parser rng in
11097         sprintf "(fun x -> List.map %s (Xml.children x))" pa
11098                                         (* virt-inspector hack: bool *)
11099     | Optional (Attribute (name, [Value "1"])) ->
11100         sprintf "(fun x -> try ignore (Xml.attrib x %S); true with Xml.No_attribute _ -> false)" name
11101     | Optional rng ->                        (* <rng> list *)
11102         let pa = generate_parser rng in
11103         sprintf "(function None -> None | Some x -> Some (%s x))" pa
11104                                         (* type name = { fields ... } *)
11105     | Element (name, fields) when is_attrs_interleave fields ->
11106         generate_parser_struct name (get_attrs_interleave fields)
11107     | Element (name, [field]) ->        (* type name = field *)
11108         let pa = generate_parser field in
11109         let parser_name = sprintf "parse_%s_%d" name (unique ()) in
11110         pr "let %s =\n" parser_name;
11111         pr "  %s\n" pa;
11112         pr "let parse_%s = %s\n" name parser_name;
11113         parser_name
11114     | Attribute (name, [field]) ->
11115         let pa = generate_parser field in
11116         let parser_name = sprintf "parse_%s_%d" name (unique ()) in
11117         pr "let %s =\n" parser_name;
11118         pr "  %s\n" pa;
11119         pr "let parse_%s = %s\n" name parser_name;
11120         parser_name
11121     | Element (name, fields) ->              (* type name = { fields ... } *)
11122         generate_parser_struct name ([], fields)
11123     | rng ->
11124         failwithf "generate_parser failed at: %s" (string_of_rng rng)
11125
11126   and is_attrs_interleave = function
11127     | [Interleave _] -> true
11128     | Attribute _ :: fields -> is_attrs_interleave fields
11129     | Optional (Attribute _) :: fields -> is_attrs_interleave fields
11130     | _ -> false
11131
11132   and get_attrs_interleave = function
11133     | [Interleave fields] -> [], fields
11134     | ((Attribute _) as field) :: fields
11135     | ((Optional (Attribute _)) as field) :: fields ->
11136         let attrs, interleaves = get_attrs_interleave fields in
11137         (field :: attrs), interleaves
11138     | _ -> assert false
11139
11140   and generate_parsers xs =
11141     List.iter (fun x -> ignore (generate_parser x)) xs
11142
11143   and generate_parser_struct name (attrs, interleaves) =
11144     (* Generate parsers for the fields first.  We have to do this
11145      * before printing anything so we are still in BOL context.
11146      *)
11147     let fields = attrs @ interleaves in
11148     let pas = List.map generate_parser fields in
11149
11150     (* Generate an intermediate tuple from all the fields first.
11151      * If the type is just a string + another field, then we will
11152      * return this directly, otherwise it is turned into a record.
11153      *
11154      * RELAX NG note: This code treats <interleave> and plain lists of
11155      * fields the same.  In other words, it doesn't bother enforcing
11156      * any ordering of fields in the XML.
11157      *)
11158     pr "let parse_%s x =\n" name;
11159     pr "  let t = (\n    ";
11160     let comma = ref false in
11161     List.iter (
11162       fun x ->
11163         if !comma then pr ",\n    ";
11164         comma := true;
11165         match x with
11166         | Optional (Attribute (fname, [field])), pa ->
11167             pr "%s x" pa
11168         | Optional (Element (fname, [field])), pa ->
11169             pr "%s (optional_child %S x)" pa fname
11170         | Attribute (fname, [Text]), _ ->
11171             pr "attribute %S x" fname
11172         | (ZeroOrMore _ | OneOrMore _), pa ->
11173             pr "%s x" pa
11174         | Text, pa ->
11175             pr "%s x" pa
11176         | (field, pa) ->
11177             let fname = name_of_field field in
11178             pr "%s (child %S x)" pa fname
11179     ) (List.combine fields pas);
11180     pr "\n  ) in\n";
11181
11182     (match fields with
11183      | [Element (_, [Text]) | Attribute (_, [Text]); _] ->
11184          pr "  t\n"
11185
11186      | _ ->
11187          pr "  (Obj.magic t : %s)\n" name
11188 (*
11189          List.iter (
11190            function
11191            | (Optional (Attribute (fname, [field])), pa) ->
11192                pr "  %s_%s =\n" name fname;
11193                pr "    %s x;\n" pa
11194            | (Optional (Element (fname, [field])), pa) ->
11195                pr "  %s_%s =\n" name fname;
11196                pr "    (let x = optional_child %S x in\n" fname;
11197                pr "     %s x);\n" pa
11198            | (field, pa) ->
11199                let fname = name_of_field field in
11200                pr "  %s_%s =\n" name fname;
11201                pr "    (let x = child %S x in\n" fname;
11202                pr "     %s x);\n" pa
11203          ) (List.combine fields pas);
11204          pr "}\n"
11205 *)
11206     );
11207     sprintf "parse_%s" name
11208   in
11209
11210   generate_parsers xs
11211
11212 (* Generate ocaml/guestfs_inspector.mli. *)
11213 let generate_ocaml_inspector_mli () =
11214   generate_header ~extra_inputs:[rng_input] OCamlStyle LGPLv2plus;
11215
11216   pr "\
11217 (** This is an OCaml language binding to the external [virt-inspector]
11218     program.
11219
11220     For more information, please read the man page [virt-inspector(1)].
11221 *)
11222
11223 ";
11224
11225   generate_types grammar;
11226   pr "(** The nested information returned from the {!inspect} function. *)\n";
11227   pr "\n";
11228
11229   pr "\
11230 val inspect : ?connect:string -> ?xml:string -> string list -> operatingsystems
11231 (** To inspect a libvirt domain called [name], pass a singleton
11232     list: [inspect [name]].  When using libvirt only, you may
11233     optionally pass a libvirt URI using [inspect ~connect:uri ...].
11234
11235     To inspect a disk image or images, pass a list of the filenames
11236     of the disk images: [inspect filenames]
11237
11238     This function inspects the given guest or disk images and
11239     returns a list of operating system(s) found and a large amount
11240     of information about them.  In the vast majority of cases,
11241     a virtual machine only contains a single operating system.
11242
11243     If the optional [~xml] parameter is given, then this function
11244     skips running the external virt-inspector program and just
11245     parses the given XML directly (which is expected to be XML
11246     produced from a previous run of virt-inspector).  The list of
11247     names and connect URI are ignored in this case.
11248
11249     This function can throw a wide variety of exceptions, for example
11250     if the external virt-inspector program cannot be found, or if
11251     it doesn't generate valid XML.
11252 *)
11253 "
11254
11255 (* Generate ocaml/guestfs_inspector.ml. *)
11256 let generate_ocaml_inspector_ml () =
11257   generate_header ~extra_inputs:[rng_input] OCamlStyle LGPLv2plus;
11258
11259   pr "open Unix\n";
11260   pr "\n";
11261
11262   generate_types grammar;
11263   pr "\n";
11264
11265   pr "\
11266 (* Misc functions which are used by the parser code below. *)
11267 let first_child = function
11268   | Xml.Element (_, _, c::_) -> c
11269   | Xml.Element (name, _, []) ->
11270       failwith (\"expected <\" ^ name ^ \"/> to have a child node\")
11271   | Xml.PCData str ->
11272       failwith (\"expected XML tag, but read PCDATA '\" ^ str ^ \"' instead\")
11273
11274 let string_child_or_empty = function
11275   | Xml.Element (_, _, [Xml.PCData s]) -> s
11276   | Xml.Element (_, _, []) -> \"\"
11277   | Xml.Element (x, _, _) ->
11278       failwith (\"expected XML tag with a single PCDATA child, but got \" ^
11279                 x ^ \" instead\")
11280   | Xml.PCData str ->
11281       failwith (\"expected XML tag, but read PCDATA '\" ^ str ^ \"' instead\")
11282
11283 let optional_child name xml =
11284   let children = Xml.children xml in
11285   try
11286     Some (List.find (function
11287                      | Xml.Element (n, _, _) when n = name -> true
11288                      | _ -> false) children)
11289   with
11290     Not_found -> None
11291
11292 let child name xml =
11293   match optional_child name xml with
11294   | Some c -> c
11295   | None ->
11296       failwith (\"mandatory field <\" ^ name ^ \"/> missing in XML output\")
11297
11298 let attribute name xml =
11299   try Xml.attrib xml name
11300   with Xml.No_attribute _ ->
11301     failwith (\"mandatory attribute \" ^ name ^ \" missing in XML output\")
11302
11303 ";
11304
11305   generate_parsers grammar;
11306   pr "\n";
11307
11308   pr "\
11309 (* Run external virt-inspector, then use parser to parse the XML. *)
11310 let inspect ?connect ?xml names =
11311   let xml =
11312     match xml with
11313     | None ->
11314         if names = [] then invalid_arg \"inspect: no names given\";
11315         let cmd = [ \"virt-inspector\"; \"--xml\" ] @
11316           (match connect with None -> [] | Some uri -> [ \"--connect\"; uri ]) @
11317           names in
11318         let cmd = List.map Filename.quote cmd in
11319         let cmd = String.concat \" \" cmd in
11320         let chan = open_process_in cmd in
11321         let xml = Xml.parse_in chan in
11322         (match close_process_in chan with
11323          | WEXITED 0 -> ()
11324          | WEXITED _ -> failwith \"external virt-inspector command failed\"
11325          | WSIGNALED i | WSTOPPED i ->
11326              failwith (\"external virt-inspector command died or stopped on sig \" ^
11327                        string_of_int i)
11328         );
11329         xml
11330     | Some doc ->
11331         Xml.parse_string doc in
11332   parse_operatingsystems xml
11333 "
11334
11335 (* This is used to generate the src/MAX_PROC_NR file which
11336  * contains the maximum procedure number, a surrogate for the
11337  * ABI version number.  See src/Makefile.am for the details.
11338  *)
11339 and generate_max_proc_nr () =
11340   let proc_nrs = List.map (
11341     fun (_, _, proc_nr, _, _, _, _) -> proc_nr
11342   ) daemon_functions in
11343
11344   let max_proc_nr = List.fold_left max 0 proc_nrs in
11345
11346   pr "%d\n" max_proc_nr
11347
11348 let output_to filename k =
11349   let filename_new = filename ^ ".new" in
11350   chan := open_out filename_new;
11351   k ();
11352   close_out !chan;
11353   chan := Pervasives.stdout;
11354
11355   (* Is the new file different from the current file? *)
11356   if Sys.file_exists filename && files_equal filename filename_new then
11357     unlink filename_new                 (* same, so skip it *)
11358   else (
11359     (* different, overwrite old one *)
11360     (try chmod filename 0o644 with Unix_error _ -> ());
11361     rename filename_new filename;
11362     chmod filename 0o444;
11363     printf "written %s\n%!" filename;
11364   )
11365
11366 let perror msg = function
11367   | Unix_error (err, _, _) ->
11368       eprintf "%s: %s\n" msg (error_message err)
11369   | exn ->
11370       eprintf "%s: %s\n" msg (Printexc.to_string exn)
11371
11372 (* Main program. *)
11373 let () =
11374   let lock_fd =
11375     try openfile "HACKING" [O_RDWR] 0
11376     with
11377     | Unix_error (ENOENT, _, _) ->
11378         eprintf "\
11379 You are probably running this from the wrong directory.
11380 Run it from the top source directory using the command
11381   src/generator.ml
11382 ";
11383         exit 1
11384     | exn ->
11385         perror "open: HACKING" exn;
11386         exit 1 in
11387
11388   (* Acquire a lock so parallel builds won't try to run the generator
11389    * twice at the same time.  Subsequent builds will wait for the first
11390    * one to finish.  Note the lock is released implicitly when the
11391    * program exits.
11392    *)
11393   (try lockf lock_fd F_LOCK 1
11394    with exn ->
11395      perror "lock: HACKING" exn;
11396      exit 1);
11397
11398   check_functions ();
11399
11400   output_to "src/guestfs_protocol.x" generate_xdr;
11401   output_to "src/guestfs-structs.h" generate_structs_h;
11402   output_to "src/guestfs-actions.h" generate_actions_h;
11403   output_to "src/guestfs-internal-actions.h" generate_internal_actions_h;
11404   output_to "src/guestfs-actions.c" generate_client_actions;
11405   output_to "src/guestfs-bindtests.c" generate_bindtests;
11406   output_to "src/guestfs-structs.pod" generate_structs_pod;
11407   output_to "src/guestfs-actions.pod" generate_actions_pod;
11408   output_to "src/guestfs-availability.pod" generate_availability_pod;
11409   output_to "src/MAX_PROC_NR" generate_max_proc_nr;
11410   output_to "src/libguestfs.syms" generate_linker_script;
11411   output_to "daemon/actions.h" generate_daemon_actions_h;
11412   output_to "daemon/stubs.c" generate_daemon_actions;
11413   output_to "daemon/names.c" generate_daemon_names;
11414   output_to "daemon/optgroups.c" generate_daemon_optgroups_c;
11415   output_to "daemon/optgroups.h" generate_daemon_optgroups_h;
11416   output_to "capitests/tests.c" generate_tests;
11417   output_to "fish/cmds.c" generate_fish_cmds;
11418   output_to "fish/completion.c" generate_fish_completion;
11419   output_to "fish/guestfish-actions.pod" generate_fish_actions_pod;
11420   output_to "ocaml/guestfs.mli" generate_ocaml_mli;
11421   output_to "ocaml/guestfs.ml" generate_ocaml_ml;
11422   output_to "ocaml/guestfs_c_actions.c" generate_ocaml_c;
11423   output_to "ocaml/bindtests.ml" generate_ocaml_bindtests;
11424   output_to "ocaml/guestfs_inspector.mli" generate_ocaml_inspector_mli;
11425   output_to "ocaml/guestfs_inspector.ml" generate_ocaml_inspector_ml;
11426   output_to "perl/Guestfs.xs" generate_perl_xs;
11427   output_to "perl/lib/Sys/Guestfs.pm" generate_perl_pm;
11428   output_to "perl/bindtests.pl" generate_perl_bindtests;
11429   output_to "python/guestfs-py.c" generate_python_c;
11430   output_to "python/guestfs.py" generate_python_py;
11431   output_to "python/bindtests.py" generate_python_bindtests;
11432   output_to "ruby/ext/guestfs/_guestfs.c" generate_ruby_c;
11433   output_to "ruby/bindtests.rb" generate_ruby_bindtests;
11434   output_to "java/com/redhat/et/libguestfs/GuestFS.java" generate_java_java;
11435
11436   List.iter (
11437     fun (typ, jtyp) ->
11438       let cols = cols_of_struct typ in
11439       let filename = sprintf "java/com/redhat/et/libguestfs/%s.java" jtyp in
11440       output_to filename (generate_java_struct jtyp cols);
11441   ) java_structs;
11442
11443   output_to "java/Makefile.inc" generate_java_makefile_inc;
11444   output_to "java/com_redhat_et_libguestfs_GuestFS.c" generate_java_c;
11445   output_to "java/Bindtests.java" generate_java_bindtests;
11446   output_to "haskell/Guestfs.hs" generate_haskell_hs;
11447   output_to "haskell/Bindtests.hs" generate_haskell_bindtests;
11448   output_to "csharp/Libguestfs.cs" generate_csharp;
11449
11450   (* Always generate this file last, and unconditionally.  It's used
11451    * by the Makefile to know when we must re-run the generator.
11452    *)
11453   let chan = open_out "src/stamp-generator" in
11454   fprintf chan "1\n";
11455   close_out chan;
11456
11457   printf "generated %d lines of code\n" !lines