f206be1be822d383c6946200f45bf7272a9f1c1b
[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 "      return -1;\n";
7407   pr "    }\n";
7408   pr "  return 0;\n";
7409   pr "}\n";
7410   pr "\n"
7411
7412 (* Readline completion for guestfish. *)
7413 and generate_fish_completion () =
7414   generate_header CStyle GPLv2plus;
7415
7416   let all_functions =
7417     List.filter (
7418       fun (_, _, _, flags, _, _, _) -> not (List.mem NotInFish flags)
7419     ) all_functions in
7420
7421   pr "\
7422 #include <config.h>
7423
7424 #include <stdio.h>
7425 #include <stdlib.h>
7426 #include <string.h>
7427
7428 #ifdef HAVE_LIBREADLINE
7429 #include <readline/readline.h>
7430 #endif
7431
7432 #include \"fish.h\"
7433
7434 #ifdef HAVE_LIBREADLINE
7435
7436 static const char *const commands[] = {
7437   BUILTIN_COMMANDS_FOR_COMPLETION,
7438 ";
7439
7440   (* Get the commands, including the aliases.  They don't need to be
7441    * sorted - the generator() function just does a dumb linear search.
7442    *)
7443   let commands =
7444     List.map (
7445       fun (name, _, _, flags, _, _, _) ->
7446         let name2 = replace_char name '_' '-' in
7447         let alias =
7448           try find_map (function FishAlias n -> Some n | _ -> None) flags
7449           with Not_found -> name in
7450
7451         if name <> alias then [name2; alias] else [name2]
7452     ) all_functions in
7453   let commands = List.flatten commands in
7454
7455   List.iter (pr "  \"%s\",\n") commands;
7456
7457   pr "  NULL
7458 };
7459
7460 static char *
7461 generator (const char *text, int state)
7462 {
7463   static int index, len;
7464   const char *name;
7465
7466   if (!state) {
7467     index = 0;
7468     len = strlen (text);
7469   }
7470
7471   rl_attempted_completion_over = 1;
7472
7473   while ((name = commands[index]) != NULL) {
7474     index++;
7475     if (STRCASEEQLEN (name, text, len))
7476       return strdup (name);
7477   }
7478
7479   return NULL;
7480 }
7481
7482 #endif /* HAVE_LIBREADLINE */
7483
7484 #ifdef HAVE_RL_COMPLETION_MATCHES
7485 #define RL_COMPLETION_MATCHES rl_completion_matches
7486 #else
7487 #ifdef HAVE_COMPLETION_MATCHES
7488 #define RL_COMPLETION_MATCHES completion_matches
7489 #endif
7490 #endif /* else just fail if we don't have either symbol */
7491
7492 char **
7493 do_completion (const char *text, int start, int end)
7494 {
7495   char **matches = NULL;
7496
7497 #ifdef HAVE_LIBREADLINE
7498   rl_completion_append_character = ' ';
7499
7500   if (start == 0)
7501     matches = RL_COMPLETION_MATCHES (text, generator);
7502   else if (complete_dest_paths)
7503     matches = RL_COMPLETION_MATCHES (text, complete_dest_paths_generator);
7504 #endif
7505
7506   return matches;
7507 }
7508 ";
7509
7510 (* Generate the POD documentation for guestfish. *)
7511 and generate_fish_actions_pod () =
7512   let all_functions_sorted =
7513     List.filter (
7514       fun (_, _, _, flags, _, _, _) ->
7515         not (List.mem NotInFish flags || List.mem NotInDocs flags)
7516     ) all_functions_sorted in
7517
7518   let rex = Str.regexp "C<guestfs_\\([^>]+\\)>" in
7519
7520   List.iter (
7521     fun (name, style, _, flags, _, _, longdesc) ->
7522       let longdesc =
7523         Str.global_substitute rex (
7524           fun s ->
7525             let sub =
7526               try Str.matched_group 1 s
7527               with Not_found ->
7528                 failwithf "error substituting C<guestfs_...> in longdesc of function %s" name in
7529             "C<" ^ replace_char sub '_' '-' ^ ">"
7530         ) longdesc in
7531       let name = replace_char name '_' '-' in
7532       let alias =
7533         try find_map (function FishAlias n -> Some n | _ -> None) flags
7534         with Not_found -> name in
7535
7536       pr "=head2 %s" name;
7537       if name <> alias then
7538         pr " | %s" alias;
7539       pr "\n";
7540       pr "\n";
7541       pr " %s" name;
7542       List.iter (
7543         function
7544         | Pathname n | Device n | Dev_or_Path n | String n -> pr " %s" n
7545         | OptString n -> pr " %s" n
7546         | StringList n | DeviceList n -> pr " '%s ...'" n
7547         | Bool _ -> pr " true|false"
7548         | Int n -> pr " %s" n
7549         | Int64 n -> pr " %s" n
7550         | FileIn n | FileOut n -> pr " (%s|-)" n
7551       ) (snd style);
7552       pr "\n";
7553       pr "\n";
7554       pr "%s\n\n" longdesc;
7555
7556       if List.exists (function FileIn _ | FileOut _ -> true
7557                       | _ -> false) (snd style) then
7558         pr "Use C<-> instead of a filename to read/write from stdin/stdout.\n\n";
7559
7560       if List.mem ProtocolLimitWarning flags then
7561         pr "%s\n\n" protocol_limit_warning;
7562
7563       if List.mem DangerWillRobinson flags then
7564         pr "%s\n\n" danger_will_robinson;
7565
7566       match deprecation_notice flags with
7567       | None -> ()
7568       | Some txt -> pr "%s\n\n" txt
7569   ) all_functions_sorted
7570
7571 (* Generate a C function prototype. *)
7572 and generate_prototype ?(extern = true) ?(static = false) ?(semicolon = true)
7573     ?(single_line = false) ?(newline = false) ?(in_daemon = false)
7574     ?(prefix = "")
7575     ?handle name style =
7576   if extern then pr "extern ";
7577   if static then pr "static ";
7578   (match fst style with
7579    | RErr -> pr "int "
7580    | RInt _ -> pr "int "
7581    | RInt64 _ -> pr "int64_t "
7582    | RBool _ -> pr "int "
7583    | RConstString _ | RConstOptString _ -> pr "const char *"
7584    | RString _ | RBufferOut _ -> pr "char *"
7585    | RStringList _ | RHashtable _ -> pr "char **"
7586    | RStruct (_, typ) ->
7587        if not in_daemon then pr "struct guestfs_%s *" typ
7588        else pr "guestfs_int_%s *" typ
7589    | RStructList (_, typ) ->
7590        if not in_daemon then pr "struct guestfs_%s_list *" typ
7591        else pr "guestfs_int_%s_list *" typ
7592   );
7593   let is_RBufferOut = match fst style with RBufferOut _ -> true | _ -> false in
7594   pr "%s%s (" prefix name;
7595   if handle = None && List.length (snd style) = 0 && not is_RBufferOut then
7596     pr "void"
7597   else (
7598     let comma = ref false in
7599     (match handle with
7600      | None -> ()
7601      | Some handle -> pr "guestfs_h *%s" handle; comma := true
7602     );
7603     let next () =
7604       if !comma then (
7605         if single_line then pr ", " else pr ",\n\t\t"
7606       );
7607       comma := true
7608     in
7609     List.iter (
7610       function
7611       | Pathname n
7612       | Device n | Dev_or_Path n
7613       | String n
7614       | OptString n ->
7615           next ();
7616           pr "const char *%s" n
7617       | StringList n | DeviceList n ->
7618           next ();
7619           pr "char *const *%s" n
7620       | Bool n -> next (); pr "int %s" n
7621       | Int n -> next (); pr "int %s" n
7622       | Int64 n -> next (); pr "int64_t %s" n
7623       | FileIn n
7624       | FileOut n ->
7625           if not in_daemon then (next (); pr "const char *%s" n)
7626     ) (snd style);
7627     if is_RBufferOut then (next (); pr "size_t *size_r");
7628   );
7629   pr ")";
7630   if semicolon then pr ";";
7631   if newline then pr "\n"
7632
7633 (* Generate C call arguments, eg "(handle, foo, bar)" *)
7634 and generate_c_call_args ?handle ?(decl = false) style =
7635   pr "(";
7636   let comma = ref false in
7637   let next () =
7638     if !comma then pr ", ";
7639     comma := true
7640   in
7641   (match handle with
7642    | None -> ()
7643    | Some handle -> pr "%s" handle; comma := true
7644   );
7645   List.iter (
7646     fun arg ->
7647       next ();
7648       pr "%s" (name_of_argt arg)
7649   ) (snd style);
7650   (* For RBufferOut calls, add implicit &size parameter. *)
7651   if not decl then (
7652     match fst style with
7653     | RBufferOut _ ->
7654         next ();
7655         pr "&size"
7656     | _ -> ()
7657   );
7658   pr ")"
7659
7660 (* Generate the OCaml bindings interface. *)
7661 and generate_ocaml_mli () =
7662   generate_header OCamlStyle LGPLv2plus;
7663
7664   pr "\
7665 (** For API documentation you should refer to the C API
7666     in the guestfs(3) manual page.  The OCaml API uses almost
7667     exactly the same calls. *)
7668
7669 type t
7670 (** A [guestfs_h] handle. *)
7671
7672 exception Error of string
7673 (** This exception is raised when there is an error. *)
7674
7675 exception Handle_closed of string
7676 (** This exception is raised if you use a {!Guestfs.t} handle
7677     after calling {!close} on it.  The string is the name of
7678     the function. *)
7679
7680 val create : unit -> t
7681 (** Create a {!Guestfs.t} handle. *)
7682
7683 val close : t -> unit
7684 (** Close the {!Guestfs.t} handle and free up all resources used
7685     by it immediately.
7686
7687     Handles are closed by the garbage collector when they become
7688     unreferenced, but callers can call this in order to provide
7689     predictable cleanup. *)
7690
7691 ";
7692   generate_ocaml_structure_decls ();
7693
7694   (* The actions. *)
7695   List.iter (
7696     fun (name, style, _, _, _, shortdesc, _) ->
7697       generate_ocaml_prototype name style;
7698       pr "(** %s *)\n" shortdesc;
7699       pr "\n"
7700   ) all_functions_sorted
7701
7702 (* Generate the OCaml bindings implementation. *)
7703 and generate_ocaml_ml () =
7704   generate_header OCamlStyle LGPLv2plus;
7705
7706   pr "\
7707 type t
7708
7709 exception Error of string
7710 exception Handle_closed of string
7711
7712 external create : unit -> t = \"ocaml_guestfs_create\"
7713 external close : t -> unit = \"ocaml_guestfs_close\"
7714
7715 (* Give the exceptions names, so they can be raised from the C code. *)
7716 let () =
7717   Callback.register_exception \"ocaml_guestfs_error\" (Error \"\");
7718   Callback.register_exception \"ocaml_guestfs_closed\" (Handle_closed \"\")
7719
7720 ";
7721
7722   generate_ocaml_structure_decls ();
7723
7724   (* The actions. *)
7725   List.iter (
7726     fun (name, style, _, _, _, shortdesc, _) ->
7727       generate_ocaml_prototype ~is_external:true name style;
7728   ) all_functions_sorted
7729
7730 (* Generate the OCaml bindings C implementation. *)
7731 and generate_ocaml_c () =
7732   generate_header CStyle LGPLv2plus;
7733
7734   pr "\
7735 #include <stdio.h>
7736 #include <stdlib.h>
7737 #include <string.h>
7738
7739 #include <caml/config.h>
7740 #include <caml/alloc.h>
7741 #include <caml/callback.h>
7742 #include <caml/fail.h>
7743 #include <caml/memory.h>
7744 #include <caml/mlvalues.h>
7745 #include <caml/signals.h>
7746
7747 #include <guestfs.h>
7748
7749 #include \"guestfs_c.h\"
7750
7751 /* Copy a hashtable of string pairs into an assoc-list.  We return
7752  * the list in reverse order, but hashtables aren't supposed to be
7753  * ordered anyway.
7754  */
7755 static CAMLprim value
7756 copy_table (char * const * argv)
7757 {
7758   CAMLparam0 ();
7759   CAMLlocal5 (rv, pairv, kv, vv, cons);
7760   int i;
7761
7762   rv = Val_int (0);
7763   for (i = 0; argv[i] != NULL; i += 2) {
7764     kv = caml_copy_string (argv[i]);
7765     vv = caml_copy_string (argv[i+1]);
7766     pairv = caml_alloc (2, 0);
7767     Store_field (pairv, 0, kv);
7768     Store_field (pairv, 1, vv);
7769     cons = caml_alloc (2, 0);
7770     Store_field (cons, 1, rv);
7771     rv = cons;
7772     Store_field (cons, 0, pairv);
7773   }
7774
7775   CAMLreturn (rv);
7776 }
7777
7778 ";
7779
7780   (* Struct copy functions. *)
7781
7782   let emit_ocaml_copy_list_function typ =
7783     pr "static CAMLprim value\n";
7784     pr "copy_%s_list (const struct guestfs_%s_list *%ss)\n" typ typ typ;
7785     pr "{\n";
7786     pr "  CAMLparam0 ();\n";
7787     pr "  CAMLlocal2 (rv, v);\n";
7788     pr "  unsigned int i;\n";
7789     pr "\n";
7790     pr "  if (%ss->len == 0)\n" typ;
7791     pr "    CAMLreturn (Atom (0));\n";
7792     pr "  else {\n";
7793     pr "    rv = caml_alloc (%ss->len, 0);\n" typ;
7794     pr "    for (i = 0; i < %ss->len; ++i) {\n" typ;
7795     pr "      v = copy_%s (&%ss->val[i]);\n" typ typ;
7796     pr "      caml_modify (&Field (rv, i), v);\n";
7797     pr "    }\n";
7798     pr "    CAMLreturn (rv);\n";
7799     pr "  }\n";
7800     pr "}\n";
7801     pr "\n";
7802   in
7803
7804   List.iter (
7805     fun (typ, cols) ->
7806       let has_optpercent_col =
7807         List.exists (function (_, FOptPercent) -> true | _ -> false) cols in
7808
7809       pr "static CAMLprim value\n";
7810       pr "copy_%s (const struct guestfs_%s *%s)\n" typ typ typ;
7811       pr "{\n";
7812       pr "  CAMLparam0 ();\n";
7813       if has_optpercent_col then
7814         pr "  CAMLlocal3 (rv, v, v2);\n"
7815       else
7816         pr "  CAMLlocal2 (rv, v);\n";
7817       pr "\n";
7818       pr "  rv = caml_alloc (%d, 0);\n" (List.length cols);
7819       iteri (
7820         fun i col ->
7821           (match col with
7822            | name, FString ->
7823                pr "  v = caml_copy_string (%s->%s);\n" typ name
7824            | name, FBuffer ->
7825                pr "  v = caml_alloc_string (%s->%s_len);\n" typ name;
7826                pr "  memcpy (String_val (v), %s->%s, %s->%s_len);\n"
7827                  typ name typ name
7828            | name, FUUID ->
7829                pr "  v = caml_alloc_string (32);\n";
7830                pr "  memcpy (String_val (v), %s->%s, 32);\n" typ name
7831            | name, (FBytes|FInt64|FUInt64) ->
7832                pr "  v = caml_copy_int64 (%s->%s);\n" typ name
7833            | name, (FInt32|FUInt32) ->
7834                pr "  v = caml_copy_int32 (%s->%s);\n" typ name
7835            | name, FOptPercent ->
7836                pr "  if (%s->%s >= 0) { /* Some %s */\n" typ name name;
7837                pr "    v2 = caml_copy_double (%s->%s);\n" typ name;
7838                pr "    v = caml_alloc (1, 0);\n";
7839                pr "    Store_field (v, 0, v2);\n";
7840                pr "  } else /* None */\n";
7841                pr "    v = Val_int (0);\n";
7842            | name, FChar ->
7843                pr "  v = Val_int (%s->%s);\n" typ name
7844           );
7845           pr "  Store_field (rv, %d, v);\n" i
7846       ) cols;
7847       pr "  CAMLreturn (rv);\n";
7848       pr "}\n";
7849       pr "\n";
7850   ) structs;
7851
7852   (* Emit a copy_TYPE_list function definition only if that function is used. *)
7853   List.iter (
7854     function
7855     | typ, (RStructListOnly | RStructAndList) ->
7856         (* generate the function for typ *)
7857         emit_ocaml_copy_list_function typ
7858     | typ, _ -> () (* empty *)
7859   ) (rstructs_used_by all_functions);
7860
7861   (* The wrappers. *)
7862   List.iter (
7863     fun (name, style, _, _, _, _, _) ->
7864       pr "/* Automatically generated wrapper for function\n";
7865       pr " * ";
7866       generate_ocaml_prototype name style;
7867       pr " */\n";
7868       pr "\n";
7869
7870       let params =
7871         "gv" :: List.map (fun arg -> name_of_argt arg ^ "v") (snd style) in
7872
7873       let needs_extra_vs =
7874         match fst style with RConstOptString _ -> true | _ -> false in
7875
7876       pr "/* Emit prototype to appease gcc's -Wmissing-prototypes. */\n";
7877       pr "CAMLprim value ocaml_guestfs_%s (value %s" name (List.hd params);
7878       List.iter (pr ", value %s") (List.tl params); pr ");\n";
7879       pr "\n";
7880
7881       pr "CAMLprim value\n";
7882       pr "ocaml_guestfs_%s (value %s" name (List.hd params);
7883       List.iter (pr ", value %s") (List.tl params);
7884       pr ")\n";
7885       pr "{\n";
7886
7887       (match params with
7888        | [p1; p2; p3; p4; p5] ->
7889            pr "  CAMLparam5 (%s);\n" (String.concat ", " params)
7890        | p1 :: p2 :: p3 :: p4 :: p5 :: rest ->
7891            pr "  CAMLparam5 (%s);\n" (String.concat ", " [p1; p2; p3; p4; p5]);
7892            pr "  CAMLxparam%d (%s);\n"
7893              (List.length rest) (String.concat ", " rest)
7894        | ps ->
7895            pr "  CAMLparam%d (%s);\n" (List.length ps) (String.concat ", " ps)
7896       );
7897       if not needs_extra_vs then
7898         pr "  CAMLlocal1 (rv);\n"
7899       else
7900         pr "  CAMLlocal3 (rv, v, v2);\n";
7901       pr "\n";
7902
7903       pr "  guestfs_h *g = Guestfs_val (gv);\n";
7904       pr "  if (g == NULL)\n";
7905       pr "    ocaml_guestfs_raise_closed (\"%s\");\n" name;
7906       pr "\n";
7907
7908       List.iter (
7909         function
7910         | Pathname n
7911         | Device n | Dev_or_Path n
7912         | String n
7913         | FileIn n
7914         | FileOut n ->
7915             pr "  const char *%s = String_val (%sv);\n" n n
7916         | OptString n ->
7917             pr "  const char *%s =\n" n;
7918             pr "    %sv != Val_int (0) ? String_val (Field (%sv, 0)) : NULL;\n"
7919               n n
7920         | StringList n | DeviceList n ->
7921             pr "  char **%s = ocaml_guestfs_strings_val (g, %sv);\n" n n
7922         | Bool n ->
7923             pr "  int %s = Bool_val (%sv);\n" n n
7924         | Int n ->
7925             pr "  int %s = Int_val (%sv);\n" n n
7926         | Int64 n ->
7927             pr "  int64_t %s = Int64_val (%sv);\n" n n
7928       ) (snd style);
7929       let error_code =
7930         match fst style with
7931         | RErr -> pr "  int r;\n"; "-1"
7932         | RInt _ -> pr "  int r;\n"; "-1"
7933         | RInt64 _ -> pr "  int64_t r;\n"; "-1"
7934         | RBool _ -> pr "  int r;\n"; "-1"
7935         | RConstString _ | RConstOptString _ ->
7936             pr "  const char *r;\n"; "NULL"
7937         | RString _ -> pr "  char *r;\n"; "NULL"
7938         | RStringList _ ->
7939             pr "  int i;\n";
7940             pr "  char **r;\n";
7941             "NULL"
7942         | RStruct (_, typ) ->
7943             pr "  struct guestfs_%s *r;\n" typ; "NULL"
7944         | RStructList (_, typ) ->
7945             pr "  struct guestfs_%s_list *r;\n" typ; "NULL"
7946         | RHashtable _ ->
7947             pr "  int i;\n";
7948             pr "  char **r;\n";
7949             "NULL"
7950         | RBufferOut _ ->
7951             pr "  char *r;\n";
7952             pr "  size_t size;\n";
7953             "NULL" in
7954       pr "\n";
7955
7956       pr "  caml_enter_blocking_section ();\n";
7957       pr "  r = guestfs_%s " name;
7958       generate_c_call_args ~handle:"g" style;
7959       pr ";\n";
7960       pr "  caml_leave_blocking_section ();\n";
7961
7962       List.iter (
7963         function
7964         | StringList n | DeviceList n ->
7965             pr "  ocaml_guestfs_free_strings (%s);\n" n;
7966         | Pathname _ | Device _ | Dev_or_Path _ | String _ | OptString _
7967         | Bool _ | Int _ | Int64 _
7968         | FileIn _ | FileOut _ -> ()
7969       ) (snd style);
7970
7971       pr "  if (r == %s)\n" error_code;
7972       pr "    ocaml_guestfs_raise_error (g, \"%s\");\n" name;
7973       pr "\n";
7974
7975       (match fst style with
7976        | RErr -> pr "  rv = Val_unit;\n"
7977        | RInt _ -> pr "  rv = Val_int (r);\n"
7978        | RInt64 _ ->
7979            pr "  rv = caml_copy_int64 (r);\n"
7980        | RBool _ -> pr "  rv = Val_bool (r);\n"
7981        | RConstString _ ->
7982            pr "  rv = caml_copy_string (r);\n"
7983        | RConstOptString _ ->
7984            pr "  if (r) { /* Some string */\n";
7985            pr "    v = caml_alloc (1, 0);\n";
7986            pr "    v2 = caml_copy_string (r);\n";
7987            pr "    Store_field (v, 0, v2);\n";
7988            pr "  } else /* None */\n";
7989            pr "    v = Val_int (0);\n";
7990        | RString _ ->
7991            pr "  rv = caml_copy_string (r);\n";
7992            pr "  free (r);\n"
7993        | RStringList _ ->
7994            pr "  rv = caml_copy_string_array ((const char **) r);\n";
7995            pr "  for (i = 0; r[i] != NULL; ++i) free (r[i]);\n";
7996            pr "  free (r);\n"
7997        | RStruct (_, typ) ->
7998            pr "  rv = copy_%s (r);\n" typ;
7999            pr "  guestfs_free_%s (r);\n" typ;
8000        | RStructList (_, typ) ->
8001            pr "  rv = copy_%s_list (r);\n" typ;
8002            pr "  guestfs_free_%s_list (r);\n" typ;
8003        | RHashtable _ ->
8004            pr "  rv = copy_table (r);\n";
8005            pr "  for (i = 0; r[i] != NULL; ++i) free (r[i]);\n";
8006            pr "  free (r);\n";
8007        | RBufferOut _ ->
8008            pr "  rv = caml_alloc_string (size);\n";
8009            pr "  memcpy (String_val (rv), r, size);\n";
8010       );
8011
8012       pr "  CAMLreturn (rv);\n";
8013       pr "}\n";
8014       pr "\n";
8015
8016       if List.length params > 5 then (
8017         pr "/* Emit prototype to appease gcc's -Wmissing-prototypes. */\n";
8018         pr "CAMLprim value ";
8019         pr "ocaml_guestfs_%s_byte (value *argv, int argn);\n" name;
8020         pr "CAMLprim value\n";
8021         pr "ocaml_guestfs_%s_byte (value *argv, int argn)\n" name;
8022         pr "{\n";
8023         pr "  return ocaml_guestfs_%s (argv[0]" name;
8024         iteri (fun i _ -> pr ", argv[%d]" i) (List.tl params);
8025         pr ");\n";
8026         pr "}\n";
8027         pr "\n"
8028       )
8029   ) all_functions_sorted
8030
8031 and generate_ocaml_structure_decls () =
8032   List.iter (
8033     fun (typ, cols) ->
8034       pr "type %s = {\n" typ;
8035       List.iter (
8036         function
8037         | name, FString -> pr "  %s : string;\n" name
8038         | name, FBuffer -> pr "  %s : string;\n" name
8039         | name, FUUID -> pr "  %s : string;\n" name
8040         | name, (FBytes|FInt64|FUInt64) -> pr "  %s : int64;\n" name
8041         | name, (FInt32|FUInt32) -> pr "  %s : int32;\n" name
8042         | name, FChar -> pr "  %s : char;\n" name
8043         | name, FOptPercent -> pr "  %s : float option;\n" name
8044       ) cols;
8045       pr "}\n";
8046       pr "\n"
8047   ) structs
8048
8049 and generate_ocaml_prototype ?(is_external = false) name style =
8050   if is_external then pr "external " else pr "val ";
8051   pr "%s : t -> " name;
8052   List.iter (
8053     function
8054     | Pathname _ | Device _ | Dev_or_Path _ | String _ | FileIn _ | FileOut _ -> pr "string -> "
8055     | OptString _ -> pr "string option -> "
8056     | StringList _ | DeviceList _ -> pr "string array -> "
8057     | Bool _ -> pr "bool -> "
8058     | Int _ -> pr "int -> "
8059     | Int64 _ -> pr "int64 -> "
8060   ) (snd style);
8061   (match fst style with
8062    | RErr -> pr "unit" (* all errors are turned into exceptions *)
8063    | RInt _ -> pr "int"
8064    | RInt64 _ -> pr "int64"
8065    | RBool _ -> pr "bool"
8066    | RConstString _ -> pr "string"
8067    | RConstOptString _ -> pr "string option"
8068    | RString _ | RBufferOut _ -> pr "string"
8069    | RStringList _ -> pr "string array"
8070    | RStruct (_, typ) -> pr "%s" typ
8071    | RStructList (_, typ) -> pr "%s array" typ
8072    | RHashtable _ -> pr "(string * string) list"
8073   );
8074   if is_external then (
8075     pr " = ";
8076     if List.length (snd style) + 1 > 5 then
8077       pr "\"ocaml_guestfs_%s_byte\" " name;
8078     pr "\"ocaml_guestfs_%s\"" name
8079   );
8080   pr "\n"
8081
8082 (* Generate Perl xs code, a sort of crazy variation of C with macros. *)
8083 and generate_perl_xs () =
8084   generate_header CStyle LGPLv2plus;
8085
8086   pr "\
8087 #include \"EXTERN.h\"
8088 #include \"perl.h\"
8089 #include \"XSUB.h\"
8090
8091 #include <guestfs.h>
8092
8093 #ifndef PRId64
8094 #define PRId64 \"lld\"
8095 #endif
8096
8097 static SV *
8098 my_newSVll(long long val) {
8099 #ifdef USE_64_BIT_ALL
8100   return newSViv(val);
8101 #else
8102   char buf[100];
8103   int len;
8104   len = snprintf(buf, 100, \"%%\" PRId64, val);
8105   return newSVpv(buf, len);
8106 #endif
8107 }
8108
8109 #ifndef PRIu64
8110 #define PRIu64 \"llu\"
8111 #endif
8112
8113 static SV *
8114 my_newSVull(unsigned long long val) {
8115 #ifdef USE_64_BIT_ALL
8116   return newSVuv(val);
8117 #else
8118   char buf[100];
8119   int len;
8120   len = snprintf(buf, 100, \"%%\" PRIu64, val);
8121   return newSVpv(buf, len);
8122 #endif
8123 }
8124
8125 /* http://www.perlmonks.org/?node_id=680842 */
8126 static char **
8127 XS_unpack_charPtrPtr (SV *arg) {
8128   char **ret;
8129   AV *av;
8130   I32 i;
8131
8132   if (!arg || !SvOK (arg) || !SvROK (arg) || SvTYPE (SvRV (arg)) != SVt_PVAV)
8133     croak (\"array reference expected\");
8134
8135   av = (AV *)SvRV (arg);
8136   ret = malloc ((av_len (av) + 1 + 1) * sizeof (char *));
8137   if (!ret)
8138     croak (\"malloc failed\");
8139
8140   for (i = 0; i <= av_len (av); i++) {
8141     SV **elem = av_fetch (av, i, 0);
8142
8143     if (!elem || !*elem)
8144       croak (\"missing element in list\");
8145
8146     ret[i] = SvPV_nolen (*elem);
8147   }
8148
8149   ret[i] = NULL;
8150
8151   return ret;
8152 }
8153
8154 MODULE = Sys::Guestfs  PACKAGE = Sys::Guestfs
8155
8156 PROTOTYPES: ENABLE
8157
8158 guestfs_h *
8159 _create ()
8160    CODE:
8161       RETVAL = guestfs_create ();
8162       if (!RETVAL)
8163         croak (\"could not create guestfs handle\");
8164       guestfs_set_error_handler (RETVAL, NULL, NULL);
8165  OUTPUT:
8166       RETVAL
8167
8168 void
8169 DESTROY (g)
8170       guestfs_h *g;
8171  PPCODE:
8172       guestfs_close (g);
8173
8174 ";
8175
8176   List.iter (
8177     fun (name, style, _, _, _, _, _) ->
8178       (match fst style with
8179        | RErr -> pr "void\n"
8180        | RInt _ -> pr "SV *\n"
8181        | RInt64 _ -> pr "SV *\n"
8182        | RBool _ -> pr "SV *\n"
8183        | RConstString _ -> pr "SV *\n"
8184        | RConstOptString _ -> pr "SV *\n"
8185        | RString _ -> pr "SV *\n"
8186        | RBufferOut _ -> pr "SV *\n"
8187        | RStringList _
8188        | RStruct _ | RStructList _
8189        | RHashtable _ ->
8190            pr "void\n" (* all lists returned implictly on the stack *)
8191       );
8192       (* Call and arguments. *)
8193       pr "%s " name;
8194       generate_c_call_args ~handle:"g" ~decl:true style;
8195       pr "\n";
8196       pr "      guestfs_h *g;\n";
8197       iteri (
8198         fun i ->
8199           function
8200           | Pathname n | Device n | Dev_or_Path n | String n | FileIn n | FileOut n ->
8201               pr "      char *%s;\n" n
8202           | OptString n ->
8203               (* http://www.perlmonks.org/?node_id=554277
8204                * Note that the implicit handle argument means we have
8205                * to add 1 to the ST(x) operator.
8206                *)
8207               pr "      char *%s = SvOK(ST(%d)) ? SvPV_nolen(ST(%d)) : NULL;\n" n (i+1) (i+1)
8208           | StringList n | DeviceList n -> pr "      char **%s;\n" n
8209           | Bool n -> pr "      int %s;\n" n
8210           | Int n -> pr "      int %s;\n" n
8211           | Int64 n -> pr "      int64_t %s;\n" n
8212       ) (snd style);
8213
8214       let do_cleanups () =
8215         List.iter (
8216           function
8217           | Pathname _ | Device _ | Dev_or_Path _ | String _ | OptString _
8218           | Bool _ | Int _ | Int64 _
8219           | FileIn _ | FileOut _ -> ()
8220           | StringList n | DeviceList n -> pr "      free (%s);\n" n
8221         ) (snd style)
8222       in
8223
8224       (* Code. *)
8225       (match fst style with
8226        | RErr ->
8227            pr "PREINIT:\n";
8228            pr "      int r;\n";
8229            pr " PPCODE:\n";
8230            pr "      r = guestfs_%s " name;
8231            generate_c_call_args ~handle:"g" style;
8232            pr ";\n";
8233            do_cleanups ();
8234            pr "      if (r == -1)\n";
8235            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8236        | RInt n
8237        | RBool n ->
8238            pr "PREINIT:\n";
8239            pr "      int %s;\n" n;
8240            pr "   CODE:\n";
8241            pr "      %s = guestfs_%s " n name;
8242            generate_c_call_args ~handle:"g" style;
8243            pr ";\n";
8244            do_cleanups ();
8245            pr "      if (%s == -1)\n" n;
8246            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8247            pr "      RETVAL = newSViv (%s);\n" n;
8248            pr " OUTPUT:\n";
8249            pr "      RETVAL\n"
8250        | RInt64 n ->
8251            pr "PREINIT:\n";
8252            pr "      int64_t %s;\n" n;
8253            pr "   CODE:\n";
8254            pr "      %s = guestfs_%s " n name;
8255            generate_c_call_args ~handle:"g" style;
8256            pr ";\n";
8257            do_cleanups ();
8258            pr "      if (%s == -1)\n" n;
8259            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8260            pr "      RETVAL = my_newSVll (%s);\n" n;
8261            pr " OUTPUT:\n";
8262            pr "      RETVAL\n"
8263        | RConstString n ->
8264            pr "PREINIT:\n";
8265            pr "      const char *%s;\n" n;
8266            pr "   CODE:\n";
8267            pr "      %s = guestfs_%s " n name;
8268            generate_c_call_args ~handle:"g" style;
8269            pr ";\n";
8270            do_cleanups ();
8271            pr "      if (%s == NULL)\n" n;
8272            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8273            pr "      RETVAL = newSVpv (%s, 0);\n" n;
8274            pr " OUTPUT:\n";
8275            pr "      RETVAL\n"
8276        | RConstOptString n ->
8277            pr "PREINIT:\n";
8278            pr "      const char *%s;\n" n;
8279            pr "   CODE:\n";
8280            pr "      %s = guestfs_%s " n name;
8281            generate_c_call_args ~handle:"g" style;
8282            pr ";\n";
8283            do_cleanups ();
8284            pr "      if (%s == NULL)\n" n;
8285            pr "        RETVAL = &PL_sv_undef;\n";
8286            pr "      else\n";
8287            pr "        RETVAL = newSVpv (%s, 0);\n" n;
8288            pr " OUTPUT:\n";
8289            pr "      RETVAL\n"
8290        | RString n ->
8291            pr "PREINIT:\n";
8292            pr "      char *%s;\n" n;
8293            pr "   CODE:\n";
8294            pr "      %s = guestfs_%s " n name;
8295            generate_c_call_args ~handle:"g" style;
8296            pr ";\n";
8297            do_cleanups ();
8298            pr "      if (%s == NULL)\n" n;
8299            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8300            pr "      RETVAL = newSVpv (%s, 0);\n" n;
8301            pr "      free (%s);\n" n;
8302            pr " OUTPUT:\n";
8303            pr "      RETVAL\n"
8304        | RStringList n | RHashtable n ->
8305            pr "PREINIT:\n";
8306            pr "      char **%s;\n" n;
8307            pr "      int i, n;\n";
8308            pr " PPCODE:\n";
8309            pr "      %s = guestfs_%s " n name;
8310            generate_c_call_args ~handle:"g" style;
8311            pr ";\n";
8312            do_cleanups ();
8313            pr "      if (%s == NULL)\n" n;
8314            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8315            pr "      for (n = 0; %s[n] != NULL; ++n) /**/;\n" n;
8316            pr "      EXTEND (SP, n);\n";
8317            pr "      for (i = 0; i < n; ++i) {\n";
8318            pr "        PUSHs (sv_2mortal (newSVpv (%s[i], 0)));\n" n;
8319            pr "        free (%s[i]);\n" n;
8320            pr "      }\n";
8321            pr "      free (%s);\n" n;
8322        | RStruct (n, typ) ->
8323            let cols = cols_of_struct typ in
8324            generate_perl_struct_code typ cols name style n do_cleanups
8325        | RStructList (n, typ) ->
8326            let cols = cols_of_struct typ in
8327            generate_perl_struct_list_code typ cols name style n do_cleanups
8328        | RBufferOut n ->
8329            pr "PREINIT:\n";
8330            pr "      char *%s;\n" n;
8331            pr "      size_t size;\n";
8332            pr "   CODE:\n";
8333            pr "      %s = guestfs_%s " n name;
8334            generate_c_call_args ~handle:"g" style;
8335            pr ";\n";
8336            do_cleanups ();
8337            pr "      if (%s == NULL)\n" n;
8338            pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8339            pr "      RETVAL = newSVpvn (%s, size);\n" n;
8340            pr "      free (%s);\n" n;
8341            pr " OUTPUT:\n";
8342            pr "      RETVAL\n"
8343       );
8344
8345       pr "\n"
8346   ) all_functions
8347
8348 and generate_perl_struct_list_code typ cols name style n do_cleanups =
8349   pr "PREINIT:\n";
8350   pr "      struct guestfs_%s_list *%s;\n" typ n;
8351   pr "      int i;\n";
8352   pr "      HV *hv;\n";
8353   pr " PPCODE:\n";
8354   pr "      %s = guestfs_%s " n name;
8355   generate_c_call_args ~handle:"g" style;
8356   pr ";\n";
8357   do_cleanups ();
8358   pr "      if (%s == NULL)\n" n;
8359   pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8360   pr "      EXTEND (SP, %s->len);\n" n;
8361   pr "      for (i = 0; i < %s->len; ++i) {\n" n;
8362   pr "        hv = newHV ();\n";
8363   List.iter (
8364     function
8365     | name, FString ->
8366         pr "        (void) hv_store (hv, \"%s\", %d, newSVpv (%s->val[i].%s, 0), 0);\n"
8367           name (String.length name) n name
8368     | name, FUUID ->
8369         pr "        (void) hv_store (hv, \"%s\", %d, newSVpv (%s->val[i].%s, 32), 0);\n"
8370           name (String.length name) n name
8371     | name, FBuffer ->
8372         pr "        (void) hv_store (hv, \"%s\", %d, newSVpvn (%s->val[i].%s, %s->val[i].%s_len), 0);\n"
8373           name (String.length name) n name n name
8374     | name, (FBytes|FUInt64) ->
8375         pr "        (void) hv_store (hv, \"%s\", %d, my_newSVull (%s->val[i].%s), 0);\n"
8376           name (String.length name) n name
8377     | name, FInt64 ->
8378         pr "        (void) hv_store (hv, \"%s\", %d, my_newSVll (%s->val[i].%s), 0);\n"
8379           name (String.length name) n name
8380     | name, (FInt32|FUInt32) ->
8381         pr "        (void) hv_store (hv, \"%s\", %d, newSVnv (%s->val[i].%s), 0);\n"
8382           name (String.length name) n name
8383     | name, FChar ->
8384         pr "        (void) hv_store (hv, \"%s\", %d, newSVpv (&%s->val[i].%s, 1), 0);\n"
8385           name (String.length name) n name
8386     | name, FOptPercent ->
8387         pr "        (void) hv_store (hv, \"%s\", %d, newSVnv (%s->val[i].%s), 0);\n"
8388           name (String.length name) n name
8389   ) cols;
8390   pr "        PUSHs (sv_2mortal (newRV ((SV *) hv)));\n";
8391   pr "      }\n";
8392   pr "      guestfs_free_%s_list (%s);\n" typ n
8393
8394 and generate_perl_struct_code typ cols name style n do_cleanups =
8395   pr "PREINIT:\n";
8396   pr "      struct guestfs_%s *%s;\n" typ n;
8397   pr " PPCODE:\n";
8398   pr "      %s = guestfs_%s " n name;
8399   generate_c_call_args ~handle:"g" style;
8400   pr ";\n";
8401   do_cleanups ();
8402   pr "      if (%s == NULL)\n" n;
8403   pr "        croak (\"%%s\", guestfs_last_error (g));\n";
8404   pr "      EXTEND (SP, 2 * %d);\n" (List.length cols);
8405   List.iter (
8406     fun ((name, _) as col) ->
8407       pr "      PUSHs (sv_2mortal (newSVpv (\"%s\", 0)));\n" name;
8408
8409       match col with
8410       | name, FString ->
8411           pr "      PUSHs (sv_2mortal (newSVpv (%s->%s, 0)));\n"
8412             n name
8413       | name, FBuffer ->
8414           pr "      PUSHs (sv_2mortal (newSVpvn (%s->%s, %s->%s_len)));\n"
8415             n name n name
8416       | name, FUUID ->
8417           pr "      PUSHs (sv_2mortal (newSVpv (%s->%s, 32)));\n"
8418             n name
8419       | name, (FBytes|FUInt64) ->
8420           pr "      PUSHs (sv_2mortal (my_newSVull (%s->%s)));\n"
8421             n name
8422       | name, FInt64 ->
8423           pr "      PUSHs (sv_2mortal (my_newSVll (%s->%s)));\n"
8424             n name
8425       | name, (FInt32|FUInt32) ->
8426           pr "      PUSHs (sv_2mortal (newSVnv (%s->%s)));\n"
8427             n name
8428       | name, FChar ->
8429           pr "      PUSHs (sv_2mortal (newSVpv (&%s->%s, 1)));\n"
8430             n name
8431       | name, FOptPercent ->
8432           pr "      PUSHs (sv_2mortal (newSVnv (%s->%s)));\n"
8433             n name
8434   ) cols;
8435   pr "      free (%s);\n" n
8436
8437 (* Generate Sys/Guestfs.pm. *)
8438 and generate_perl_pm () =
8439   generate_header HashStyle LGPLv2plus;
8440
8441   pr "\
8442 =pod
8443
8444 =head1 NAME
8445
8446 Sys::Guestfs - Perl bindings for libguestfs
8447
8448 =head1 SYNOPSIS
8449
8450  use Sys::Guestfs;
8451
8452  my $h = Sys::Guestfs->new ();
8453  $h->add_drive ('guest.img');
8454  $h->launch ();
8455  $h->mount ('/dev/sda1', '/');
8456  $h->touch ('/hello');
8457  $h->sync ();
8458
8459 =head1 DESCRIPTION
8460
8461 The C<Sys::Guestfs> module provides a Perl XS binding to the
8462 libguestfs API for examining and modifying virtual machine
8463 disk images.
8464
8465 Amongst the things this is good for: making batch configuration
8466 changes to guests, getting disk used/free statistics (see also:
8467 virt-df), migrating between virtualization systems (see also:
8468 virt-p2v), performing partial backups, performing partial guest
8469 clones, cloning guests and changing registry/UUID/hostname info, and
8470 much else besides.
8471
8472 Libguestfs uses Linux kernel and qemu code, and can access any type of
8473 guest filesystem that Linux and qemu can, including but not limited
8474 to: ext2/3/4, btrfs, FAT and NTFS, LVM, many different disk partition
8475 schemes, qcow, qcow2, vmdk.
8476
8477 Libguestfs provides ways to enumerate guest storage (eg. partitions,
8478 LVs, what filesystem is in each LV, etc.).  It can also run commands
8479 in the context of the guest.  Also you can access filesystems over
8480 FUSE.
8481
8482 See also L<Sys::Guestfs::Lib(3)> for a set of useful library
8483 functions for using libguestfs from Perl, including integration
8484 with libvirt.
8485
8486 =head1 ERRORS
8487
8488 All errors turn into calls to C<croak> (see L<Carp(3)>).
8489
8490 =head1 METHODS
8491
8492 =over 4
8493
8494 =cut
8495
8496 package Sys::Guestfs;
8497
8498 use strict;
8499 use warnings;
8500
8501 require XSLoader;
8502 XSLoader::load ('Sys::Guestfs');
8503
8504 =item $h = Sys::Guestfs->new ();
8505
8506 Create a new guestfs handle.
8507
8508 =cut
8509
8510 sub new {
8511   my $proto = shift;
8512   my $class = ref ($proto) || $proto;
8513
8514   my $self = Sys::Guestfs::_create ();
8515   bless $self, $class;
8516   return $self;
8517 }
8518
8519 ";
8520
8521   (* Actions.  We only need to print documentation for these as
8522    * they are pulled in from the XS code automatically.
8523    *)
8524   List.iter (
8525     fun (name, style, _, flags, _, _, longdesc) ->
8526       if not (List.mem NotInDocs flags) then (
8527         let longdesc = replace_str longdesc "C<guestfs_" "C<$h-E<gt>" in
8528         pr "=item ";
8529         generate_perl_prototype name style;
8530         pr "\n\n";
8531         pr "%s\n\n" longdesc;
8532         if List.mem ProtocolLimitWarning flags then
8533           pr "%s\n\n" protocol_limit_warning;
8534         if List.mem DangerWillRobinson flags then
8535           pr "%s\n\n" danger_will_robinson;
8536         match deprecation_notice flags with
8537         | None -> ()
8538         | Some txt -> pr "%s\n\n" txt
8539       )
8540   ) all_functions_sorted;
8541
8542   (* End of file. *)
8543   pr "\
8544 =cut
8545
8546 1;
8547
8548 =back
8549
8550 =head1 COPYRIGHT
8551
8552 Copyright (C) %s Red Hat Inc.
8553
8554 =head1 LICENSE
8555
8556 Please see the file COPYING.LIB for the full license.
8557
8558 =head1 SEE ALSO
8559
8560 L<guestfs(3)>,
8561 L<guestfish(1)>,
8562 L<http://libguestfs.org>,
8563 L<Sys::Guestfs::Lib(3)>.
8564
8565 =cut
8566 " copyright_years
8567
8568 and generate_perl_prototype name style =
8569   (match fst style with
8570    | RErr -> ()
8571    | RBool n
8572    | RInt n
8573    | RInt64 n
8574    | RConstString n
8575    | RConstOptString n
8576    | RString n
8577    | RBufferOut n -> pr "$%s = " n
8578    | RStruct (n,_)
8579    | RHashtable n -> pr "%%%s = " n
8580    | RStringList n
8581    | RStructList (n,_) -> pr "@%s = " n
8582   );
8583   pr "$h->%s (" name;
8584   let comma = ref false in
8585   List.iter (
8586     fun arg ->
8587       if !comma then pr ", ";
8588       comma := true;
8589       match arg with
8590       | Pathname n | Device n | Dev_or_Path n | String n
8591       | OptString n | Bool n | Int n | Int64 n | FileIn n | FileOut n ->
8592           pr "$%s" n
8593       | StringList n | DeviceList n ->
8594           pr "\\@%s" n
8595   ) (snd style);
8596   pr ");"
8597
8598 (* Generate Python C module. *)
8599 and generate_python_c () =
8600   generate_header CStyle LGPLv2plus;
8601
8602   pr "\
8603 #include <Python.h>
8604
8605 #include <stdio.h>
8606 #include <stdlib.h>
8607 #include <assert.h>
8608
8609 #include \"guestfs.h\"
8610
8611 typedef struct {
8612   PyObject_HEAD
8613   guestfs_h *g;
8614 } Pyguestfs_Object;
8615
8616 static guestfs_h *
8617 get_handle (PyObject *obj)
8618 {
8619   assert (obj);
8620   assert (obj != Py_None);
8621   return ((Pyguestfs_Object *) obj)->g;
8622 }
8623
8624 static PyObject *
8625 put_handle (guestfs_h *g)
8626 {
8627   assert (g);
8628   return
8629     PyCObject_FromVoidPtrAndDesc ((void *) g, (char *) \"guestfs_h\", NULL);
8630 }
8631
8632 /* This list should be freed (but not the strings) after use. */
8633 static char **
8634 get_string_list (PyObject *obj)
8635 {
8636   int i, len;
8637   char **r;
8638
8639   assert (obj);
8640
8641   if (!PyList_Check (obj)) {
8642     PyErr_SetString (PyExc_RuntimeError, \"expecting a list parameter\");
8643     return NULL;
8644   }
8645
8646   len = PyList_Size (obj);
8647   r = malloc (sizeof (char *) * (len+1));
8648   if (r == NULL) {
8649     PyErr_SetString (PyExc_RuntimeError, \"get_string_list: out of memory\");
8650     return NULL;
8651   }
8652
8653   for (i = 0; i < len; ++i)
8654     r[i] = PyString_AsString (PyList_GetItem (obj, i));
8655   r[len] = NULL;
8656
8657   return r;
8658 }
8659
8660 static PyObject *
8661 put_string_list (char * const * const argv)
8662 {
8663   PyObject *list;
8664   int argc, i;
8665
8666   for (argc = 0; argv[argc] != NULL; ++argc)
8667     ;
8668
8669   list = PyList_New (argc);
8670   for (i = 0; i < argc; ++i)
8671     PyList_SetItem (list, i, PyString_FromString (argv[i]));
8672
8673   return list;
8674 }
8675
8676 static PyObject *
8677 put_table (char * const * const argv)
8678 {
8679   PyObject *list, *item;
8680   int argc, i;
8681
8682   for (argc = 0; argv[argc] != NULL; ++argc)
8683     ;
8684
8685   list = PyList_New (argc >> 1);
8686   for (i = 0; i < argc; i += 2) {
8687     item = PyTuple_New (2);
8688     PyTuple_SetItem (item, 0, PyString_FromString (argv[i]));
8689     PyTuple_SetItem (item, 1, PyString_FromString (argv[i+1]));
8690     PyList_SetItem (list, i >> 1, item);
8691   }
8692
8693   return list;
8694 }
8695
8696 static void
8697 free_strings (char **argv)
8698 {
8699   int argc;
8700
8701   for (argc = 0; argv[argc] != NULL; ++argc)
8702     free (argv[argc]);
8703   free (argv);
8704 }
8705
8706 static PyObject *
8707 py_guestfs_create (PyObject *self, PyObject *args)
8708 {
8709   guestfs_h *g;
8710
8711   g = guestfs_create ();
8712   if (g == NULL) {
8713     PyErr_SetString (PyExc_RuntimeError,
8714                      \"guestfs.create: failed to allocate handle\");
8715     return NULL;
8716   }
8717   guestfs_set_error_handler (g, NULL, NULL);
8718   return put_handle (g);
8719 }
8720
8721 static PyObject *
8722 py_guestfs_close (PyObject *self, PyObject *args)
8723 {
8724   PyObject *py_g;
8725   guestfs_h *g;
8726
8727   if (!PyArg_ParseTuple (args, (char *) \"O:guestfs_close\", &py_g))
8728     return NULL;
8729   g = get_handle (py_g);
8730
8731   guestfs_close (g);
8732
8733   Py_INCREF (Py_None);
8734   return Py_None;
8735 }
8736
8737 ";
8738
8739   let emit_put_list_function typ =
8740     pr "static PyObject *\n";
8741     pr "put_%s_list (struct guestfs_%s_list *%ss)\n" typ typ typ;
8742     pr "{\n";
8743     pr "  PyObject *list;\n";
8744     pr "  int i;\n";
8745     pr "\n";
8746     pr "  list = PyList_New (%ss->len);\n" typ;
8747     pr "  for (i = 0; i < %ss->len; ++i)\n" typ;
8748     pr "    PyList_SetItem (list, i, put_%s (&%ss->val[i]));\n" typ typ;
8749     pr "  return list;\n";
8750     pr "};\n";
8751     pr "\n"
8752   in
8753
8754   (* Structures, turned into Python dictionaries. *)
8755   List.iter (
8756     fun (typ, cols) ->
8757       pr "static PyObject *\n";
8758       pr "put_%s (struct guestfs_%s *%s)\n" typ typ typ;
8759       pr "{\n";
8760       pr "  PyObject *dict;\n";
8761       pr "\n";
8762       pr "  dict = PyDict_New ();\n";
8763       List.iter (
8764         function
8765         | name, FString ->
8766             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8767             pr "                        PyString_FromString (%s->%s));\n"
8768               typ name
8769         | name, FBuffer ->
8770             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8771             pr "                        PyString_FromStringAndSize (%s->%s, %s->%s_len));\n"
8772               typ name typ name
8773         | name, FUUID ->
8774             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8775             pr "                        PyString_FromStringAndSize (%s->%s, 32));\n"
8776               typ name
8777         | name, (FBytes|FUInt64) ->
8778             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8779             pr "                        PyLong_FromUnsignedLongLong (%s->%s));\n"
8780               typ name
8781         | name, FInt64 ->
8782             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8783             pr "                        PyLong_FromLongLong (%s->%s));\n"
8784               typ name
8785         | name, FUInt32 ->
8786             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8787             pr "                        PyLong_FromUnsignedLong (%s->%s));\n"
8788               typ name
8789         | name, FInt32 ->
8790             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8791             pr "                        PyLong_FromLong (%s->%s));\n"
8792               typ name
8793         | name, FOptPercent ->
8794             pr "  if (%s->%s >= 0)\n" typ name;
8795             pr "    PyDict_SetItemString (dict, \"%s\",\n" name;
8796             pr "                          PyFloat_FromDouble ((double) %s->%s));\n"
8797               typ name;
8798             pr "  else {\n";
8799             pr "    Py_INCREF (Py_None);\n";
8800             pr "    PyDict_SetItemString (dict, \"%s\", Py_None);\n" name;
8801             pr "  }\n"
8802         | name, FChar ->
8803             pr "  PyDict_SetItemString (dict, \"%s\",\n" name;
8804             pr "                        PyString_FromStringAndSize (&dirent->%s, 1));\n" name
8805       ) cols;
8806       pr "  return dict;\n";
8807       pr "};\n";
8808       pr "\n";
8809
8810   ) structs;
8811
8812   (* Emit a put_TYPE_list function definition only if that function is used. *)
8813   List.iter (
8814     function
8815     | typ, (RStructListOnly | RStructAndList) ->
8816         (* generate the function for typ *)
8817         emit_put_list_function typ
8818     | typ, _ -> () (* empty *)
8819   ) (rstructs_used_by all_functions);
8820
8821   (* Python wrapper functions. *)
8822   List.iter (
8823     fun (name, style, _, _, _, _, _) ->
8824       pr "static PyObject *\n";
8825       pr "py_guestfs_%s (PyObject *self, PyObject *args)\n" name;
8826       pr "{\n";
8827
8828       pr "  PyObject *py_g;\n";
8829       pr "  guestfs_h *g;\n";
8830       pr "  PyObject *py_r;\n";
8831
8832       let error_code =
8833         match fst style with
8834         | RErr | RInt _ | RBool _ -> pr "  int r;\n"; "-1"
8835         | RInt64 _ -> pr "  int64_t r;\n"; "-1"
8836         | RConstString _ | RConstOptString _ ->
8837             pr "  const char *r;\n"; "NULL"
8838         | RString _ -> pr "  char *r;\n"; "NULL"
8839         | RStringList _ | RHashtable _ -> pr "  char **r;\n"; "NULL"
8840         | RStruct (_, typ) -> pr "  struct guestfs_%s *r;\n" typ; "NULL"
8841         | RStructList (_, typ) ->
8842             pr "  struct guestfs_%s_list *r;\n" typ; "NULL"
8843         | RBufferOut _ ->
8844             pr "  char *r;\n";
8845             pr "  size_t size;\n";
8846             "NULL" in
8847
8848       List.iter (
8849         function
8850         | Pathname n | Device n | Dev_or_Path n | String n | FileIn n | FileOut n ->
8851             pr "  const char *%s;\n" n
8852         | OptString n -> pr "  const char *%s;\n" n
8853         | StringList n | DeviceList n ->
8854             pr "  PyObject *py_%s;\n" n;
8855             pr "  char **%s;\n" n
8856         | Bool n -> pr "  int %s;\n" n
8857         | Int n -> pr "  int %s;\n" n
8858         | Int64 n -> pr "  long long %s;\n" n
8859       ) (snd style);
8860
8861       pr "\n";
8862
8863       (* Convert the parameters. *)
8864       pr "  if (!PyArg_ParseTuple (args, (char *) \"O";
8865       List.iter (
8866         function
8867         | Pathname _ | Device _ | Dev_or_Path _ | String _ | FileIn _ | FileOut _ -> pr "s"
8868         | OptString _ -> pr "z"
8869         | StringList _ | DeviceList _ -> pr "O"
8870         | Bool _ -> pr "i" (* XXX Python has booleans? *)
8871         | Int _ -> pr "i"
8872         | Int64 _ -> pr "L" (* XXX Whoever thought it was a good idea to
8873                              * emulate C's int/long/long long in Python?
8874                              *)
8875       ) (snd style);
8876       pr ":guestfs_%s\",\n" name;
8877       pr "                         &py_g";
8878       List.iter (
8879         function
8880         | Pathname n | Device n | Dev_or_Path n | String n | FileIn n | FileOut n -> pr ", &%s" n
8881         | OptString n -> pr ", &%s" n
8882         | StringList n | DeviceList n -> pr ", &py_%s" n
8883         | Bool n -> pr ", &%s" n
8884         | Int n -> pr ", &%s" n
8885         | Int64 n -> pr ", &%s" n
8886       ) (snd style);
8887
8888       pr "))\n";
8889       pr "    return NULL;\n";
8890
8891       pr "  g = get_handle (py_g);\n";
8892       List.iter (
8893         function
8894         | Pathname _ | Device _ | Dev_or_Path _ | String _
8895         | FileIn _ | FileOut _ | OptString _ | Bool _ | Int _ | Int64 _ -> ()
8896         | StringList n | DeviceList n ->
8897             pr "  %s = get_string_list (py_%s);\n" n n;
8898             pr "  if (!%s) return NULL;\n" n
8899       ) (snd style);
8900
8901       pr "\n";
8902
8903       pr "  r = guestfs_%s " name;
8904       generate_c_call_args ~handle:"g" style;
8905       pr ";\n";
8906
8907       List.iter (
8908         function
8909         | Pathname _ | Device _ | Dev_or_Path _ | String _
8910         | FileIn _ | FileOut _ | OptString _ | Bool _ | Int _ | Int64 _ -> ()
8911         | StringList n | DeviceList n ->
8912             pr "  free (%s);\n" n
8913       ) (snd style);
8914
8915       pr "  if (r == %s) {\n" error_code;
8916       pr "    PyErr_SetString (PyExc_RuntimeError, guestfs_last_error (g));\n";
8917       pr "    return NULL;\n";
8918       pr "  }\n";
8919       pr "\n";
8920
8921       (match fst style with
8922        | RErr ->
8923            pr "  Py_INCREF (Py_None);\n";
8924            pr "  py_r = Py_None;\n"
8925        | RInt _
8926        | RBool _ -> pr "  py_r = PyInt_FromLong ((long) r);\n"
8927        | RInt64 _ -> pr "  py_r = PyLong_FromLongLong (r);\n"
8928        | RConstString _ -> pr "  py_r = PyString_FromString (r);\n"
8929        | RConstOptString _ ->
8930            pr "  if (r)\n";
8931            pr "    py_r = PyString_FromString (r);\n";
8932            pr "  else {\n";
8933            pr "    Py_INCREF (Py_None);\n";
8934            pr "    py_r = Py_None;\n";
8935            pr "  }\n"
8936        | RString _ ->
8937            pr "  py_r = PyString_FromString (r);\n";
8938            pr "  free (r);\n"
8939        | RStringList _ ->
8940            pr "  py_r = put_string_list (r);\n";
8941            pr "  free_strings (r);\n"
8942        | RStruct (_, typ) ->
8943            pr "  py_r = put_%s (r);\n" typ;
8944            pr "  guestfs_free_%s (r);\n" typ
8945        | RStructList (_, typ) ->
8946            pr "  py_r = put_%s_list (r);\n" typ;
8947            pr "  guestfs_free_%s_list (r);\n" typ
8948        | RHashtable n ->
8949            pr "  py_r = put_table (r);\n";
8950            pr "  free_strings (r);\n"
8951        | RBufferOut _ ->
8952            pr "  py_r = PyString_FromStringAndSize (r, size);\n";
8953            pr "  free (r);\n"
8954       );
8955
8956       pr "  return py_r;\n";
8957       pr "}\n";
8958       pr "\n"
8959   ) all_functions;
8960
8961   (* Table of functions. *)
8962   pr "static PyMethodDef methods[] = {\n";
8963   pr "  { (char *) \"create\", py_guestfs_create, METH_VARARGS, NULL },\n";
8964   pr "  { (char *) \"close\", py_guestfs_close, METH_VARARGS, NULL },\n";
8965   List.iter (
8966     fun (name, _, _, _, _, _, _) ->
8967       pr "  { (char *) \"%s\", py_guestfs_%s, METH_VARARGS, NULL },\n"
8968         name name
8969   ) all_functions;
8970   pr "  { NULL, NULL, 0, NULL }\n";
8971   pr "};\n";
8972   pr "\n";
8973
8974   (* Init function. *)
8975   pr "\
8976 void
8977 initlibguestfsmod (void)
8978 {
8979   static int initialized = 0;
8980
8981   if (initialized) return;
8982   Py_InitModule ((char *) \"libguestfsmod\", methods);
8983   initialized = 1;
8984 }
8985 "
8986
8987 (* Generate Python module. *)
8988 and generate_python_py () =
8989   generate_header HashStyle LGPLv2plus;
8990
8991   pr "\
8992 u\"\"\"Python bindings for libguestfs
8993
8994 import guestfs
8995 g = guestfs.GuestFS ()
8996 g.add_drive (\"guest.img\")
8997 g.launch ()
8998 parts = g.list_partitions ()
8999
9000 The guestfs module provides a Python binding to the libguestfs API
9001 for examining and modifying virtual machine disk images.
9002
9003 Amongst the things this is good for: making batch configuration
9004 changes to guests, getting disk used/free statistics (see also:
9005 virt-df), migrating between virtualization systems (see also:
9006 virt-p2v), performing partial backups, performing partial guest
9007 clones, cloning guests and changing registry/UUID/hostname info, and
9008 much else besides.
9009
9010 Libguestfs uses Linux kernel and qemu code, and can access any type of
9011 guest filesystem that Linux and qemu can, including but not limited
9012 to: ext2/3/4, btrfs, FAT and NTFS, LVM, many different disk partition
9013 schemes, qcow, qcow2, vmdk.
9014
9015 Libguestfs provides ways to enumerate guest storage (eg. partitions,
9016 LVs, what filesystem is in each LV, etc.).  It can also run commands
9017 in the context of the guest.  Also you can access filesystems over
9018 FUSE.
9019
9020 Errors which happen while using the API are turned into Python
9021 RuntimeError exceptions.
9022
9023 To create a guestfs handle you usually have to perform the following
9024 sequence of calls:
9025
9026 # Create the handle, call add_drive at least once, and possibly
9027 # several times if the guest has multiple block devices:
9028 g = guestfs.GuestFS ()
9029 g.add_drive (\"guest.img\")
9030
9031 # Launch the qemu subprocess and wait for it to become ready:
9032 g.launch ()
9033
9034 # Now you can issue commands, for example:
9035 logvols = g.lvs ()
9036
9037 \"\"\"
9038
9039 import libguestfsmod
9040
9041 class GuestFS:
9042     \"\"\"Instances of this class are libguestfs API handles.\"\"\"
9043
9044     def __init__ (self):
9045         \"\"\"Create a new libguestfs handle.\"\"\"
9046         self._o = libguestfsmod.create ()
9047
9048     def __del__ (self):
9049         libguestfsmod.close (self._o)
9050
9051 ";
9052
9053   List.iter (
9054     fun (name, style, _, flags, _, _, longdesc) ->
9055       pr "    def %s " name;
9056       generate_py_call_args ~handle:"self" (snd style);
9057       pr ":\n";
9058
9059       if not (List.mem NotInDocs flags) then (
9060         let doc = replace_str longdesc "C<guestfs_" "C<g." in
9061         let doc =
9062           match fst style with
9063           | RErr | RInt _ | RInt64 _ | RBool _
9064           | RConstOptString _ | RConstString _
9065           | RString _ | RBufferOut _ -> doc
9066           | RStringList _ ->
9067               doc ^ "\n\nThis function returns a list of strings."
9068           | RStruct (_, typ) ->
9069               doc ^ sprintf "\n\nThis function returns a dictionary, with keys matching the various fields in the guestfs_%s structure." typ
9070           | RStructList (_, typ) ->
9071               doc ^ sprintf "\n\nThis function returns a list of %ss.  Each %s is represented as a dictionary." typ typ
9072           | RHashtable _ ->
9073               doc ^ "\n\nThis function returns a dictionary." in
9074         let doc =
9075           if List.mem ProtocolLimitWarning flags then
9076             doc ^ "\n\n" ^ protocol_limit_warning
9077           else doc in
9078         let doc =
9079           if List.mem DangerWillRobinson flags then
9080             doc ^ "\n\n" ^ danger_will_robinson
9081           else doc in
9082         let doc =
9083           match deprecation_notice flags with
9084           | None -> doc
9085           | Some txt -> doc ^ "\n\n" ^ txt in
9086         let doc = pod2text ~width:60 name doc in
9087         let doc = List.map (fun line -> replace_str line "\\" "\\\\") doc in
9088         let doc = String.concat "\n        " doc in
9089         pr "        u\"\"\"%s\"\"\"\n" doc;
9090       );
9091       pr "        return libguestfsmod.%s " name;
9092       generate_py_call_args ~handle:"self._o" (snd style);
9093       pr "\n";
9094       pr "\n";
9095   ) all_functions
9096
9097 (* Generate Python call arguments, eg "(handle, foo, bar)" *)
9098 and generate_py_call_args ~handle args =
9099   pr "(%s" handle;
9100   List.iter (fun arg -> pr ", %s" (name_of_argt arg)) args;
9101   pr ")"
9102
9103 (* Useful if you need the longdesc POD text as plain text.  Returns a
9104  * list of lines.
9105  *
9106  * Because this is very slow (the slowest part of autogeneration),
9107  * we memoize the results.
9108  *)
9109 and pod2text ~width name longdesc =
9110   let key = width, name, longdesc in
9111   try Hashtbl.find pod2text_memo key
9112   with Not_found ->
9113     let filename, chan = Filename.open_temp_file "gen" ".tmp" in
9114     fprintf chan "=head1 %s\n\n%s\n" name longdesc;
9115     close_out chan;
9116     let cmd = sprintf "pod2text -w %d %s" width (Filename.quote filename) in
9117     let chan = open_process_in cmd in
9118     let lines = ref [] in
9119     let rec loop i =
9120       let line = input_line chan in
9121       if i = 1 then             (* discard the first line of output *)
9122         loop (i+1)
9123       else (
9124         let line = triml line in
9125         lines := line :: !lines;
9126         loop (i+1)
9127       ) in
9128     let lines = try loop 1 with End_of_file -> List.rev !lines in
9129     unlink filename;
9130     (match close_process_in chan with
9131      | WEXITED 0 -> ()
9132      | WEXITED i ->
9133          failwithf "pod2text: process exited with non-zero status (%d)" i
9134      | WSIGNALED i | WSTOPPED i ->
9135          failwithf "pod2text: process signalled or stopped by signal %d" i
9136     );
9137     Hashtbl.add pod2text_memo key lines;
9138     pod2text_memo_updated ();
9139     lines
9140
9141 (* Generate ruby bindings. *)
9142 and generate_ruby_c () =
9143   generate_header CStyle LGPLv2plus;
9144
9145   pr "\
9146 #include <stdio.h>
9147 #include <stdlib.h>
9148
9149 #include <ruby.h>
9150
9151 #include \"guestfs.h\"
9152
9153 #include \"extconf.h\"
9154
9155 /* For Ruby < 1.9 */
9156 #ifndef RARRAY_LEN
9157 #define RARRAY_LEN(r) (RARRAY((r))->len)
9158 #endif
9159
9160 static VALUE m_guestfs;                 /* guestfs module */
9161 static VALUE c_guestfs;                 /* guestfs_h handle */
9162 static VALUE e_Error;                   /* used for all errors */
9163
9164 static void ruby_guestfs_free (void *p)
9165 {
9166   if (!p) return;
9167   guestfs_close ((guestfs_h *) p);
9168 }
9169
9170 static VALUE ruby_guestfs_create (VALUE m)
9171 {
9172   guestfs_h *g;
9173
9174   g = guestfs_create ();
9175   if (!g)
9176     rb_raise (e_Error, \"failed to create guestfs handle\");
9177
9178   /* Don't print error messages to stderr by default. */
9179   guestfs_set_error_handler (g, NULL, NULL);
9180
9181   /* Wrap it, and make sure the close function is called when the
9182    * handle goes away.
9183    */
9184   return Data_Wrap_Struct (c_guestfs, NULL, ruby_guestfs_free, g);
9185 }
9186
9187 static VALUE ruby_guestfs_close (VALUE gv)
9188 {
9189   guestfs_h *g;
9190   Data_Get_Struct (gv, guestfs_h, g);
9191
9192   ruby_guestfs_free (g);
9193   DATA_PTR (gv) = NULL;
9194
9195   return Qnil;
9196 }
9197
9198 ";
9199
9200   List.iter (
9201     fun (name, style, _, _, _, _, _) ->
9202       pr "static VALUE ruby_guestfs_%s (VALUE gv" name;
9203       List.iter (fun arg -> pr ", VALUE %sv" (name_of_argt arg)) (snd style);
9204       pr ")\n";
9205       pr "{\n";
9206       pr "  guestfs_h *g;\n";
9207       pr "  Data_Get_Struct (gv, guestfs_h, g);\n";
9208       pr "  if (!g)\n";
9209       pr "    rb_raise (rb_eArgError, \"%%s: used handle after closing it\", \"%s\");\n"
9210         name;
9211       pr "\n";
9212
9213       List.iter (
9214         function
9215         | Pathname n | Device n | Dev_or_Path n | String n | FileIn n | FileOut n ->
9216             pr "  Check_Type (%sv, T_STRING);\n" n;
9217             pr "  const char *%s = StringValueCStr (%sv);\n" n n;
9218             pr "  if (!%s)\n" n;
9219             pr "    rb_raise (rb_eTypeError, \"expected string for parameter %%s of %%s\",\n";
9220             pr "              \"%s\", \"%s\");\n" n name
9221         | OptString n ->
9222             pr "  const char *%s = !NIL_P (%sv) ? StringValueCStr (%sv) : NULL;\n" n n n
9223         | StringList n | DeviceList n ->
9224             pr "  char **%s;\n" n;
9225             pr "  Check_Type (%sv, T_ARRAY);\n" n;
9226             pr "  {\n";
9227             pr "    int i, len;\n";
9228             pr "    len = RARRAY_LEN (%sv);\n" n;
9229             pr "    %s = guestfs_safe_malloc (g, sizeof (char *) * (len+1));\n"
9230               n;
9231             pr "    for (i = 0; i < len; ++i) {\n";
9232             pr "      VALUE v = rb_ary_entry (%sv, i);\n" n;
9233             pr "      %s[i] = StringValueCStr (v);\n" n;
9234             pr "    }\n";
9235             pr "    %s[len] = NULL;\n" n;
9236             pr "  }\n";
9237         | Bool n ->
9238             pr "  int %s = RTEST (%sv);\n" n n
9239         | Int n ->
9240             pr "  int %s = NUM2INT (%sv);\n" n n
9241         | Int64 n ->
9242             pr "  long long %s = NUM2LL (%sv);\n" n n
9243       ) (snd style);
9244       pr "\n";
9245
9246       let error_code =
9247         match fst style with
9248         | RErr | RInt _ | RBool _ -> pr "  int r;\n"; "-1"
9249         | RInt64 _ -> pr "  int64_t r;\n"; "-1"
9250         | RConstString _ | RConstOptString _ ->
9251             pr "  const char *r;\n"; "NULL"
9252         | RString _ -> pr "  char *r;\n"; "NULL"
9253         | RStringList _ | RHashtable _ -> pr "  char **r;\n"; "NULL"
9254         | RStruct (_, typ) -> pr "  struct guestfs_%s *r;\n" typ; "NULL"
9255         | RStructList (_, typ) ->
9256             pr "  struct guestfs_%s_list *r;\n" typ; "NULL"
9257         | RBufferOut _ ->
9258             pr "  char *r;\n";
9259             pr "  size_t size;\n";
9260             "NULL" in
9261       pr "\n";
9262
9263       pr "  r = guestfs_%s " name;
9264       generate_c_call_args ~handle:"g" style;
9265       pr ";\n";
9266
9267       List.iter (
9268         function
9269         | Pathname _ | Device _ | Dev_or_Path _ | String _
9270         | FileIn _ | FileOut _ | OptString _ | Bool _ | Int _ | Int64 _ -> ()
9271         | StringList n | DeviceList n ->
9272             pr "  free (%s);\n" n
9273       ) (snd style);
9274
9275       pr "  if (r == %s)\n" error_code;
9276       pr "    rb_raise (e_Error, \"%%s\", guestfs_last_error (g));\n";
9277       pr "\n";
9278
9279       (match fst style with
9280        | RErr ->
9281            pr "  return Qnil;\n"
9282        | RInt _ | RBool _ ->
9283            pr "  return INT2NUM (r);\n"
9284        | RInt64 _ ->
9285            pr "  return ULL2NUM (r);\n"
9286        | RConstString _ ->
9287            pr "  return rb_str_new2 (r);\n";
9288        | RConstOptString _ ->
9289            pr "  if (r)\n";
9290            pr "    return rb_str_new2 (r);\n";
9291            pr "  else\n";
9292            pr "    return Qnil;\n";
9293        | RString _ ->
9294            pr "  VALUE rv = rb_str_new2 (r);\n";
9295            pr "  free (r);\n";
9296            pr "  return rv;\n";
9297        | RStringList _ ->
9298            pr "  int i, len = 0;\n";
9299            pr "  for (i = 0; r[i] != NULL; ++i) len++;\n";
9300            pr "  VALUE rv = rb_ary_new2 (len);\n";
9301            pr "  for (i = 0; r[i] != NULL; ++i) {\n";
9302            pr "    rb_ary_push (rv, rb_str_new2 (r[i]));\n";
9303            pr "    free (r[i]);\n";
9304            pr "  }\n";
9305            pr "  free (r);\n";
9306            pr "  return rv;\n"
9307        | RStruct (_, typ) ->
9308            let cols = cols_of_struct typ in
9309            generate_ruby_struct_code typ cols
9310        | RStructList (_, typ) ->
9311            let cols = cols_of_struct typ in
9312            generate_ruby_struct_list_code typ cols
9313        | RHashtable _ ->
9314            pr "  VALUE rv = rb_hash_new ();\n";
9315            pr "  int i;\n";
9316            pr "  for (i = 0; r[i] != NULL; i+=2) {\n";
9317            pr "    rb_hash_aset (rv, rb_str_new2 (r[i]), rb_str_new2 (r[i+1]));\n";
9318            pr "    free (r[i]);\n";
9319            pr "    free (r[i+1]);\n";
9320            pr "  }\n";
9321            pr "  free (r);\n";
9322            pr "  return rv;\n"
9323        | RBufferOut _ ->
9324            pr "  VALUE rv = rb_str_new (r, size);\n";
9325            pr "  free (r);\n";
9326            pr "  return rv;\n";
9327       );
9328
9329       pr "}\n";
9330       pr "\n"
9331   ) all_functions;
9332
9333   pr "\
9334 /* Initialize the module. */
9335 void Init__guestfs ()
9336 {
9337   m_guestfs = rb_define_module (\"Guestfs\");
9338   c_guestfs = rb_define_class_under (m_guestfs, \"Guestfs\", rb_cObject);
9339   e_Error = rb_define_class_under (m_guestfs, \"Error\", rb_eStandardError);
9340
9341   rb_define_module_function (m_guestfs, \"create\", ruby_guestfs_create, 0);
9342   rb_define_method (c_guestfs, \"close\", ruby_guestfs_close, 0);
9343
9344 ";
9345   (* Define the rest of the methods. *)
9346   List.iter (
9347     fun (name, style, _, _, _, _, _) ->
9348       pr "  rb_define_method (c_guestfs, \"%s\",\n" name;
9349       pr "        ruby_guestfs_%s, %d);\n" name (List.length (snd style))
9350   ) all_functions;
9351
9352   pr "}\n"
9353
9354 (* Ruby code to return a struct. *)
9355 and generate_ruby_struct_code typ cols =
9356   pr "  VALUE rv = rb_hash_new ();\n";
9357   List.iter (
9358     function
9359     | name, FString ->
9360         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), rb_str_new2 (r->%s));\n" name name
9361     | name, FBuffer ->
9362         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), rb_str_new (r->%s, r->%s_len));\n" name name name
9363     | name, FUUID ->
9364         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), rb_str_new (r->%s, 32));\n" name name
9365     | name, (FBytes|FUInt64) ->
9366         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), ULL2NUM (r->%s));\n" name name
9367     | name, FInt64 ->
9368         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), LL2NUM (r->%s));\n" name name
9369     | name, FUInt32 ->
9370         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), UINT2NUM (r->%s));\n" name name
9371     | name, FInt32 ->
9372         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), INT2NUM (r->%s));\n" name name
9373     | name, FOptPercent ->
9374         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), rb_dbl2big (r->%s));\n" name name
9375     | name, FChar -> (* XXX wrong? *)
9376         pr "  rb_hash_aset (rv, rb_str_new2 (\"%s\"), ULL2NUM (r->%s));\n" name name
9377   ) cols;
9378   pr "  guestfs_free_%s (r);\n" typ;
9379   pr "  return rv;\n"
9380
9381 (* Ruby code to return a struct list. *)
9382 and generate_ruby_struct_list_code typ cols =
9383   pr "  VALUE rv = rb_ary_new2 (r->len);\n";
9384   pr "  int i;\n";
9385   pr "  for (i = 0; i < r->len; ++i) {\n";
9386   pr "    VALUE hv = rb_hash_new ();\n";
9387   List.iter (
9388     function
9389     | name, FString ->
9390         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), rb_str_new2 (r->val[i].%s));\n" name name
9391     | name, FBuffer ->
9392         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
9393     | name, FUUID ->
9394         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), rb_str_new (r->val[i].%s, 32));\n" name name
9395     | name, (FBytes|FUInt64) ->
9396         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), ULL2NUM (r->val[i].%s));\n" name name
9397     | name, FInt64 ->
9398         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), LL2NUM (r->val[i].%s));\n" name name
9399     | name, FUInt32 ->
9400         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), UINT2NUM (r->val[i].%s));\n" name name
9401     | name, FInt32 ->
9402         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), INT2NUM (r->val[i].%s));\n" name name
9403     | name, FOptPercent ->
9404         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), rb_dbl2big (r->val[i].%s));\n" name name
9405     | name, FChar -> (* XXX wrong? *)
9406         pr "    rb_hash_aset (hv, rb_str_new2 (\"%s\"), ULL2NUM (r->val[i].%s));\n" name name
9407   ) cols;
9408   pr "    rb_ary_push (rv, hv);\n";
9409   pr "  }\n";
9410   pr "  guestfs_free_%s_list (r);\n" typ;
9411   pr "  return rv;\n"
9412
9413 (* Generate Java bindings GuestFS.java file. *)
9414 and generate_java_java () =
9415   generate_header CStyle LGPLv2plus;
9416
9417   pr "\
9418 package com.redhat.et.libguestfs;
9419
9420 import java.util.HashMap;
9421 import com.redhat.et.libguestfs.LibGuestFSException;
9422 import com.redhat.et.libguestfs.PV;
9423 import com.redhat.et.libguestfs.VG;
9424 import com.redhat.et.libguestfs.LV;
9425 import com.redhat.et.libguestfs.Stat;
9426 import com.redhat.et.libguestfs.StatVFS;
9427 import com.redhat.et.libguestfs.IntBool;
9428 import com.redhat.et.libguestfs.Dirent;
9429
9430 /**
9431  * The GuestFS object is a libguestfs handle.
9432  *
9433  * @author rjones
9434  */
9435 public class GuestFS {
9436   // Load the native code.
9437   static {
9438     System.loadLibrary (\"guestfs_jni\");
9439   }
9440
9441   /**
9442    * The native guestfs_h pointer.
9443    */
9444   long g;
9445
9446   /**
9447    * Create a libguestfs handle.
9448    *
9449    * @throws LibGuestFSException
9450    */
9451   public GuestFS () throws LibGuestFSException
9452   {
9453     g = _create ();
9454   }
9455   private native long _create () throws LibGuestFSException;
9456
9457   /**
9458    * Close a libguestfs handle.
9459    *
9460    * You can also leave handles to be collected by the garbage
9461    * collector, but this method ensures that the resources used
9462    * by the handle are freed up immediately.  If you call any
9463    * other methods after closing the handle, you will get an
9464    * exception.
9465    *
9466    * @throws LibGuestFSException
9467    */
9468   public void close () throws LibGuestFSException
9469   {
9470     if (g != 0)
9471       _close (g);
9472     g = 0;
9473   }
9474   private native void _close (long g) throws LibGuestFSException;
9475
9476   public void finalize () throws LibGuestFSException
9477   {
9478     close ();
9479   }
9480
9481 ";
9482
9483   List.iter (
9484     fun (name, style, _, flags, _, shortdesc, longdesc) ->
9485       if not (List.mem NotInDocs flags); then (
9486         let doc = replace_str longdesc "C<guestfs_" "C<g." in
9487         let doc =
9488           if List.mem ProtocolLimitWarning flags then
9489             doc ^ "\n\n" ^ protocol_limit_warning
9490           else doc in
9491         let doc =
9492           if List.mem DangerWillRobinson flags then
9493             doc ^ "\n\n" ^ danger_will_robinson
9494           else doc in
9495         let doc =
9496           match deprecation_notice flags with
9497           | None -> doc
9498           | Some txt -> doc ^ "\n\n" ^ txt in
9499         let doc = pod2text ~width:60 name doc in
9500         let doc = List.map (            (* RHBZ#501883 *)
9501           function
9502           | "" -> "<p>"
9503           | nonempty -> nonempty
9504         ) doc in
9505         let doc = String.concat "\n   * " doc in
9506
9507         pr "  /**\n";
9508         pr "   * %s\n" shortdesc;
9509         pr "   * <p>\n";
9510         pr "   * %s\n" doc;
9511         pr "   * @throws LibGuestFSException\n";
9512         pr "   */\n";
9513         pr "  ";
9514       );
9515       generate_java_prototype ~public:true ~semicolon:false name style;
9516       pr "\n";
9517       pr "  {\n";
9518       pr "    if (g == 0)\n";
9519       pr "      throw new LibGuestFSException (\"%s: handle is closed\");\n"
9520         name;
9521       pr "    ";
9522       if fst style <> RErr then pr "return ";
9523       pr "_%s " name;
9524       generate_java_call_args ~handle:"g" (snd style);
9525       pr ";\n";
9526       pr "  }\n";
9527       pr "  ";
9528       generate_java_prototype ~privat:true ~native:true name style;
9529       pr "\n";
9530       pr "\n";
9531   ) all_functions;
9532
9533   pr "}\n"
9534
9535 (* Generate Java call arguments, eg "(handle, foo, bar)" *)
9536 and generate_java_call_args ~handle args =
9537   pr "(%s" handle;
9538   List.iter (fun arg -> pr ", %s" (name_of_argt arg)) args;
9539   pr ")"
9540
9541 and generate_java_prototype ?(public=false) ?(privat=false) ?(native=false)
9542     ?(semicolon=true) name style =
9543   if privat then pr "private ";
9544   if public then pr "public ";
9545   if native then pr "native ";
9546
9547   (* return type *)
9548   (match fst style with
9549    | RErr -> pr "void ";
9550    | RInt _ -> pr "int ";
9551    | RInt64 _ -> pr "long ";
9552    | RBool _ -> pr "boolean ";
9553    | RConstString _ | RConstOptString _ | RString _
9554    | RBufferOut _ -> pr "String ";
9555    | RStringList _ -> pr "String[] ";
9556    | RStruct (_, typ) ->
9557        let name = java_name_of_struct typ in
9558        pr "%s " name;
9559    | RStructList (_, typ) ->
9560        let name = java_name_of_struct typ in
9561        pr "%s[] " name;
9562    | RHashtable _ -> pr "HashMap<String,String> ";
9563   );
9564
9565   if native then pr "_%s " name else pr "%s " name;
9566   pr "(";
9567   let needs_comma = ref false in
9568   if native then (
9569     pr "long g";
9570     needs_comma := true
9571   );
9572
9573   (* args *)
9574   List.iter (
9575     fun arg ->
9576       if !needs_comma then pr ", ";
9577       needs_comma := true;
9578
9579       match arg with
9580       | Pathname n
9581       | Device n | Dev_or_Path n
9582       | String n
9583       | OptString n
9584       | FileIn n
9585       | FileOut n ->
9586           pr "String %s" n
9587       | StringList n | DeviceList n ->
9588           pr "String[] %s" n
9589       | Bool n ->
9590           pr "boolean %s" n
9591       | Int n ->
9592           pr "int %s" n
9593       | Int64 n ->
9594           pr "long %s" n
9595   ) (snd style);
9596
9597   pr ")\n";
9598   pr "    throws LibGuestFSException";
9599   if semicolon then pr ";"
9600
9601 and generate_java_struct jtyp cols () =
9602   generate_header CStyle LGPLv2plus;
9603
9604   pr "\
9605 package com.redhat.et.libguestfs;
9606
9607 /**
9608  * Libguestfs %s structure.
9609  *
9610  * @author rjones
9611  * @see GuestFS
9612  */
9613 public class %s {
9614 " jtyp jtyp;
9615
9616   List.iter (
9617     function
9618     | name, FString
9619     | name, FUUID
9620     | name, FBuffer -> pr "  public String %s;\n" name
9621     | name, (FBytes|FUInt64|FInt64) -> pr "  public long %s;\n" name
9622     | name, (FUInt32|FInt32) -> pr "  public int %s;\n" name
9623     | name, FChar -> pr "  public char %s;\n" name
9624     | name, FOptPercent ->
9625         pr "  /* The next field is [0..100] or -1 meaning 'not present': */\n";
9626         pr "  public float %s;\n" name
9627   ) cols;
9628
9629   pr "}\n"
9630
9631 and generate_java_c () =
9632   generate_header CStyle LGPLv2plus;
9633
9634   pr "\
9635 #include <stdio.h>
9636 #include <stdlib.h>
9637 #include <string.h>
9638
9639 #include \"com_redhat_et_libguestfs_GuestFS.h\"
9640 #include \"guestfs.h\"
9641
9642 /* Note that this function returns.  The exception is not thrown
9643  * until after the wrapper function returns.
9644  */
9645 static void
9646 throw_exception (JNIEnv *env, const char *msg)
9647 {
9648   jclass cl;
9649   cl = (*env)->FindClass (env,
9650                           \"com/redhat/et/libguestfs/LibGuestFSException\");
9651   (*env)->ThrowNew (env, cl, msg);
9652 }
9653
9654 JNIEXPORT jlong JNICALL
9655 Java_com_redhat_et_libguestfs_GuestFS__1create
9656   (JNIEnv *env, jobject obj)
9657 {
9658   guestfs_h *g;
9659
9660   g = guestfs_create ();
9661   if (g == NULL) {
9662     throw_exception (env, \"GuestFS.create: failed to allocate handle\");
9663     return 0;
9664   }
9665   guestfs_set_error_handler (g, NULL, NULL);
9666   return (jlong) (long) g;
9667 }
9668
9669 JNIEXPORT void JNICALL
9670 Java_com_redhat_et_libguestfs_GuestFS__1close
9671   (JNIEnv *env, jobject obj, jlong jg)
9672 {
9673   guestfs_h *g = (guestfs_h *) (long) jg;
9674   guestfs_close (g);
9675 }
9676
9677 ";
9678
9679   List.iter (
9680     fun (name, style, _, _, _, _, _) ->
9681       pr "JNIEXPORT ";
9682       (match fst style with
9683        | RErr -> pr "void ";
9684        | RInt _ -> pr "jint ";
9685        | RInt64 _ -> pr "jlong ";
9686        | RBool _ -> pr "jboolean ";
9687        | RConstString _ | RConstOptString _ | RString _
9688        | RBufferOut _ -> pr "jstring ";
9689        | RStruct _ | RHashtable _ ->
9690            pr "jobject ";
9691        | RStringList _ | RStructList _ ->
9692            pr "jobjectArray ";
9693       );
9694       pr "JNICALL\n";
9695       pr "Java_com_redhat_et_libguestfs_GuestFS_";
9696       pr "%s" (replace_str ("_" ^ name) "_" "_1");
9697       pr "\n";
9698       pr "  (JNIEnv *env, jobject obj, jlong jg";
9699       List.iter (
9700         function
9701         | Pathname n
9702         | Device n | Dev_or_Path n
9703         | String n
9704         | OptString n
9705         | FileIn n
9706         | FileOut n ->
9707             pr ", jstring j%s" n
9708         | StringList n | DeviceList n ->
9709             pr ", jobjectArray j%s" n
9710         | Bool n ->
9711             pr ", jboolean j%s" n
9712         | Int n ->
9713             pr ", jint j%s" n
9714         | Int64 n ->
9715             pr ", jlong j%s" n
9716       ) (snd style);
9717       pr ")\n";
9718       pr "{\n";
9719       pr "  guestfs_h *g = (guestfs_h *) (long) jg;\n";
9720       let error_code, no_ret =
9721         match fst style with
9722         | RErr -> pr "  int r;\n"; "-1", ""
9723         | RBool _
9724         | RInt _ -> pr "  int r;\n"; "-1", "0"
9725         | RInt64 _ -> pr "  int64_t r;\n"; "-1", "0"
9726         | RConstString _ -> pr "  const char *r;\n"; "NULL", "NULL"
9727         | RConstOptString _ -> pr "  const char *r;\n"; "NULL", "NULL"
9728         | RString _ ->
9729             pr "  jstring jr;\n";
9730             pr "  char *r;\n"; "NULL", "NULL"
9731         | RStringList _ ->
9732             pr "  jobjectArray jr;\n";
9733             pr "  int r_len;\n";
9734             pr "  jclass cl;\n";
9735             pr "  jstring jstr;\n";
9736             pr "  char **r;\n"; "NULL", "NULL"
9737         | RStruct (_, typ) ->
9738             pr "  jobject jr;\n";
9739             pr "  jclass cl;\n";
9740             pr "  jfieldID fl;\n";
9741             pr "  struct guestfs_%s *r;\n" typ; "NULL", "NULL"
9742         | RStructList (_, typ) ->
9743             pr "  jobjectArray jr;\n";
9744             pr "  jclass cl;\n";
9745             pr "  jfieldID fl;\n";
9746             pr "  jobject jfl;\n";
9747             pr "  struct guestfs_%s_list *r;\n" typ; "NULL", "NULL"
9748         | RHashtable _ -> pr "  char **r;\n"; "NULL", "NULL"
9749         | RBufferOut _ ->
9750             pr "  jstring jr;\n";
9751             pr "  char *r;\n";
9752             pr "  size_t size;\n";
9753             "NULL", "NULL" in
9754       List.iter (
9755         function
9756         | Pathname n
9757         | Device n | Dev_or_Path n
9758         | String n
9759         | OptString n
9760         | FileIn n
9761         | FileOut n ->
9762             pr "  const char *%s;\n" n
9763         | StringList n | DeviceList n ->
9764             pr "  int %s_len;\n" n;
9765             pr "  const char **%s;\n" n
9766         | Bool n
9767         | Int n ->
9768             pr "  int %s;\n" n
9769         | Int64 n ->
9770             pr "  int64_t %s;\n" n
9771       ) (snd style);
9772
9773       let needs_i =
9774         (match fst style with
9775          | RStringList _ | RStructList _ -> true
9776          | RErr | RBool _ | RInt _ | RInt64 _ | RConstString _
9777          | RConstOptString _
9778          | RString _ | RBufferOut _ | RStruct _ | RHashtable _ -> false) ||
9779           List.exists (function
9780                        | StringList _ -> true
9781                        | DeviceList _ -> true
9782                        | _ -> false) (snd style) in
9783       if needs_i then
9784         pr "  int i;\n";
9785
9786       pr "\n";
9787
9788       (* Get the parameters. *)
9789       List.iter (
9790         function
9791         | Pathname n
9792         | Device n | Dev_or_Path n
9793         | String n
9794         | FileIn n
9795         | FileOut n ->
9796             pr "  %s = (*env)->GetStringUTFChars (env, j%s, NULL);\n" n n
9797         | OptString n ->
9798             (* This is completely undocumented, but Java null becomes
9799              * a NULL parameter.
9800              *)
9801             pr "  %s = j%s ? (*env)->GetStringUTFChars (env, j%s, NULL) : NULL;\n" n n n
9802         | StringList n | DeviceList n ->
9803             pr "  %s_len = (*env)->GetArrayLength (env, j%s);\n" n n;
9804             pr "  %s = guestfs_safe_malloc (g, sizeof (char *) * (%s_len+1));\n" n n;
9805             pr "  for (i = 0; i < %s_len; ++i) {\n" n;
9806             pr "    jobject o = (*env)->GetObjectArrayElement (env, j%s, i);\n"
9807               n;
9808             pr "    %s[i] = (*env)->GetStringUTFChars (env, o, NULL);\n" n;
9809             pr "  }\n";
9810             pr "  %s[%s_len] = NULL;\n" n n;
9811         | Bool n
9812         | Int n
9813         | Int64 n ->
9814             pr "  %s = j%s;\n" n n
9815       ) (snd style);
9816
9817       (* Make the call. *)
9818       pr "  r = guestfs_%s " name;
9819       generate_c_call_args ~handle:"g" style;
9820       pr ";\n";
9821
9822       (* Release the parameters. *)
9823       List.iter (
9824         function
9825         | Pathname n
9826         | Device n | Dev_or_Path n
9827         | String n
9828         | FileIn n
9829         | FileOut n ->
9830             pr "  (*env)->ReleaseStringUTFChars (env, j%s, %s);\n" n n
9831         | OptString n ->
9832             pr "  if (j%s)\n" n;
9833             pr "    (*env)->ReleaseStringUTFChars (env, j%s, %s);\n" n n
9834         | StringList n | DeviceList n ->
9835             pr "  for (i = 0; i < %s_len; ++i) {\n" n;
9836             pr "    jobject o = (*env)->GetObjectArrayElement (env, j%s, i);\n"
9837               n;
9838             pr "    (*env)->ReleaseStringUTFChars (env, o, %s[i]);\n" n;
9839             pr "  }\n";
9840             pr "  free (%s);\n" n
9841         | Bool n
9842         | Int n
9843         | Int64 n -> ()
9844       ) (snd style);
9845
9846       (* Check for errors. *)
9847       pr "  if (r == %s) {\n" error_code;
9848       pr "    throw_exception (env, guestfs_last_error (g));\n";
9849       pr "    return %s;\n" no_ret;
9850       pr "  }\n";
9851
9852       (* Return value. *)
9853       (match fst style with
9854        | RErr -> ()
9855        | RInt _ -> pr "  return (jint) r;\n"
9856        | RBool _ -> pr "  return (jboolean) r;\n"
9857        | RInt64 _ -> pr "  return (jlong) r;\n"
9858        | RConstString _ -> pr "  return (*env)->NewStringUTF (env, r);\n"
9859        | RConstOptString _ ->
9860            pr "  return (*env)->NewStringUTF (env, r); /* XXX r NULL? */\n"
9861        | RString _ ->
9862            pr "  jr = (*env)->NewStringUTF (env, r);\n";
9863            pr "  free (r);\n";
9864            pr "  return jr;\n"
9865        | RStringList _ ->
9866            pr "  for (r_len = 0; r[r_len] != NULL; ++r_len) ;\n";
9867            pr "  cl = (*env)->FindClass (env, \"java/lang/String\");\n";
9868            pr "  jstr = (*env)->NewStringUTF (env, \"\");\n";
9869            pr "  jr = (*env)->NewObjectArray (env, r_len, cl, jstr);\n";
9870            pr "  for (i = 0; i < r_len; ++i) {\n";
9871            pr "    jstr = (*env)->NewStringUTF (env, r[i]);\n";
9872            pr "    (*env)->SetObjectArrayElement (env, jr, i, jstr);\n";
9873            pr "    free (r[i]);\n";
9874            pr "  }\n";
9875            pr "  free (r);\n";
9876            pr "  return jr;\n"
9877        | RStruct (_, typ) ->
9878            let jtyp = java_name_of_struct typ in
9879            let cols = cols_of_struct typ in
9880            generate_java_struct_return typ jtyp cols
9881        | RStructList (_, typ) ->
9882            let jtyp = java_name_of_struct typ in
9883            let cols = cols_of_struct typ in
9884            generate_java_struct_list_return typ jtyp cols
9885        | RHashtable _ ->
9886            (* XXX *)
9887            pr "  throw_exception (env, \"%s: internal error: please let us know how to make a Java HashMap from JNI bindings!\");\n" name;
9888            pr "  return NULL;\n"
9889        | RBufferOut _ ->
9890            pr "  jr = (*env)->NewStringUTF (env, r); /* XXX size */\n";
9891            pr "  free (r);\n";
9892            pr "  return jr;\n"
9893       );
9894
9895       pr "}\n";
9896       pr "\n"
9897   ) all_functions
9898
9899 and generate_java_struct_return typ jtyp cols =
9900   pr "  cl = (*env)->FindClass (env, \"com/redhat/et/libguestfs/%s\");\n" jtyp;
9901   pr "  jr = (*env)->AllocObject (env, cl);\n";
9902   List.iter (
9903     function
9904     | name, FString ->
9905         pr "  fl = (*env)->GetFieldID (env, cl, \"%s\", \"Ljava/lang/String;\");\n" name;
9906         pr "  (*env)->SetObjectField (env, jr, fl, (*env)->NewStringUTF (env, r->%s));\n" name;
9907     | name, FUUID ->
9908         pr "  {\n";
9909         pr "    char s[33];\n";
9910         pr "    memcpy (s, r->%s, 32);\n" name;
9911         pr "    s[32] = 0;\n";
9912         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"Ljava/lang/String;\");\n" name;
9913         pr "    (*env)->SetObjectField (env, jr, fl, (*env)->NewStringUTF (env, s));\n";
9914         pr "  }\n";
9915     | name, FBuffer ->
9916         pr "  {\n";
9917         pr "    int len = r->%s_len;\n" name;
9918         pr "    char s[len+1];\n";
9919         pr "    memcpy (s, r->%s, len);\n" name;
9920         pr "    s[len] = 0;\n";
9921         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"Ljava/lang/String;\");\n" name;
9922         pr "    (*env)->SetObjectField (env, jr, fl, (*env)->NewStringUTF (env, s));\n";
9923         pr "  }\n";
9924     | name, (FBytes|FUInt64|FInt64) ->
9925         pr "  fl = (*env)->GetFieldID (env, cl, \"%s\", \"J\");\n" name;
9926         pr "  (*env)->SetLongField (env, jr, fl, r->%s);\n" name;
9927     | name, (FUInt32|FInt32) ->
9928         pr "  fl = (*env)->GetFieldID (env, cl, \"%s\", \"I\");\n" name;
9929         pr "  (*env)->SetLongField (env, jr, fl, r->%s);\n" name;
9930     | name, FOptPercent ->
9931         pr "  fl = (*env)->GetFieldID (env, cl, \"%s\", \"F\");\n" name;
9932         pr "  (*env)->SetFloatField (env, jr, fl, r->%s);\n" name;
9933     | name, FChar ->
9934         pr "  fl = (*env)->GetFieldID (env, cl, \"%s\", \"C\");\n" name;
9935         pr "  (*env)->SetLongField (env, jr, fl, r->%s);\n" name;
9936   ) cols;
9937   pr "  free (r);\n";
9938   pr "  return jr;\n"
9939
9940 and generate_java_struct_list_return typ jtyp cols =
9941   pr "  cl = (*env)->FindClass (env, \"com/redhat/et/libguestfs/%s\");\n" jtyp;
9942   pr "  jr = (*env)->NewObjectArray (env, r->len, cl, NULL);\n";
9943   pr "  for (i = 0; i < r->len; ++i) {\n";
9944   pr "    jfl = (*env)->AllocObject (env, cl);\n";
9945   List.iter (
9946     function
9947     | name, FString ->
9948         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"Ljava/lang/String;\");\n" name;
9949         pr "    (*env)->SetObjectField (env, jfl, fl, (*env)->NewStringUTF (env, r->val[i].%s));\n" name;
9950     | name, FUUID ->
9951         pr "    {\n";
9952         pr "      char s[33];\n";
9953         pr "      memcpy (s, r->val[i].%s, 32);\n" name;
9954         pr "      s[32] = 0;\n";
9955         pr "      fl = (*env)->GetFieldID (env, cl, \"%s\", \"Ljava/lang/String;\");\n" name;
9956         pr "      (*env)->SetObjectField (env, jfl, fl, (*env)->NewStringUTF (env, s));\n";
9957         pr "    }\n";
9958     | name, FBuffer ->
9959         pr "    {\n";
9960         pr "      int len = r->val[i].%s_len;\n" name;
9961         pr "      char s[len+1];\n";
9962         pr "      memcpy (s, r->val[i].%s, len);\n" name;
9963         pr "      s[len] = 0;\n";
9964         pr "      fl = (*env)->GetFieldID (env, cl, \"%s\", \"Ljava/lang/String;\");\n" name;
9965         pr "      (*env)->SetObjectField (env, jfl, fl, (*env)->NewStringUTF (env, s));\n";
9966         pr "    }\n";
9967     | name, (FBytes|FUInt64|FInt64) ->
9968         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"J\");\n" name;
9969         pr "    (*env)->SetLongField (env, jfl, fl, r->val[i].%s);\n" name;
9970     | name, (FUInt32|FInt32) ->
9971         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"I\");\n" name;
9972         pr "    (*env)->SetLongField (env, jfl, fl, r->val[i].%s);\n" name;
9973     | name, FOptPercent ->
9974         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"F\");\n" name;
9975         pr "    (*env)->SetFloatField (env, jfl, fl, r->val[i].%s);\n" name;
9976     | name, FChar ->
9977         pr "    fl = (*env)->GetFieldID (env, cl, \"%s\", \"C\");\n" name;
9978         pr "    (*env)->SetLongField (env, jfl, fl, r->val[i].%s);\n" name;
9979   ) cols;
9980   pr "    (*env)->SetObjectArrayElement (env, jfl, i, jfl);\n";
9981   pr "  }\n";
9982   pr "  guestfs_free_%s_list (r);\n" typ;
9983   pr "  return jr;\n"
9984
9985 and generate_java_makefile_inc () =
9986   generate_header HashStyle GPLv2plus;
9987
9988   pr "java_built_sources = \\\n";
9989   List.iter (
9990     fun (typ, jtyp) ->
9991         pr "\tcom/redhat/et/libguestfs/%s.java \\\n" jtyp;
9992   ) java_structs;
9993   pr "\tcom/redhat/et/libguestfs/GuestFS.java\n"
9994
9995 and generate_haskell_hs () =
9996   generate_header HaskellStyle LGPLv2plus;
9997
9998   (* XXX We only know how to generate partial FFI for Haskell
9999    * at the moment.  Please help out!
10000    *)
10001   let can_generate style =
10002     match style with
10003     | RErr, _
10004     | RInt _, _
10005     | RInt64 _, _ -> true
10006     | RBool _, _
10007     | RConstString _, _
10008     | RConstOptString _, _
10009     | RString _, _
10010     | RStringList _, _
10011     | RStruct _, _
10012     | RStructList _, _
10013     | RHashtable _, _
10014     | RBufferOut _, _ -> false in
10015
10016   pr "\
10017 {-# INCLUDE <guestfs.h> #-}
10018 {-# LANGUAGE ForeignFunctionInterface #-}
10019
10020 module Guestfs (
10021   create";
10022
10023   (* List out the names of the actions we want to export. *)
10024   List.iter (
10025     fun (name, style, _, _, _, _, _) ->
10026       if can_generate style then pr ",\n  %s" name
10027   ) all_functions;
10028
10029   pr "
10030   ) where
10031
10032 -- Unfortunately some symbols duplicate ones already present
10033 -- in Prelude.  We don't know which, so we hard-code a list
10034 -- here.
10035 import Prelude hiding (truncate)
10036
10037 import Foreign
10038 import Foreign.C
10039 import Foreign.C.Types
10040 import IO
10041 import Control.Exception
10042 import Data.Typeable
10043
10044 data GuestfsS = GuestfsS            -- represents the opaque C struct
10045 type GuestfsP = Ptr GuestfsS        -- guestfs_h *
10046 type GuestfsH = ForeignPtr GuestfsS -- guestfs_h * with attached finalizer
10047
10048 -- XXX define properly later XXX
10049 data PV = PV
10050 data VG = VG
10051 data LV = LV
10052 data IntBool = IntBool
10053 data Stat = Stat
10054 data StatVFS = StatVFS
10055 data Hashtable = Hashtable
10056
10057 foreign import ccall unsafe \"guestfs_create\" c_create
10058   :: IO GuestfsP
10059 foreign import ccall unsafe \"&guestfs_close\" c_close
10060   :: FunPtr (GuestfsP -> IO ())
10061 foreign import ccall unsafe \"guestfs_set_error_handler\" c_set_error_handler
10062   :: GuestfsP -> Ptr CInt -> Ptr CInt -> IO ()
10063
10064 create :: IO GuestfsH
10065 create = do
10066   p <- c_create
10067   c_set_error_handler p nullPtr nullPtr
10068   h <- newForeignPtr c_close p
10069   return h
10070
10071 foreign import ccall unsafe \"guestfs_last_error\" c_last_error
10072   :: GuestfsP -> IO CString
10073
10074 -- last_error :: GuestfsH -> IO (Maybe String)
10075 -- last_error h = do
10076 --   str <- withForeignPtr h (\\p -> c_last_error p)
10077 --   maybePeek peekCString str
10078
10079 last_error :: GuestfsH -> IO (String)
10080 last_error h = do
10081   str <- withForeignPtr h (\\p -> c_last_error p)
10082   if (str == nullPtr)
10083     then return \"no error\"
10084     else peekCString str
10085
10086 ";
10087
10088   (* Generate wrappers for each foreign function. *)
10089   List.iter (
10090     fun (name, style, _, _, _, _, _) ->
10091       if can_generate style then (
10092         pr "foreign import ccall unsafe \"guestfs_%s\" c_%s\n" name name;
10093         pr "  :: ";
10094         generate_haskell_prototype ~handle:"GuestfsP" style;
10095         pr "\n";
10096         pr "\n";
10097         pr "%s :: " name;
10098         generate_haskell_prototype ~handle:"GuestfsH" ~hs:true style;
10099         pr "\n";
10100         pr "%s %s = do\n" name
10101           (String.concat " " ("h" :: List.map name_of_argt (snd style)));
10102         pr "  r <- ";
10103         (* Convert pointer arguments using with* functions. *)
10104         List.iter (
10105           function
10106           | FileIn n
10107           | FileOut n
10108           | Pathname n | Device n | Dev_or_Path n | String n -> pr "withCString %s $ \\%s -> " n n
10109           | OptString n -> pr "maybeWith withCString %s $ \\%s -> " n n
10110           | StringList n | DeviceList n -> pr "withMany withCString %s $ \\%s -> withArray0 nullPtr %s $ \\%s -> " n n n n
10111           | Bool _ | Int _ | Int64 _ -> ()
10112         ) (snd style);
10113         (* Convert integer arguments. *)
10114         let args =
10115           List.map (
10116             function
10117             | Bool n -> sprintf "(fromBool %s)" n
10118             | Int n -> sprintf "(fromIntegral %s)" n
10119             | Int64 n -> sprintf "(fromIntegral %s)" n
10120             | FileIn n | FileOut n
10121             | Pathname n | Device n | Dev_or_Path n | String n | OptString n | StringList n | DeviceList n -> n
10122           ) (snd style) in
10123         pr "withForeignPtr h (\\p -> c_%s %s)\n" name
10124           (String.concat " " ("p" :: args));
10125         (match fst style with
10126          | RErr | RInt _ | RInt64 _ | RBool _ ->
10127              pr "  if (r == -1)\n";
10128              pr "    then do\n";
10129              pr "      err <- last_error h\n";
10130              pr "      fail err\n";
10131          | RConstString _ | RConstOptString _ | RString _
10132          | RStringList _ | RStruct _
10133          | RStructList _ | RHashtable _ | RBufferOut _ ->
10134              pr "  if (r == nullPtr)\n";
10135              pr "    then do\n";
10136              pr "      err <- last_error h\n";
10137              pr "      fail err\n";
10138         );
10139         (match fst style with
10140          | RErr ->
10141              pr "    else return ()\n"
10142          | RInt _ ->
10143              pr "    else return (fromIntegral r)\n"
10144          | RInt64 _ ->
10145              pr "    else return (fromIntegral r)\n"
10146          | RBool _ ->
10147              pr "    else return (toBool r)\n"
10148          | RConstString _
10149          | RConstOptString _
10150          | RString _
10151          | RStringList _
10152          | RStruct _
10153          | RStructList _
10154          | RHashtable _
10155          | RBufferOut _ ->
10156              pr "    else return ()\n" (* XXXXXXXXXXXXXXXXXXXX *)
10157         );
10158         pr "\n";
10159       )
10160   ) all_functions
10161
10162 and generate_haskell_prototype ~handle ?(hs = false) style =
10163   pr "%s -> " handle;
10164   let string = if hs then "String" else "CString" in
10165   let int = if hs then "Int" else "CInt" in
10166   let bool = if hs then "Bool" else "CInt" in
10167   let int64 = if hs then "Integer" else "Int64" in
10168   List.iter (
10169     fun arg ->
10170       (match arg with
10171        | Pathname _ | Device _ | Dev_or_Path _ | String _ -> pr "%s" string
10172        | OptString _ -> if hs then pr "Maybe String" else pr "CString"
10173        | StringList _ | DeviceList _ -> if hs then pr "[String]" else pr "Ptr CString"
10174        | Bool _ -> pr "%s" bool
10175        | Int _ -> pr "%s" int
10176        | Int64 _ -> pr "%s" int
10177        | FileIn _ -> pr "%s" string
10178        | FileOut _ -> pr "%s" string
10179       );
10180       pr " -> ";
10181   ) (snd style);
10182   pr "IO (";
10183   (match fst style with
10184    | RErr -> if not hs then pr "CInt"
10185    | RInt _ -> pr "%s" int
10186    | RInt64 _ -> pr "%s" int64
10187    | RBool _ -> pr "%s" bool
10188    | RConstString _ -> pr "%s" string
10189    | RConstOptString _ -> pr "Maybe %s" string
10190    | RString _ -> pr "%s" string
10191    | RStringList _ -> pr "[%s]" string
10192    | RStruct (_, typ) ->
10193        let name = java_name_of_struct typ in
10194        pr "%s" name
10195    | RStructList (_, typ) ->
10196        let name = java_name_of_struct typ in
10197        pr "[%s]" name
10198    | RHashtable _ -> pr "Hashtable"
10199    | RBufferOut _ -> pr "%s" string
10200   );
10201   pr ")"
10202
10203 and generate_csharp () =
10204   generate_header CPlusPlusStyle LGPLv2plus;
10205
10206   (* XXX Make this configurable by the C# assembly users. *)
10207   let library = "libguestfs.so.0" in
10208
10209   pr "\
10210 // These C# bindings are highly experimental at present.
10211 //
10212 // Firstly they only work on Linux (ie. Mono).  In order to get them
10213 // to work on Windows (ie. .Net) you would need to port the library
10214 // itself to Windows first.
10215 //
10216 // The second issue is that some calls are known to be incorrect and
10217 // can cause Mono to segfault.  Particularly: calls which pass or
10218 // return string[], or return any structure value.  This is because
10219 // we haven't worked out the correct way to do this from C#.
10220 //
10221 // The third issue is that when compiling you get a lot of warnings.
10222 // We are not sure whether the warnings are important or not.
10223 //
10224 // Fourthly we do not routinely build or test these bindings as part
10225 // of the make && make check cycle, which means that regressions might
10226 // go unnoticed.
10227 //
10228 // Suggestions and patches are welcome.
10229
10230 // To compile:
10231 //
10232 // gmcs Libguestfs.cs
10233 // mono Libguestfs.exe
10234 //
10235 // (You'll probably want to add a Test class / static main function
10236 // otherwise this won't do anything useful).
10237
10238 using System;
10239 using System.IO;
10240 using System.Runtime.InteropServices;
10241 using System.Runtime.Serialization;
10242 using System.Collections;
10243
10244 namespace Guestfs
10245 {
10246   class Error : System.ApplicationException
10247   {
10248     public Error (string message) : base (message) {}
10249     protected Error (SerializationInfo info, StreamingContext context) {}
10250   }
10251
10252   class Guestfs
10253   {
10254     IntPtr _handle;
10255
10256     [DllImport (\"%s\")]
10257     static extern IntPtr guestfs_create ();
10258
10259     public Guestfs ()
10260     {
10261       _handle = guestfs_create ();
10262       if (_handle == IntPtr.Zero)
10263         throw new Error (\"could not create guestfs handle\");
10264     }
10265
10266     [DllImport (\"%s\")]
10267     static extern void guestfs_close (IntPtr h);
10268
10269     ~Guestfs ()
10270     {
10271       guestfs_close (_handle);
10272     }
10273
10274     [DllImport (\"%s\")]
10275     static extern string guestfs_last_error (IntPtr h);
10276
10277 " library library library;
10278
10279   (* Generate C# structure bindings.  We prefix struct names with
10280    * underscore because C# cannot have conflicting struct names and
10281    * method names (eg. "class stat" and "stat").
10282    *)
10283   List.iter (
10284     fun (typ, cols) ->
10285       pr "    [StructLayout (LayoutKind.Sequential)]\n";
10286       pr "    public class _%s {\n" typ;
10287       List.iter (
10288         function
10289         | name, FChar -> pr "      char %s;\n" name
10290         | name, FString -> pr "      string %s;\n" name
10291         | name, FBuffer ->
10292             pr "      uint %s_len;\n" name;
10293             pr "      string %s;\n" name
10294         | name, FUUID ->
10295             pr "      [MarshalAs (UnmanagedType.ByValTStr, SizeConst=16)]\n";
10296             pr "      string %s;\n" name
10297         | name, FUInt32 -> pr "      uint %s;\n" name
10298         | name, FInt32 -> pr "      int %s;\n" name
10299         | name, (FUInt64|FBytes) -> pr "      ulong %s;\n" name
10300         | name, FInt64 -> pr "      long %s;\n" name
10301         | name, FOptPercent -> pr "      float %s; /* [0..100] or -1 */\n" name
10302       ) cols;
10303       pr "    }\n";
10304       pr "\n"
10305   ) structs;
10306
10307   (* Generate C# function bindings. *)
10308   List.iter (
10309     fun (name, style, _, _, _, shortdesc, _) ->
10310       let rec csharp_return_type () =
10311         match fst style with
10312         | RErr -> "void"
10313         | RBool n -> "bool"
10314         | RInt n -> "int"
10315         | RInt64 n -> "long"
10316         | RConstString n
10317         | RConstOptString n
10318         | RString n
10319         | RBufferOut n -> "string"
10320         | RStruct (_,n) -> "_" ^ n
10321         | RHashtable n -> "Hashtable"
10322         | RStringList n -> "string[]"
10323         | RStructList (_,n) -> sprintf "_%s[]" n
10324
10325       and c_return_type () =
10326         match fst style with
10327         | RErr
10328         | RBool _
10329         | RInt _ -> "int"
10330         | RInt64 _ -> "long"
10331         | RConstString _
10332         | RConstOptString _
10333         | RString _
10334         | RBufferOut _ -> "string"
10335         | RStruct (_,n) -> "_" ^ n
10336         | RHashtable _
10337         | RStringList _ -> "string[]"
10338         | RStructList (_,n) -> sprintf "_%s[]" n
10339
10340       and c_error_comparison () =
10341         match fst style with
10342         | RErr
10343         | RBool _
10344         | RInt _
10345         | RInt64 _ -> "== -1"
10346         | RConstString _
10347         | RConstOptString _
10348         | RString _
10349         | RBufferOut _
10350         | RStruct (_,_)
10351         | RHashtable _
10352         | RStringList _
10353         | RStructList (_,_) -> "== null"
10354
10355       and generate_extern_prototype () =
10356         pr "    static extern %s guestfs_%s (IntPtr h"
10357           (c_return_type ()) name;
10358         List.iter (
10359           function
10360           | Pathname n | Device n | Dev_or_Path n | String n | OptString n
10361           | FileIn n | FileOut n ->
10362               pr ", [In] string %s" n
10363           | StringList n | DeviceList n ->
10364               pr ", [In] string[] %s" n
10365           | Bool n ->
10366               pr ", bool %s" n
10367           | Int n ->
10368               pr ", int %s" n
10369           | Int64 n ->
10370               pr ", long %s" n
10371         ) (snd style);
10372         pr ");\n"
10373
10374       and generate_public_prototype () =
10375         pr "    public %s %s (" (csharp_return_type ()) name;
10376         let comma = ref false in
10377         let next () =
10378           if !comma then pr ", ";
10379           comma := true
10380         in
10381         List.iter (
10382           function
10383           | Pathname n | Device n | Dev_or_Path n | String n | OptString n
10384           | FileIn n | FileOut n ->
10385               next (); pr "string %s" n
10386           | StringList n | DeviceList n ->
10387               next (); pr "string[] %s" n
10388           | Bool n ->
10389               next (); pr "bool %s" n
10390           | Int n ->
10391               next (); pr "int %s" n
10392           | Int64 n ->
10393               next (); pr "long %s" n
10394         ) (snd style);
10395         pr ")\n"
10396
10397       and generate_call () =
10398         pr "guestfs_%s (_handle" name;
10399         List.iter (fun arg -> pr ", %s" (name_of_argt arg)) (snd style);
10400         pr ");\n";
10401       in
10402
10403       pr "    [DllImport (\"%s\")]\n" library;
10404       generate_extern_prototype ();
10405       pr "\n";
10406       pr "    /// <summary>\n";
10407       pr "    /// %s\n" shortdesc;
10408       pr "    /// </summary>\n";
10409       generate_public_prototype ();
10410       pr "    {\n";
10411       pr "      %s r;\n" (c_return_type ());
10412       pr "      r = ";
10413       generate_call ();
10414       pr "      if (r %s)\n" (c_error_comparison ());
10415       pr "        throw new Error (guestfs_last_error (_handle));\n";
10416       (match fst style with
10417        | RErr -> ()
10418        | RBool _ ->
10419            pr "      return r != 0 ? true : false;\n"
10420        | RHashtable _ ->
10421            pr "      Hashtable rr = new Hashtable ();\n";
10422            pr "      for (int i = 0; i < r.Length; i += 2)\n";
10423            pr "        rr.Add (r[i], r[i+1]);\n";
10424            pr "      return rr;\n"
10425        | RInt _ | RInt64 _ | RConstString _ | RConstOptString _
10426        | RString _ | RBufferOut _ | RStruct _ | RStringList _
10427        | RStructList _ ->
10428            pr "      return r;\n"
10429       );
10430       pr "    }\n";
10431       pr "\n";
10432   ) all_functions_sorted;
10433
10434   pr "  }
10435 }
10436 "
10437
10438 and generate_bindtests () =
10439   generate_header CStyle LGPLv2plus;
10440
10441   pr "\
10442 #include <stdio.h>
10443 #include <stdlib.h>
10444 #include <inttypes.h>
10445 #include <string.h>
10446
10447 #include \"guestfs.h\"
10448 #include \"guestfs-internal.h\"
10449 #include \"guestfs-internal-actions.h\"
10450 #include \"guestfs_protocol.h\"
10451
10452 #define error guestfs_error
10453 #define safe_calloc guestfs_safe_calloc
10454 #define safe_malloc guestfs_safe_malloc
10455
10456 static void
10457 print_strings (char *const *argv)
10458 {
10459   int argc;
10460
10461   printf (\"[\");
10462   for (argc = 0; argv[argc] != NULL; ++argc) {
10463     if (argc > 0) printf (\", \");
10464     printf (\"\\\"%%s\\\"\", argv[argc]);
10465   }
10466   printf (\"]\\n\");
10467 }
10468
10469 /* The test0 function prints its parameters to stdout. */
10470 ";
10471
10472   let test0, tests =
10473     match test_functions with
10474     | [] -> assert false
10475     | test0 :: tests -> test0, tests in
10476
10477   let () =
10478     let (name, style, _, _, _, _, _) = test0 in
10479     generate_prototype ~extern:false ~semicolon:false ~newline:true
10480       ~handle:"g" ~prefix:"guestfs__" name style;
10481     pr "{\n";
10482     List.iter (
10483       function
10484       | Pathname n
10485       | Device n | Dev_or_Path n
10486       | String n
10487       | FileIn n
10488       | FileOut n -> pr "  printf (\"%%s\\n\", %s);\n" n
10489       | OptString n -> pr "  printf (\"%%s\\n\", %s ? %s : \"null\");\n" n n
10490       | StringList n | DeviceList n -> pr "  print_strings (%s);\n" n
10491       | Bool n -> pr "  printf (\"%%s\\n\", %s ? \"true\" : \"false\");\n" n
10492       | Int n -> pr "  printf (\"%%d\\n\", %s);\n" n
10493       | Int64 n -> pr "  printf (\"%%\" PRIi64 \"\\n\", %s);\n" n
10494     ) (snd style);
10495     pr "  /* Java changes stdout line buffering so we need this: */\n";
10496     pr "  fflush (stdout);\n";
10497     pr "  return 0;\n";
10498     pr "}\n";
10499     pr "\n" in
10500
10501   List.iter (
10502     fun (name, style, _, _, _, _, _) ->
10503       if String.sub name (String.length name - 3) 3 <> "err" then (
10504         pr "/* Test normal return. */\n";
10505         generate_prototype ~extern:false ~semicolon:false ~newline:true
10506           ~handle:"g" ~prefix:"guestfs__" name style;
10507         pr "{\n";
10508         (match fst style with
10509          | RErr ->
10510              pr "  return 0;\n"
10511          | RInt _ ->
10512              pr "  int r;\n";
10513              pr "  sscanf (val, \"%%d\", &r);\n";
10514              pr "  return r;\n"
10515          | RInt64 _ ->
10516              pr "  int64_t r;\n";
10517              pr "  sscanf (val, \"%%\" SCNi64, &r);\n";
10518              pr "  return r;\n"
10519          | RBool _ ->
10520              pr "  return STREQ (val, \"true\");\n"
10521          | RConstString _
10522          | RConstOptString _ ->
10523              (* Can't return the input string here.  Return a static
10524               * string so we ensure we get a segfault if the caller
10525               * tries to free it.
10526               *)
10527              pr "  return \"static string\";\n"
10528          | RString _ ->
10529              pr "  return strdup (val);\n"
10530          | RStringList _ ->
10531              pr "  char **strs;\n";
10532              pr "  int n, i;\n";
10533              pr "  sscanf (val, \"%%d\", &n);\n";
10534              pr "  strs = safe_malloc (g, (n+1) * sizeof (char *));\n";
10535              pr "  for (i = 0; i < n; ++i) {\n";
10536              pr "    strs[i] = safe_malloc (g, 16);\n";
10537              pr "    snprintf (strs[i], 16, \"%%d\", i);\n";
10538              pr "  }\n";
10539              pr "  strs[n] = NULL;\n";
10540              pr "  return strs;\n"
10541          | RStruct (_, typ) ->
10542              pr "  struct guestfs_%s *r;\n" typ;
10543              pr "  r = safe_calloc (g, sizeof *r, 1);\n";
10544              pr "  return r;\n"
10545          | RStructList (_, typ) ->
10546              pr "  struct guestfs_%s_list *r;\n" typ;
10547              pr "  r = safe_calloc (g, sizeof *r, 1);\n";
10548              pr "  sscanf (val, \"%%d\", &r->len);\n";
10549              pr "  r->val = safe_calloc (g, r->len, sizeof *r->val);\n";
10550              pr "  return r;\n"
10551          | RHashtable _ ->
10552              pr "  char **strs;\n";
10553              pr "  int n, i;\n";
10554              pr "  sscanf (val, \"%%d\", &n);\n";
10555              pr "  strs = safe_malloc (g, (n*2+1) * sizeof (*strs));\n";
10556              pr "  for (i = 0; i < n; ++i) {\n";
10557              pr "    strs[i*2] = safe_malloc (g, 16);\n";
10558              pr "    strs[i*2+1] = safe_malloc (g, 16);\n";
10559              pr "    snprintf (strs[i*2], 16, \"%%d\", i);\n";
10560              pr "    snprintf (strs[i*2+1], 16, \"%%d\", i);\n";
10561              pr "  }\n";
10562              pr "  strs[n*2] = NULL;\n";
10563              pr "  return strs;\n"
10564          | RBufferOut _ ->
10565              pr "  return strdup (val);\n"
10566         );
10567         pr "}\n";
10568         pr "\n"
10569       ) else (
10570         pr "/* Test error return. */\n";
10571         generate_prototype ~extern:false ~semicolon:false ~newline:true
10572           ~handle:"g" ~prefix:"guestfs__" name style;
10573         pr "{\n";
10574         pr "  error (g, \"error\");\n";
10575         (match fst style with
10576          | RErr | RInt _ | RInt64 _ | RBool _ ->
10577              pr "  return -1;\n"
10578          | RConstString _ | RConstOptString _
10579          | RString _ | RStringList _ | RStruct _
10580          | RStructList _
10581          | RHashtable _
10582          | RBufferOut _ ->
10583              pr "  return NULL;\n"
10584         );
10585         pr "}\n";
10586         pr "\n"
10587       )
10588   ) tests
10589
10590 and generate_ocaml_bindtests () =
10591   generate_header OCamlStyle GPLv2plus;
10592
10593   pr "\
10594 let () =
10595   let g = Guestfs.create () in
10596 ";
10597
10598   let mkargs args =
10599     String.concat " " (
10600       List.map (
10601         function
10602         | CallString s -> "\"" ^ s ^ "\""
10603         | CallOptString None -> "None"
10604         | CallOptString (Some s) -> sprintf "(Some \"%s\")" s
10605         | CallStringList xs ->
10606             "[|" ^ String.concat ";" (List.map (sprintf "\"%s\"") xs) ^ "|]"
10607         | CallInt i when i >= 0 -> string_of_int i
10608         | CallInt i (* when i < 0 *) -> "(" ^ string_of_int i ^ ")"
10609         | CallInt64 i when i >= 0L -> Int64.to_string i ^ "L"
10610         | CallInt64 i (* when i < 0L *) -> "(" ^ Int64.to_string i ^ "L)"
10611         | CallBool b -> string_of_bool b
10612       ) args
10613     )
10614   in
10615
10616   generate_lang_bindtests (
10617     fun f args -> pr "  Guestfs.%s g %s;\n" f (mkargs args)
10618   );
10619
10620   pr "print_endline \"EOF\"\n"
10621
10622 and generate_perl_bindtests () =
10623   pr "#!/usr/bin/perl -w\n";
10624   generate_header HashStyle GPLv2plus;
10625
10626   pr "\
10627 use strict;
10628
10629 use Sys::Guestfs;
10630
10631 my $g = Sys::Guestfs->new ();
10632 ";
10633
10634   let mkargs args =
10635     String.concat ", " (
10636       List.map (
10637         function
10638         | CallString s -> "\"" ^ s ^ "\""
10639         | CallOptString None -> "undef"
10640         | CallOptString (Some s) -> sprintf "\"%s\"" s
10641         | CallStringList xs ->
10642             "[" ^ String.concat "," (List.map (sprintf "\"%s\"") xs) ^ "]"
10643         | CallInt i -> string_of_int i
10644         | CallInt64 i -> Int64.to_string i
10645         | CallBool b -> if b then "1" else "0"
10646       ) args
10647     )
10648   in
10649
10650   generate_lang_bindtests (
10651     fun f args -> pr "$g->%s (%s);\n" f (mkargs args)
10652   );
10653
10654   pr "print \"EOF\\n\"\n"
10655
10656 and generate_python_bindtests () =
10657   generate_header HashStyle GPLv2plus;
10658
10659   pr "\
10660 import guestfs
10661
10662 g = guestfs.GuestFS ()
10663 ";
10664
10665   let mkargs args =
10666     String.concat ", " (
10667       List.map (
10668         function
10669         | CallString s -> "\"" ^ s ^ "\""
10670         | CallOptString None -> "None"
10671         | CallOptString (Some s) -> sprintf "\"%s\"" s
10672         | CallStringList xs ->
10673             "[" ^ String.concat "," (List.map (sprintf "\"%s\"") xs) ^ "]"
10674         | CallInt i -> string_of_int i
10675         | CallInt64 i -> Int64.to_string i
10676         | CallBool b -> if b then "1" else "0"
10677       ) args
10678     )
10679   in
10680
10681   generate_lang_bindtests (
10682     fun f args -> pr "g.%s (%s)\n" f (mkargs args)
10683   );
10684
10685   pr "print \"EOF\"\n"
10686
10687 and generate_ruby_bindtests () =
10688   generate_header HashStyle GPLv2plus;
10689
10690   pr "\
10691 require 'guestfs'
10692
10693 g = Guestfs::create()
10694 ";
10695
10696   let mkargs args =
10697     String.concat ", " (
10698       List.map (
10699         function
10700         | CallString s -> "\"" ^ s ^ "\""
10701         | CallOptString None -> "nil"
10702         | CallOptString (Some s) -> sprintf "\"%s\"" s
10703         | CallStringList xs ->
10704             "[" ^ String.concat "," (List.map (sprintf "\"%s\"") xs) ^ "]"
10705         | CallInt i -> string_of_int i
10706         | CallInt64 i -> Int64.to_string i
10707         | CallBool b -> string_of_bool b
10708       ) args
10709     )
10710   in
10711
10712   generate_lang_bindtests (
10713     fun f args -> pr "g.%s(%s)\n" f (mkargs args)
10714   );
10715
10716   pr "print \"EOF\\n\"\n"
10717
10718 and generate_java_bindtests () =
10719   generate_header CStyle GPLv2plus;
10720
10721   pr "\
10722 import com.redhat.et.libguestfs.*;
10723
10724 public class Bindtests {
10725     public static void main (String[] argv)
10726     {
10727         try {
10728             GuestFS g = new GuestFS ();
10729 ";
10730
10731   let mkargs args =
10732     String.concat ", " (
10733       List.map (
10734         function
10735         | CallString s -> "\"" ^ s ^ "\""
10736         | CallOptString None -> "null"
10737         | CallOptString (Some s) -> sprintf "\"%s\"" s
10738         | CallStringList xs ->
10739             "new String[]{" ^
10740               String.concat "," (List.map (sprintf "\"%s\"") xs) ^ "}"
10741         | CallInt i -> string_of_int i
10742         | CallInt64 i -> Int64.to_string i
10743         | CallBool b -> string_of_bool b
10744       ) args
10745     )
10746   in
10747
10748   generate_lang_bindtests (
10749     fun f args -> pr "            g.%s (%s);\n" f (mkargs args)
10750   );
10751
10752   pr "
10753             System.out.println (\"EOF\");
10754         }
10755         catch (Exception exn) {
10756             System.err.println (exn);
10757             System.exit (1);
10758         }
10759     }
10760 }
10761 "
10762
10763 and generate_haskell_bindtests () =
10764   generate_header HaskellStyle GPLv2plus;
10765
10766   pr "\
10767 module Bindtests where
10768 import qualified Guestfs
10769
10770 main = do
10771   g <- Guestfs.create
10772 ";
10773
10774   let mkargs args =
10775     String.concat " " (
10776       List.map (
10777         function
10778         | CallString s -> "\"" ^ s ^ "\""
10779         | CallOptString None -> "Nothing"
10780         | CallOptString (Some s) -> sprintf "(Just \"%s\")" s
10781         | CallStringList xs ->
10782             "[" ^ String.concat "," (List.map (sprintf "\"%s\"") xs) ^ "]"
10783         | CallInt i when i < 0 -> "(" ^ string_of_int i ^ ")"
10784         | CallInt i -> string_of_int i
10785         | CallInt64 i when i < 0L -> "(" ^ Int64.to_string i ^ ")"
10786         | CallInt64 i -> Int64.to_string i
10787         | CallBool true -> "True"
10788         | CallBool false -> "False"
10789       ) args
10790     )
10791   in
10792
10793   generate_lang_bindtests (
10794     fun f args -> pr "  Guestfs.%s g %s\n" f (mkargs args)
10795   );
10796
10797   pr "  putStrLn \"EOF\"\n"
10798
10799 (* Language-independent bindings tests - we do it this way to
10800  * ensure there is parity in testing bindings across all languages.
10801  *)
10802 and generate_lang_bindtests call =
10803   call "test0" [CallString "abc"; CallOptString (Some "def");
10804                 CallStringList []; CallBool false;
10805                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10806   call "test0" [CallString "abc"; CallOptString None;
10807                 CallStringList []; CallBool false;
10808                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10809   call "test0" [CallString ""; CallOptString (Some "def");
10810                 CallStringList []; CallBool false;
10811                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10812   call "test0" [CallString ""; CallOptString (Some "");
10813                 CallStringList []; CallBool false;
10814                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10815   call "test0" [CallString "abc"; CallOptString (Some "def");
10816                 CallStringList ["1"]; CallBool false;
10817                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10818   call "test0" [CallString "abc"; CallOptString (Some "def");
10819                 CallStringList ["1"; "2"]; CallBool false;
10820                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10821   call "test0" [CallString "abc"; CallOptString (Some "def");
10822                 CallStringList ["1"]; CallBool true;
10823                 CallInt 0; CallInt64 0L; CallString "123"; CallString "456"];
10824   call "test0" [CallString "abc"; CallOptString (Some "def");
10825                 CallStringList ["1"]; CallBool false;
10826                 CallInt (-1); CallInt64 (-1L); CallString "123"; CallString "456"];
10827   call "test0" [CallString "abc"; CallOptString (Some "def");
10828                 CallStringList ["1"]; CallBool false;
10829                 CallInt (-2); CallInt64 (-2L); CallString "123"; CallString "456"];
10830   call "test0" [CallString "abc"; CallOptString (Some "def");
10831                 CallStringList ["1"]; CallBool false;
10832                 CallInt 1; CallInt64 1L; CallString "123"; CallString "456"];
10833   call "test0" [CallString "abc"; CallOptString (Some "def");
10834                 CallStringList ["1"]; CallBool false;
10835                 CallInt 2; CallInt64 2L; CallString "123"; CallString "456"];
10836   call "test0" [CallString "abc"; CallOptString (Some "def");
10837                 CallStringList ["1"]; CallBool false;
10838                 CallInt 4095; CallInt64 4095L; CallString "123"; CallString "456"];
10839   call "test0" [CallString "abc"; CallOptString (Some "def");
10840                 CallStringList ["1"]; CallBool false;
10841                 CallInt 0; CallInt64 0L; CallString ""; CallString ""]
10842
10843 (* XXX Add here tests of the return and error functions. *)
10844
10845 (* Code to generator bindings for virt-inspector.  Currently only
10846  * implemented for OCaml code (for virt-p2v 2.0).
10847  *)
10848 let rng_input = "inspector/virt-inspector.rng"
10849
10850 (* Read the input file and parse it into internal structures.  This is
10851  * by no means a complete RELAX NG parser, but is just enough to be
10852  * able to parse the specific input file.
10853  *)
10854 type rng =
10855   | Element of string * rng list        (* <element name=name/> *)
10856   | Attribute of string * rng list        (* <attribute name=name/> *)
10857   | Interleave of rng list                (* <interleave/> *)
10858   | ZeroOrMore of rng                        (* <zeroOrMore/> *)
10859   | OneOrMore of rng                        (* <oneOrMore/> *)
10860   | Optional of rng                        (* <optional/> *)
10861   | Choice of string list                (* <choice><value/>*</choice> *)
10862   | Value of string                        (* <value>str</value> *)
10863   | Text                                (* <text/> *)
10864
10865 let rec string_of_rng = function
10866   | Element (name, xs) ->
10867       "Element (\"" ^ name ^ "\", (" ^ string_of_rng_list xs ^ "))"
10868   | Attribute (name, xs) ->
10869       "Attribute (\"" ^ name ^ "\", (" ^ string_of_rng_list xs ^ "))"
10870   | Interleave xs -> "Interleave (" ^ string_of_rng_list xs ^ ")"
10871   | ZeroOrMore rng -> "ZeroOrMore (" ^ string_of_rng rng ^ ")"
10872   | OneOrMore rng -> "OneOrMore (" ^ string_of_rng rng ^ ")"
10873   | Optional rng -> "Optional (" ^ string_of_rng rng ^ ")"
10874   | Choice values -> "Choice [" ^ String.concat ", " values ^ "]"
10875   | Value value -> "Value \"" ^ value ^ "\""
10876   | Text -> "Text"
10877
10878 and string_of_rng_list xs =
10879   String.concat ", " (List.map string_of_rng xs)
10880
10881 let rec parse_rng ?defines context = function
10882   | [] -> []
10883   | Xml.Element ("element", ["name", name], children) :: rest ->
10884       Element (name, parse_rng ?defines context children)
10885       :: parse_rng ?defines context rest
10886   | Xml.Element ("attribute", ["name", name], children) :: rest ->
10887       Attribute (name, parse_rng ?defines context children)
10888       :: parse_rng ?defines context rest
10889   | Xml.Element ("interleave", [], children) :: rest ->
10890       Interleave (parse_rng ?defines context children)
10891       :: parse_rng ?defines context rest
10892   | Xml.Element ("zeroOrMore", [], [child]) :: rest ->
10893       let rng = parse_rng ?defines context [child] in
10894       (match rng with
10895        | [child] -> ZeroOrMore child :: parse_rng ?defines context rest
10896        | _ ->
10897            failwithf "%s: <zeroOrMore> contains more than one child element"
10898              context
10899       )
10900   | Xml.Element ("oneOrMore", [], [child]) :: rest ->
10901       let rng = parse_rng ?defines context [child] in
10902       (match rng with
10903        | [child] -> OneOrMore child :: parse_rng ?defines context rest
10904        | _ ->
10905            failwithf "%s: <oneOrMore> contains more than one child element"
10906              context
10907       )
10908   | Xml.Element ("optional", [], [child]) :: rest ->
10909       let rng = parse_rng ?defines context [child] in
10910       (match rng with
10911        | [child] -> Optional child :: parse_rng ?defines context rest
10912        | _ ->
10913            failwithf "%s: <optional> contains more than one child element"
10914              context
10915       )
10916   | Xml.Element ("choice", [], children) :: rest ->
10917       let values = List.map (
10918         function Xml.Element ("value", [], [Xml.PCData value]) -> value
10919         | _ ->
10920             failwithf "%s: can't handle anything except <value> in <choice>"
10921               context
10922       ) children in
10923       Choice values
10924       :: parse_rng ?defines context rest
10925   | Xml.Element ("value", [], [Xml.PCData value]) :: rest ->
10926       Value value :: parse_rng ?defines context rest
10927   | Xml.Element ("text", [], []) :: rest ->
10928       Text :: parse_rng ?defines context rest
10929   | Xml.Element ("ref", ["name", name], []) :: rest ->
10930       (* Look up the reference.  Because of limitations in this parser,
10931        * we can't handle arbitrarily nested <ref> yet.  You can only
10932        * use <ref> from inside <start>.
10933        *)
10934       (match defines with
10935        | None ->
10936            failwithf "%s: contains <ref>, but no refs are defined yet" context
10937        | Some map ->
10938            let rng = StringMap.find name map in
10939            rng @ parse_rng ?defines context rest
10940       )
10941   | x :: _ ->
10942       failwithf "%s: can't handle '%s' in schema" context (Xml.to_string x)
10943
10944 let grammar =
10945   let xml = Xml.parse_file rng_input in
10946   match xml with
10947   | Xml.Element ("grammar", _,
10948                  Xml.Element ("start", _, gram) :: defines) ->
10949       (* The <define/> elements are referenced in the <start> section,
10950        * so build a map of those first.
10951        *)
10952       let defines = List.fold_left (
10953         fun map ->
10954           function Xml.Element ("define", ["name", name], defn) ->
10955             StringMap.add name defn map
10956           | _ ->
10957               failwithf "%s: expected <define name=name/>" rng_input
10958       ) StringMap.empty defines in
10959       let defines = StringMap.mapi parse_rng defines in
10960
10961       (* Parse the <start> clause, passing the defines. *)
10962       parse_rng ~defines "<start>" gram
10963   | _ ->
10964       failwithf "%s: input is not <grammar><start/><define>*</grammar>"
10965         rng_input
10966
10967 let name_of_field = function
10968   | Element (name, _) | Attribute (name, _)
10969   | ZeroOrMore (Element (name, _))
10970   | OneOrMore (Element (name, _))
10971   | Optional (Element (name, _)) -> name
10972   | Optional (Attribute (name, _)) -> name
10973   | Text -> (* an unnamed field in an element *)
10974       "data"
10975   | rng ->
10976       failwithf "name_of_field failed at: %s" (string_of_rng rng)
10977
10978 (* At the moment this function only generates OCaml types.  However we
10979  * should parameterize it later so it can generate types/structs in a
10980  * variety of languages.
10981  *)
10982 let generate_types xs =
10983   (* A simple type is one that can be printed out directly, eg.
10984    * "string option".  A complex type is one which has a name and has
10985    * to be defined via another toplevel definition, eg. a struct.
10986    *
10987    * generate_type generates code for either simple or complex types.
10988    * In the simple case, it returns the string ("string option").  In
10989    * the complex case, it returns the name ("mountpoint").  In the
10990    * complex case it has to print out the definition before returning,
10991    * so it should only be called when we are at the beginning of a
10992    * new line (BOL context).
10993    *)
10994   let rec generate_type = function
10995     | Text ->                                (* string *)
10996         "string", true
10997     | Choice values ->                        (* [`val1|`val2|...] *)
10998         "[" ^ String.concat "|" (List.map ((^)"`") values) ^ "]", true
10999     | ZeroOrMore rng ->                        (* <rng> list *)
11000         let t, is_simple = generate_type rng in
11001         t ^ " list (* 0 or more *)", is_simple
11002     | OneOrMore rng ->                        (* <rng> list *)
11003         let t, is_simple = generate_type rng in
11004         t ^ " list (* 1 or more *)", is_simple
11005                                         (* virt-inspector hack: bool *)
11006     | Optional (Attribute (name, [Value "1"])) ->
11007         "bool", true
11008     | Optional rng ->                        (* <rng> list *)
11009         let t, is_simple = generate_type rng in
11010         t ^ " option", is_simple
11011                                         (* type name = { fields ... } *)
11012     | Element (name, fields) when is_attrs_interleave fields ->
11013         generate_type_struct name (get_attrs_interleave fields)
11014     | Element (name, [field])                (* type name = field *)
11015     | Attribute (name, [field]) ->
11016         let t, is_simple = generate_type field in
11017         if is_simple then (t, true)
11018         else (
11019           pr "type %s = %s\n" name t;
11020           name, false
11021         )
11022     | Element (name, fields) ->              (* type name = { fields ... } *)
11023         generate_type_struct name fields
11024     | rng ->
11025         failwithf "generate_type failed at: %s" (string_of_rng rng)
11026
11027   and is_attrs_interleave = function
11028     | [Interleave _] -> true
11029     | Attribute _ :: fields -> is_attrs_interleave fields
11030     | Optional (Attribute _) :: fields -> is_attrs_interleave fields
11031     | _ -> false
11032
11033   and get_attrs_interleave = function
11034     | [Interleave fields] -> fields
11035     | ((Attribute _) as field) :: fields
11036     | ((Optional (Attribute _)) as field) :: fields ->
11037         field :: get_attrs_interleave fields
11038     | _ -> assert false
11039
11040   and generate_types xs =
11041     List.iter (fun x -> ignore (generate_type x)) xs
11042
11043   and generate_type_struct name fields =
11044     (* Calculate the types of the fields first.  We have to do this
11045      * before printing anything so we are still in BOL context.
11046      *)
11047     let types = List.map fst (List.map generate_type fields) in
11048
11049     (* Special case of a struct containing just a string and another
11050      * field.  Turn it into an assoc list.
11051      *)
11052     match types with
11053     | ["string"; other] ->
11054         let fname1, fname2 =
11055           match fields with
11056           | [f1; f2] -> name_of_field f1, name_of_field f2
11057           | _ -> assert false in
11058         pr "type %s = string * %s (* %s -> %s *)\n" name other fname1 fname2;
11059         name, false
11060
11061     | types ->
11062         pr "type %s = {\n" name;
11063         List.iter (
11064           fun (field, ftype) ->
11065             let fname = name_of_field field in
11066             pr "  %s_%s : %s;\n" name fname ftype
11067         ) (List.combine fields types);
11068         pr "}\n";
11069         (* Return the name of this type, and
11070          * false because it's not a simple type.
11071          *)
11072         name, false
11073   in
11074
11075   generate_types xs
11076
11077 let generate_parsers xs =
11078   (* As for generate_type above, generate_parser makes a parser for
11079    * some type, and returns the name of the parser it has generated.
11080    * Because it (may) need to print something, it should always be
11081    * called in BOL context.
11082    *)
11083   let rec generate_parser = function
11084     | Text ->                                (* string *)
11085         "string_child_or_empty"
11086     | Choice values ->                        (* [`val1|`val2|...] *)
11087         sprintf "(fun x -> match Xml.pcdata (first_child x) with %s | str -> failwith (\"unexpected field value: \" ^ str))"
11088           (String.concat "|"
11089              (List.map (fun v -> sprintf "%S -> `%s" v v) values))
11090     | ZeroOrMore rng ->                        (* <rng> list *)
11091         let pa = generate_parser rng in
11092         sprintf "(fun x -> List.map %s (Xml.children x))" pa
11093     | OneOrMore rng ->                        (* <rng> list *)
11094         let pa = generate_parser rng in
11095         sprintf "(fun x -> List.map %s (Xml.children x))" pa
11096                                         (* virt-inspector hack: bool *)
11097     | Optional (Attribute (name, [Value "1"])) ->
11098         sprintf "(fun x -> try ignore (Xml.attrib x %S); true with Xml.No_attribute _ -> false)" name
11099     | Optional rng ->                        (* <rng> list *)
11100         let pa = generate_parser rng in
11101         sprintf "(function None -> None | Some x -> Some (%s x))" pa
11102                                         (* type name = { fields ... } *)
11103     | Element (name, fields) when is_attrs_interleave fields ->
11104         generate_parser_struct name (get_attrs_interleave fields)
11105     | Element (name, [field]) ->        (* type name = field *)
11106         let pa = generate_parser field in
11107         let parser_name = sprintf "parse_%s_%d" name (unique ()) in
11108         pr "let %s =\n" parser_name;
11109         pr "  %s\n" pa;
11110         pr "let parse_%s = %s\n" name parser_name;
11111         parser_name
11112     | Attribute (name, [field]) ->
11113         let pa = generate_parser field in
11114         let parser_name = sprintf "parse_%s_%d" name (unique ()) in
11115         pr "let %s =\n" parser_name;
11116         pr "  %s\n" pa;
11117         pr "let parse_%s = %s\n" name parser_name;
11118         parser_name
11119     | Element (name, fields) ->              (* type name = { fields ... } *)
11120         generate_parser_struct name ([], fields)
11121     | rng ->
11122         failwithf "generate_parser failed at: %s" (string_of_rng rng)
11123
11124   and is_attrs_interleave = function
11125     | [Interleave _] -> true
11126     | Attribute _ :: fields -> is_attrs_interleave fields
11127     | Optional (Attribute _) :: fields -> is_attrs_interleave fields
11128     | _ -> false
11129
11130   and get_attrs_interleave = function
11131     | [Interleave fields] -> [], fields
11132     | ((Attribute _) as field) :: fields
11133     | ((Optional (Attribute _)) as field) :: fields ->
11134         let attrs, interleaves = get_attrs_interleave fields in
11135         (field :: attrs), interleaves
11136     | _ -> assert false
11137
11138   and generate_parsers xs =
11139     List.iter (fun x -> ignore (generate_parser x)) xs
11140
11141   and generate_parser_struct name (attrs, interleaves) =
11142     (* Generate parsers for the fields first.  We have to do this
11143      * before printing anything so we are still in BOL context.
11144      *)
11145     let fields = attrs @ interleaves in
11146     let pas = List.map generate_parser fields in
11147
11148     (* Generate an intermediate tuple from all the fields first.
11149      * If the type is just a string + another field, then we will
11150      * return this directly, otherwise it is turned into a record.
11151      *
11152      * RELAX NG note: This code treats <interleave> and plain lists of
11153      * fields the same.  In other words, it doesn't bother enforcing
11154      * any ordering of fields in the XML.
11155      *)
11156     pr "let parse_%s x =\n" name;
11157     pr "  let t = (\n    ";
11158     let comma = ref false in
11159     List.iter (
11160       fun x ->
11161         if !comma then pr ",\n    ";
11162         comma := true;
11163         match x with
11164         | Optional (Attribute (fname, [field])), pa ->
11165             pr "%s x" pa
11166         | Optional (Element (fname, [field])), pa ->
11167             pr "%s (optional_child %S x)" pa fname
11168         | Attribute (fname, [Text]), _ ->
11169             pr "attribute %S x" fname
11170         | (ZeroOrMore _ | OneOrMore _), pa ->
11171             pr "%s x" pa
11172         | Text, pa ->
11173             pr "%s x" pa
11174         | (field, pa) ->
11175             let fname = name_of_field field in
11176             pr "%s (child %S x)" pa fname
11177     ) (List.combine fields pas);
11178     pr "\n  ) in\n";
11179
11180     (match fields with
11181      | [Element (_, [Text]) | Attribute (_, [Text]); _] ->
11182          pr "  t\n"
11183
11184      | _ ->
11185          pr "  (Obj.magic t : %s)\n" name
11186 (*
11187          List.iter (
11188            function
11189            | (Optional (Attribute (fname, [field])), pa) ->
11190                pr "  %s_%s =\n" name fname;
11191                pr "    %s x;\n" pa
11192            | (Optional (Element (fname, [field])), pa) ->
11193                pr "  %s_%s =\n" name fname;
11194                pr "    (let x = optional_child %S x in\n" fname;
11195                pr "     %s x);\n" pa
11196            | (field, pa) ->
11197                let fname = name_of_field field in
11198                pr "  %s_%s =\n" name fname;
11199                pr "    (let x = child %S x in\n" fname;
11200                pr "     %s x);\n" pa
11201          ) (List.combine fields pas);
11202          pr "}\n"
11203 *)
11204     );
11205     sprintf "parse_%s" name
11206   in
11207
11208   generate_parsers xs
11209
11210 (* Generate ocaml/guestfs_inspector.mli. *)
11211 let generate_ocaml_inspector_mli () =
11212   generate_header ~extra_inputs:[rng_input] OCamlStyle LGPLv2plus;
11213
11214   pr "\
11215 (** This is an OCaml language binding to the external [virt-inspector]
11216     program.
11217
11218     For more information, please read the man page [virt-inspector(1)].
11219 *)
11220
11221 ";
11222
11223   generate_types grammar;
11224   pr "(** The nested information returned from the {!inspect} function. *)\n";
11225   pr "\n";
11226
11227   pr "\
11228 val inspect : ?connect:string -> ?xml:string -> string list -> operatingsystems
11229 (** To inspect a libvirt domain called [name], pass a singleton
11230     list: [inspect [name]].  When using libvirt only, you may
11231     optionally pass a libvirt URI using [inspect ~connect:uri ...].
11232
11233     To inspect a disk image or images, pass a list of the filenames
11234     of the disk images: [inspect filenames]
11235
11236     This function inspects the given guest or disk images and
11237     returns a list of operating system(s) found and a large amount
11238     of information about them.  In the vast majority of cases,
11239     a virtual machine only contains a single operating system.
11240
11241     If the optional [~xml] parameter is given, then this function
11242     skips running the external virt-inspector program and just
11243     parses the given XML directly (which is expected to be XML
11244     produced from a previous run of virt-inspector).  The list of
11245     names and connect URI are ignored in this case.
11246
11247     This function can throw a wide variety of exceptions, for example
11248     if the external virt-inspector program cannot be found, or if
11249     it doesn't generate valid XML.
11250 *)
11251 "
11252
11253 (* Generate ocaml/guestfs_inspector.ml. *)
11254 let generate_ocaml_inspector_ml () =
11255   generate_header ~extra_inputs:[rng_input] OCamlStyle LGPLv2plus;
11256
11257   pr "open Unix\n";
11258   pr "\n";
11259
11260   generate_types grammar;
11261   pr "\n";
11262
11263   pr "\
11264 (* Misc functions which are used by the parser code below. *)
11265 let first_child = function
11266   | Xml.Element (_, _, c::_) -> c
11267   | Xml.Element (name, _, []) ->
11268       failwith (\"expected <\" ^ name ^ \"/> to have a child node\")
11269   | Xml.PCData str ->
11270       failwith (\"expected XML tag, but read PCDATA '\" ^ str ^ \"' instead\")
11271
11272 let string_child_or_empty = function
11273   | Xml.Element (_, _, [Xml.PCData s]) -> s
11274   | Xml.Element (_, _, []) -> \"\"
11275   | Xml.Element (x, _, _) ->
11276       failwith (\"expected XML tag with a single PCDATA child, but got \" ^
11277                 x ^ \" instead\")
11278   | Xml.PCData str ->
11279       failwith (\"expected XML tag, but read PCDATA '\" ^ str ^ \"' instead\")
11280
11281 let optional_child name xml =
11282   let children = Xml.children xml in
11283   try
11284     Some (List.find (function
11285                      | Xml.Element (n, _, _) when n = name -> true
11286                      | _ -> false) children)
11287   with
11288     Not_found -> None
11289
11290 let child name xml =
11291   match optional_child name xml with
11292   | Some c -> c
11293   | None ->
11294       failwith (\"mandatory field <\" ^ name ^ \"/> missing in XML output\")
11295
11296 let attribute name xml =
11297   try Xml.attrib xml name
11298   with Xml.No_attribute _ ->
11299     failwith (\"mandatory attribute \" ^ name ^ \" missing in XML output\")
11300
11301 ";
11302
11303   generate_parsers grammar;
11304   pr "\n";
11305
11306   pr "\
11307 (* Run external virt-inspector, then use parser to parse the XML. *)
11308 let inspect ?connect ?xml names =
11309   let xml =
11310     match xml with
11311     | None ->
11312         if names = [] then invalid_arg \"inspect: no names given\";
11313         let cmd = [ \"virt-inspector\"; \"--xml\" ] @
11314           (match connect with None -> [] | Some uri -> [ \"--connect\"; uri ]) @
11315           names in
11316         let cmd = List.map Filename.quote cmd in
11317         let cmd = String.concat \" \" cmd in
11318         let chan = open_process_in cmd in
11319         let xml = Xml.parse_in chan in
11320         (match close_process_in chan with
11321          | WEXITED 0 -> ()
11322          | WEXITED _ -> failwith \"external virt-inspector command failed\"
11323          | WSIGNALED i | WSTOPPED i ->
11324              failwith (\"external virt-inspector command died or stopped on sig \" ^
11325                        string_of_int i)
11326         );
11327         xml
11328     | Some doc ->
11329         Xml.parse_string doc in
11330   parse_operatingsystems xml
11331 "
11332
11333 (* This is used to generate the src/MAX_PROC_NR file which
11334  * contains the maximum procedure number, a surrogate for the
11335  * ABI version number.  See src/Makefile.am for the details.
11336  *)
11337 and generate_max_proc_nr () =
11338   let proc_nrs = List.map (
11339     fun (_, _, proc_nr, _, _, _, _) -> proc_nr
11340   ) daemon_functions in
11341
11342   let max_proc_nr = List.fold_left max 0 proc_nrs in
11343
11344   pr "%d\n" max_proc_nr
11345
11346 let output_to filename k =
11347   let filename_new = filename ^ ".new" in
11348   chan := open_out filename_new;
11349   k ();
11350   close_out !chan;
11351   chan := Pervasives.stdout;
11352
11353   (* Is the new file different from the current file? *)
11354   if Sys.file_exists filename && files_equal filename filename_new then
11355     unlink filename_new                 (* same, so skip it *)
11356   else (
11357     (* different, overwrite old one *)
11358     (try chmod filename 0o644 with Unix_error _ -> ());
11359     rename filename_new filename;
11360     chmod filename 0o444;
11361     printf "written %s\n%!" filename;
11362   )
11363
11364 let perror msg = function
11365   | Unix_error (err, _, _) ->
11366       eprintf "%s: %s\n" msg (error_message err)
11367   | exn ->
11368       eprintf "%s: %s\n" msg (Printexc.to_string exn)
11369
11370 (* Main program. *)
11371 let () =
11372   let lock_fd =
11373     try openfile "HACKING" [O_RDWR] 0
11374     with
11375     | Unix_error (ENOENT, _, _) ->
11376         eprintf "\
11377 You are probably running this from the wrong directory.
11378 Run it from the top source directory using the command
11379   src/generator.ml
11380 ";
11381         exit 1
11382     | exn ->
11383         perror "open: HACKING" exn;
11384         exit 1 in
11385
11386   (* Acquire a lock so parallel builds won't try to run the generator
11387    * twice at the same time.  Subsequent builds will wait for the first
11388    * one to finish.  Note the lock is released implicitly when the
11389    * program exits.
11390    *)
11391   (try lockf lock_fd F_LOCK 1
11392    with exn ->
11393      perror "lock: HACKING" exn;
11394      exit 1);
11395
11396   check_functions ();
11397
11398   output_to "src/guestfs_protocol.x" generate_xdr;
11399   output_to "src/guestfs-structs.h" generate_structs_h;
11400   output_to "src/guestfs-actions.h" generate_actions_h;
11401   output_to "src/guestfs-internal-actions.h" generate_internal_actions_h;
11402   output_to "src/guestfs-actions.c" generate_client_actions;
11403   output_to "src/guestfs-bindtests.c" generate_bindtests;
11404   output_to "src/guestfs-structs.pod" generate_structs_pod;
11405   output_to "src/guestfs-actions.pod" generate_actions_pod;
11406   output_to "src/guestfs-availability.pod" generate_availability_pod;
11407   output_to "src/MAX_PROC_NR" generate_max_proc_nr;
11408   output_to "src/libguestfs.syms" generate_linker_script;
11409   output_to "daemon/actions.h" generate_daemon_actions_h;
11410   output_to "daemon/stubs.c" generate_daemon_actions;
11411   output_to "daemon/names.c" generate_daemon_names;
11412   output_to "daemon/optgroups.c" generate_daemon_optgroups_c;
11413   output_to "daemon/optgroups.h" generate_daemon_optgroups_h;
11414   output_to "capitests/tests.c" generate_tests;
11415   output_to "fish/cmds.c" generate_fish_cmds;
11416   output_to "fish/completion.c" generate_fish_completion;
11417   output_to "fish/guestfish-actions.pod" generate_fish_actions_pod;
11418   output_to "ocaml/guestfs.mli" generate_ocaml_mli;
11419   output_to "ocaml/guestfs.ml" generate_ocaml_ml;
11420   output_to "ocaml/guestfs_c_actions.c" generate_ocaml_c;
11421   output_to "ocaml/bindtests.ml" generate_ocaml_bindtests;
11422   output_to "ocaml/guestfs_inspector.mli" generate_ocaml_inspector_mli;
11423   output_to "ocaml/guestfs_inspector.ml" generate_ocaml_inspector_ml;
11424   output_to "perl/Guestfs.xs" generate_perl_xs;
11425   output_to "perl/lib/Sys/Guestfs.pm" generate_perl_pm;
11426   output_to "perl/bindtests.pl" generate_perl_bindtests;
11427   output_to "python/guestfs-py.c" generate_python_c;
11428   output_to "python/guestfs.py" generate_python_py;
11429   output_to "python/bindtests.py" generate_python_bindtests;
11430   output_to "ruby/ext/guestfs/_guestfs.c" generate_ruby_c;
11431   output_to "ruby/bindtests.rb" generate_ruby_bindtests;
11432   output_to "java/com/redhat/et/libguestfs/GuestFS.java" generate_java_java;
11433
11434   List.iter (
11435     fun (typ, jtyp) ->
11436       let cols = cols_of_struct typ in
11437       let filename = sprintf "java/com/redhat/et/libguestfs/%s.java" jtyp in
11438       output_to filename (generate_java_struct jtyp cols);
11439   ) java_structs;
11440
11441   output_to "java/Makefile.inc" generate_java_makefile_inc;
11442   output_to "java/com_redhat_et_libguestfs_GuestFS.c" generate_java_c;
11443   output_to "java/Bindtests.java" generate_java_bindtests;
11444   output_to "haskell/Guestfs.hs" generate_haskell_hs;
11445   output_to "haskell/Bindtests.hs" generate_haskell_bindtests;
11446   output_to "csharp/Libguestfs.cs" generate_csharp;
11447
11448   (* Always generate this file last, and unconditionally.  It's used
11449    * by the Makefile to know when we must re-run the generator.
11450    *)
11451   let chan = open_out "src/stamp-generator" in
11452   fprintf chan "1\n";
11453   close_out chan;
11454
11455   printf "generated %d lines of code\n" !lines