2 * Copyright (C) 2018 Red Hat Inc.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions are
8 * * Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
11 * * Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
15 * * Neither the name of Red Hat nor the names of its contributors may be
16 * used to endorse or promote products derived from this software without
17 * specific prior written permission.
19 * THIS SOFTWARE IS PROVIDED BY RED HAT AND CONTRIBUTORS ''AS IS'' AND
20 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
21 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
22 * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL RED HAT OR
23 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
26 * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
27 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
28 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
29 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
48 main (int argc
, char *argv
[])
57 char tmpdir
[] = "/tmp/errorXXXXXX";
58 char error_file
[] = "/tmp/errorXXXXXX/trigger";
59 char error_file_param
[] = "error-file=/tmp/errorXXXXXX/trigger";
61 /* Create temporary directory to store the trigger file. */
62 if (mkdtemp (tmpdir
) == NULL
) {
67 memcpy (error_file
, tmpdir
, strlen (tmpdir
));
68 memcpy (&error_file_param
[11], tmpdir
, strlen (tmpdir
));
70 if (test_start_nbdkit ("--filter", "error",
78 g
= guestfs_create ();
80 perror ("guestfs_create");
84 r
= guestfs_add_drive_opts (g
, "",
85 GUESTFS_ADD_DRIVE_OPTS_FORMAT
, "raw",
86 GUESTFS_ADD_DRIVE_OPTS_PROTOCOL
, "nbd",
87 GUESTFS_ADD_DRIVE_OPTS_SERVER
, server
,
92 if (guestfs_launch (g
) == -1)
95 /* Format the disk with a filesystem. No errors are being injected
96 * yet so we expect this to work.
98 if (guestfs_part_disk (g
, "/dev/sda", "mbr") == -1)
101 if (guestfs_mkfs (g
, "ext2", "/dev/sda1") == -1)
104 if (guestfs_mount (g
, "/dev/sda1", "/") == -1)
107 #define filename "/hello.txt"
108 #define content "hello, people of the world"
110 if (guestfs_write (g
, filename
, content
, strlen (content
)) == -1)
113 /* Try as hard as we can to sync data and kill the libguestfs cache. */
114 if (guestfs_sync (g
) == -1)
116 if (guestfs_drop_caches (g
, 3) == -1)
120 /* Now start injecting EIO errors. */
121 fp
= fopen (error_file
, "w");
128 data
= guestfs_cat (g
, filename
);
132 "expecting Input/output error, but read data!\n",
138 /* Apparently libguestfs doesn't preserve the errno here yet XXX */
139 error
= guestfs_last_errno (g
);
141 fprintf (stderr
, "%s: error: expecting errno = EIO, but got %d\n",
142 program_name
, error
);
147 /* Stop injecting errors, hope that the filesystem recovers. */
150 /* But we'll probably have to remount the filesystem because ext2
151 * will get itself into a "state".
153 if (guestfs_umount (g
, "/") == -1)
155 if (guestfs_mount (g
, "/dev/sda1", "/") == -1)
158 data
= guestfs_cat (g
, filename
);
161 if (strcmp (data
, content
) != 0) {
162 fprintf (stderr
, "%s: error: read unexpected data\n", program_name
);