]>
Commit | Line | Data |
---|---|---|
e3aff4f6 AL |
1 | /* |
2 | * Command line utility to exercise the QEMU I/O path. | |
3 | * | |
4 | * Copyright (C) 2009 Red Hat, Inc. | |
5 | * Copyright (c) 2003-2005 Silicon Graphics, Inc. | |
6 | * | |
7 | * This work is licensed under the terms of the GNU GPL, version 2 or later. | |
8 | * See the COPYING file in the top-level directory. | |
9 | */ | |
c32d766a | 10 | #include <sys/time.h> |
e3aff4f6 AL |
11 | #include <sys/types.h> |
12 | #include <stdarg.h> | |
13 | #include <stdio.h> | |
14 | #include <getopt.h> | |
c32d766a | 15 | #include <libgen.h> |
e3aff4f6 AL |
16 | |
17 | #include "qemu-common.h" | |
1de7afc9 | 18 | #include "qemu/main-loop.h" |
737e150e | 19 | #include "block/block_int.h" |
e3aff4f6 | 20 | #include "cmd.h" |
d7bb72c8 | 21 | #include "trace/control.h" |
e3aff4f6 AL |
22 | |
23 | #define VERSION "0.0.1" | |
24 | ||
43642b38 | 25 | #define CMD_NOFILE_OK 0x01 |
e3aff4f6 AL |
26 | |
27 | char *progname; | |
e3aff4f6 | 28 | |
734c3b85 | 29 | BlockDriverState *qemuio_bs; |
797ac58c | 30 | extern int qemuio_misalign; |
191c2890 | 31 | |
734c3b85 | 32 | static int close_f(BlockDriverState *bs, int argc, char **argv) |
e3aff4f6 | 33 | { |
b4657855 | 34 | bdrv_delete(bs); |
734c3b85 | 35 | qemuio_bs = NULL; |
43642b38 | 36 | return 0; |
e3aff4f6 AL |
37 | } |
38 | ||
39 | static const cmdinfo_t close_cmd = { | |
43642b38 DN |
40 | .name = "close", |
41 | .altname = "c", | |
42 | .cfunc = close_f, | |
43 | .oneline = "close the current open file", | |
e3aff4f6 AL |
44 | }; |
45 | ||
9c4bab26 | 46 | static int openfile(char *name, int flags, int growable) |
e3aff4f6 | 47 | { |
734c3b85 | 48 | if (qemuio_bs) { |
43642b38 DN |
49 | fprintf(stderr, "file open already, try 'help close'\n"); |
50 | return 1; | |
51 | } | |
52 | ||
53 | if (growable) { | |
734c3b85 | 54 | if (bdrv_file_open(&qemuio_bs, name, NULL, flags)) { |
43642b38 DN |
55 | fprintf(stderr, "%s: can't open device %s\n", progname, name); |
56 | return 1; | |
57 | } | |
58 | } else { | |
734c3b85 | 59 | qemuio_bs = bdrv_new("hda"); |
43642b38 | 60 | |
734c3b85 | 61 | if (bdrv_open(qemuio_bs, name, NULL, flags, NULL) < 0) { |
43642b38 | 62 | fprintf(stderr, "%s: can't open device %s\n", progname, name); |
734c3b85 KW |
63 | bdrv_delete(qemuio_bs); |
64 | qemuio_bs = NULL; | |
43642b38 DN |
65 | return 1; |
66 | } | |
67 | } | |
68 | ||
69 | return 0; | |
e3aff4f6 AL |
70 | } |
71 | ||
43642b38 | 72 | static void open_help(void) |
e3aff4f6 | 73 | { |
43642b38 | 74 | printf( |
e3aff4f6 AL |
75 | "\n" |
76 | " opens a new file in the requested mode\n" | |
77 | "\n" | |
78 | " Example:\n" | |
79 | " 'open -Cn /tmp/data' - creates/opens data file read-write and uncached\n" | |
80 | "\n" | |
81 | " Opens a file for subsequent use by all of the other qemu-io commands.\n" | |
e3aff4f6 AL |
82 | " -r, -- open file read-only\n" |
83 | " -s, -- use snapshot file\n" | |
84 | " -n, -- disable host cache\n" | |
9c4bab26 | 85 | " -g, -- allow file to grow (only applies to protocols)" |
e3aff4f6 AL |
86 | "\n"); |
87 | } | |
88 | ||
734c3b85 | 89 | static int open_f(BlockDriverState *bs, int argc, char **argv); |
22a2bdcb BS |
90 | |
91 | static const cmdinfo_t open_cmd = { | |
43642b38 DN |
92 | .name = "open", |
93 | .altname = "o", | |
94 | .cfunc = open_f, | |
95 | .argmin = 1, | |
96 | .argmax = -1, | |
97 | .flags = CMD_NOFILE_OK, | |
98 | .args = "[-Crsn] [path]", | |
99 | .oneline = "open the file specified by path", | |
100 | .help = open_help, | |
22a2bdcb | 101 | }; |
e3aff4f6 | 102 | |
734c3b85 | 103 | static int open_f(BlockDriverState *bs, int argc, char **argv) |
e3aff4f6 | 104 | { |
43642b38 DN |
105 | int flags = 0; |
106 | int readonly = 0; | |
107 | int growable = 0; | |
108 | int c; | |
109 | ||
110 | while ((c = getopt(argc, argv, "snrg")) != EOF) { | |
111 | switch (c) { | |
112 | case 's': | |
113 | flags |= BDRV_O_SNAPSHOT; | |
114 | break; | |
115 | case 'n': | |
116 | flags |= BDRV_O_NOCACHE | BDRV_O_CACHE_WB; | |
117 | break; | |
118 | case 'r': | |
119 | readonly = 1; | |
120 | break; | |
121 | case 'g': | |
122 | growable = 1; | |
123 | break; | |
124 | default: | |
125 | return command_usage(&open_cmd); | |
f5edb014 | 126 | } |
43642b38 DN |
127 | } |
128 | ||
129 | if (!readonly) { | |
130 | flags |= BDRV_O_RDWR; | |
131 | } | |
e3aff4f6 | 132 | |
43642b38 DN |
133 | if (optind != argc - 1) { |
134 | return command_usage(&open_cmd); | |
135 | } | |
e3aff4f6 | 136 | |
43642b38 | 137 | return openfile(argv[optind], flags, growable); |
e3aff4f6 AL |
138 | } |
139 | ||
e681be7e KW |
140 | static int quit_f(BlockDriverState *bs, int argc, char **argv) |
141 | { | |
142 | return 1; | |
143 | } | |
144 | ||
145 | static const cmdinfo_t quit_cmd = { | |
146 | .name = "quit", | |
147 | .altname = "q", | |
148 | .cfunc = quit_f, | |
149 | .argmin = -1, | |
150 | .argmax = -1, | |
151 | .flags = CMD_FLAG_GLOBAL, | |
152 | .oneline = "exit the program", | |
153 | }; | |
154 | ||
e3aff4f6 AL |
155 | static void usage(const char *name) |
156 | { | |
43642b38 | 157 | printf( |
9a2d77ad | 158 | "Usage: %s [-h] [-V] [-rsnm] [-c cmd] ... [file]\n" |
84844a20 | 159 | "QEMU Disk exerciser\n" |
e3aff4f6 | 160 | "\n" |
e3aff4f6 AL |
161 | " -c, --cmd command to execute\n" |
162 | " -r, --read-only export read-only\n" | |
163 | " -s, --snapshot use snapshot file\n" | |
164 | " -n, --nocache disable host cache\n" | |
1db6947d | 165 | " -g, --growable allow file to grow (only applies to protocols)\n" |
e3aff4f6 | 166 | " -m, --misalign misalign allocations for O_DIRECT\n" |
5c6c3a6c | 167 | " -k, --native-aio use kernel AIO implementation (on Linux only)\n" |
592fa070 | 168 | " -t, --cache=MODE use the given cache mode for the image\n" |
d7bb72c8 | 169 | " -T, --trace FILE enable trace events listed in the given file\n" |
e3aff4f6 AL |
170 | " -h, --help display this help and exit\n" |
171 | " -V, --version output version information and exit\n" | |
172 | "\n", | |
43642b38 | 173 | name); |
e3aff4f6 AL |
174 | } |
175 | ||
176 | ||
177 | int main(int argc, char **argv) | |
178 | { | |
43642b38 DN |
179 | int readonly = 0; |
180 | int growable = 0; | |
9e8f1835 | 181 | const char *sopt = "hVc:d:rsnmgkt:T:"; |
43642b38 DN |
182 | const struct option lopt[] = { |
183 | { "help", 0, NULL, 'h' }, | |
184 | { "version", 0, NULL, 'V' }, | |
185 | { "offset", 1, NULL, 'o' }, | |
186 | { "cmd", 1, NULL, 'c' }, | |
187 | { "read-only", 0, NULL, 'r' }, | |
188 | { "snapshot", 0, NULL, 's' }, | |
189 | { "nocache", 0, NULL, 'n' }, | |
190 | { "misalign", 0, NULL, 'm' }, | |
191 | { "growable", 0, NULL, 'g' }, | |
192 | { "native-aio", 0, NULL, 'k' }, | |
9e8f1835 | 193 | { "discard", 1, NULL, 'd' }, |
592fa070 | 194 | { "cache", 1, NULL, 't' }, |
d7bb72c8 | 195 | { "trace", 1, NULL, 'T' }, |
43642b38 DN |
196 | { NULL, 0, NULL, 0 } |
197 | }; | |
198 | int c; | |
199 | int opt_index = 0; | |
9e8f1835 | 200 | int flags = BDRV_O_UNMAP; |
43642b38 DN |
201 | |
202 | progname = basename(argv[0]); | |
203 | ||
204 | while ((c = getopt_long(argc, argv, sopt, lopt, &opt_index)) != -1) { | |
205 | switch (c) { | |
206 | case 's': | |
207 | flags |= BDRV_O_SNAPSHOT; | |
208 | break; | |
209 | case 'n': | |
210 | flags |= BDRV_O_NOCACHE | BDRV_O_CACHE_WB; | |
211 | break; | |
9e8f1835 PB |
212 | case 'd': |
213 | if (bdrv_parse_discard_flags(optarg, &flags) < 0) { | |
214 | error_report("Invalid discard option: %s", optarg); | |
215 | exit(1); | |
216 | } | |
217 | break; | |
43642b38 DN |
218 | case 'c': |
219 | add_user_command(optarg); | |
220 | break; | |
221 | case 'r': | |
222 | readonly = 1; | |
223 | break; | |
224 | case 'm': | |
797ac58c | 225 | qemuio_misalign = 1; |
43642b38 DN |
226 | break; |
227 | case 'g': | |
228 | growable = 1; | |
229 | break; | |
230 | case 'k': | |
231 | flags |= BDRV_O_NATIVE_AIO; | |
232 | break; | |
592fa070 KW |
233 | case 't': |
234 | if (bdrv_parse_cache_flags(optarg, &flags) < 0) { | |
235 | error_report("Invalid cache option: %s", optarg); | |
236 | exit(1); | |
237 | } | |
238 | break; | |
d7bb72c8 SH |
239 | case 'T': |
240 | if (!trace_backend_init(optarg, NULL)) { | |
241 | exit(1); /* error message will have been printed */ | |
242 | } | |
243 | break; | |
43642b38 DN |
244 | case 'V': |
245 | printf("%s version %s\n", progname, VERSION); | |
246 | exit(0); | |
247 | case 'h': | |
248 | usage(progname); | |
249 | exit(0); | |
250 | default: | |
251 | usage(progname); | |
252 | exit(1); | |
f5edb014 | 253 | } |
43642b38 DN |
254 | } |
255 | ||
256 | if ((argc - optind) > 1) { | |
257 | usage(progname); | |
258 | exit(1); | |
259 | } | |
e3aff4f6 | 260 | |
a57d1143 | 261 | qemu_init_main_loop(); |
2592c59a | 262 | bdrv_init(); |
a57d1143 | 263 | |
43642b38 | 264 | /* initialize commands */ |
e681be7e | 265 | add_command(&quit_cmd); |
43642b38 DN |
266 | add_command(&open_cmd); |
267 | add_command(&close_cmd); | |
43642b38 DN |
268 | |
269 | /* open the device */ | |
270 | if (!readonly) { | |
271 | flags |= BDRV_O_RDWR; | |
272 | } | |
273 | ||
274 | if ((argc - optind) == 1) { | |
275 | openfile(argv[optind], flags, growable); | |
276 | } | |
277 | command_loop(); | |
e3aff4f6 | 278 | |
43642b38 | 279 | /* |
922453bc | 280 | * Make sure all outstanding requests complete before the program exits. |
43642b38 | 281 | */ |
922453bc | 282 | bdrv_drain_all(); |
95533d5f | 283 | |
734c3b85 KW |
284 | if (qemuio_bs) { |
285 | bdrv_delete(qemuio_bs); | |
43642b38 DN |
286 | } |
287 | return 0; | |
e3aff4f6 | 288 | } |