Statistics
| Branch: | Revision:

root / qemu-io-cmds.c @ dc364f4c

History | View | Annotate | Download (53.3 kB)

1 797ac58c Kevin Wolf
/*
2 797ac58c Kevin Wolf
 * Command line utility to exercise the QEMU I/O path.
3 797ac58c Kevin Wolf
 *
4 797ac58c Kevin Wolf
 * Copyright (C) 2009 Red Hat, Inc.
5 797ac58c Kevin Wolf
 * Copyright (c) 2003-2005 Silicon Graphics, Inc.
6 797ac58c Kevin Wolf
 *
7 797ac58c Kevin Wolf
 * This work is licensed under the terms of the GNU GPL, version 2 or later.
8 797ac58c Kevin Wolf
 * See the COPYING file in the top-level directory.
9 797ac58c Kevin Wolf
 */
10 797ac58c Kevin Wolf
11 3d21994f Kevin Wolf
#include "qemu-io.h"
12 797ac58c Kevin Wolf
#include "block/block_int.h"
13 a8d8ecb7 Max Reitz
#include "block/qapi.h"
14 6a1751b7 Alex Bligh
#include "qemu/main-loop.h"
15 797ac58c Kevin Wolf
16 797ac58c Kevin Wolf
#define CMD_NOFILE_OK   0x01
17 797ac58c Kevin Wolf
18 797ac58c Kevin Wolf
int qemuio_misalign;
19 797ac58c Kevin Wolf
20 c2cdf5c5 Kevin Wolf
static cmdinfo_t *cmdtab;
21 c2cdf5c5 Kevin Wolf
static int ncmds;
22 c2cdf5c5 Kevin Wolf
23 c2cdf5c5 Kevin Wolf
static int compare_cmdname(const void *a, const void *b)
24 c2cdf5c5 Kevin Wolf
{
25 c2cdf5c5 Kevin Wolf
    return strcmp(((const cmdinfo_t *)a)->name,
26 c2cdf5c5 Kevin Wolf
                  ((const cmdinfo_t *)b)->name);
27 c2cdf5c5 Kevin Wolf
}
28 c2cdf5c5 Kevin Wolf
29 c2cdf5c5 Kevin Wolf
void qemuio_add_command(const cmdinfo_t *ci)
30 c2cdf5c5 Kevin Wolf
{
31 c2cdf5c5 Kevin Wolf
    cmdtab = g_realloc(cmdtab, ++ncmds * sizeof(*cmdtab));
32 c2cdf5c5 Kevin Wolf
    cmdtab[ncmds - 1] = *ci;
33 c2cdf5c5 Kevin Wolf
    qsort(cmdtab, ncmds, sizeof(*cmdtab), compare_cmdname);
34 c2cdf5c5 Kevin Wolf
}
35 c2cdf5c5 Kevin Wolf
36 c2cdf5c5 Kevin Wolf
int qemuio_command_usage(const cmdinfo_t *ci)
37 c2cdf5c5 Kevin Wolf
{
38 c2cdf5c5 Kevin Wolf
    printf("%s %s -- %s\n", ci->name, ci->args, ci->oneline);
39 c2cdf5c5 Kevin Wolf
    return 0;
40 c2cdf5c5 Kevin Wolf
}
41 c2cdf5c5 Kevin Wolf
42 c2cdf5c5 Kevin Wolf
static int init_check_command(BlockDriverState *bs, const cmdinfo_t *ct)
43 c2cdf5c5 Kevin Wolf
{
44 c2cdf5c5 Kevin Wolf
    if (ct->flags & CMD_FLAG_GLOBAL) {
45 c2cdf5c5 Kevin Wolf
        return 1;
46 c2cdf5c5 Kevin Wolf
    }
47 c2cdf5c5 Kevin Wolf
    if (!(ct->flags & CMD_NOFILE_OK) && !bs) {
48 c2cdf5c5 Kevin Wolf
        fprintf(stderr, "no file open, try 'help open'\n");
49 c2cdf5c5 Kevin Wolf
        return 0;
50 c2cdf5c5 Kevin Wolf
    }
51 c2cdf5c5 Kevin Wolf
    return 1;
52 c2cdf5c5 Kevin Wolf
}
53 c2cdf5c5 Kevin Wolf
54 3d21994f Kevin Wolf
static int command(BlockDriverState *bs, const cmdinfo_t *ct, int argc,
55 3d21994f Kevin Wolf
                   char **argv)
56 c2cdf5c5 Kevin Wolf
{
57 c2cdf5c5 Kevin Wolf
    char *cmd = argv[0];
58 c2cdf5c5 Kevin Wolf
59 3d21994f Kevin Wolf
    if (!init_check_command(bs, ct)) {
60 c2cdf5c5 Kevin Wolf
        return 0;
61 c2cdf5c5 Kevin Wolf
    }
62 c2cdf5c5 Kevin Wolf
63 c2cdf5c5 Kevin Wolf
    if (argc - 1 < ct->argmin || (ct->argmax != -1 && argc - 1 > ct->argmax)) {
64 c2cdf5c5 Kevin Wolf
        if (ct->argmax == -1) {
65 c2cdf5c5 Kevin Wolf
            fprintf(stderr,
66 c2cdf5c5 Kevin Wolf
                    "bad argument count %d to %s, expected at least %d arguments\n",
67 c2cdf5c5 Kevin Wolf
                    argc-1, cmd, ct->argmin);
68 c2cdf5c5 Kevin Wolf
        } else if (ct->argmin == ct->argmax) {
69 c2cdf5c5 Kevin Wolf
            fprintf(stderr,
70 c2cdf5c5 Kevin Wolf
                    "bad argument count %d to %s, expected %d arguments\n",
71 c2cdf5c5 Kevin Wolf
                    argc-1, cmd, ct->argmin);
72 c2cdf5c5 Kevin Wolf
        } else {
73 c2cdf5c5 Kevin Wolf
            fprintf(stderr,
74 c2cdf5c5 Kevin Wolf
                    "bad argument count %d to %s, expected between %d and %d arguments\n",
75 c2cdf5c5 Kevin Wolf
                    argc-1, cmd, ct->argmin, ct->argmax);
76 c2cdf5c5 Kevin Wolf
        }
77 c2cdf5c5 Kevin Wolf
        return 0;
78 c2cdf5c5 Kevin Wolf
    }
79 c2cdf5c5 Kevin Wolf
    optind = 0;
80 3d21994f Kevin Wolf
    return ct->cfunc(bs, argc, argv);
81 c2cdf5c5 Kevin Wolf
}
82 c2cdf5c5 Kevin Wolf
83 c2cdf5c5 Kevin Wolf
static const cmdinfo_t *find_command(const char *cmd)
84 c2cdf5c5 Kevin Wolf
{
85 c2cdf5c5 Kevin Wolf
    cmdinfo_t *ct;
86 c2cdf5c5 Kevin Wolf
87 c2cdf5c5 Kevin Wolf
    for (ct = cmdtab; ct < &cmdtab[ncmds]; ct++) {
88 c2cdf5c5 Kevin Wolf
        if (strcmp(ct->name, cmd) == 0 ||
89 c2cdf5c5 Kevin Wolf
            (ct->altname && strcmp(ct->altname, cmd) == 0))
90 c2cdf5c5 Kevin Wolf
        {
91 c2cdf5c5 Kevin Wolf
            return (const cmdinfo_t *)ct;
92 c2cdf5c5 Kevin Wolf
        }
93 c2cdf5c5 Kevin Wolf
    }
94 c2cdf5c5 Kevin Wolf
    return NULL;
95 c2cdf5c5 Kevin Wolf
}
96 c2cdf5c5 Kevin Wolf
97 4694020d Stefan Hajnoczi
/* Invoke fn() for commands with a matching prefix */
98 4694020d Stefan Hajnoczi
void qemuio_complete_command(const char *input,
99 4694020d Stefan Hajnoczi
                             void (*fn)(const char *cmd, void *opaque),
100 4694020d Stefan Hajnoczi
                             void *opaque)
101 4694020d Stefan Hajnoczi
{
102 4694020d Stefan Hajnoczi
    cmdinfo_t *ct;
103 4694020d Stefan Hajnoczi
    size_t input_len = strlen(input);
104 4694020d Stefan Hajnoczi
105 4694020d Stefan Hajnoczi
    for (ct = cmdtab; ct < &cmdtab[ncmds]; ct++) {
106 4694020d Stefan Hajnoczi
        if (strncmp(input, ct->name, input_len) == 0) {
107 4694020d Stefan Hajnoczi
            fn(ct->name, opaque);
108 4694020d Stefan Hajnoczi
        }
109 4694020d Stefan Hajnoczi
    }
110 4694020d Stefan Hajnoczi
}
111 4694020d Stefan Hajnoczi
112 c2cdf5c5 Kevin Wolf
static char **breakline(char *input, int *count)
113 c2cdf5c5 Kevin Wolf
{
114 c2cdf5c5 Kevin Wolf
    int c = 0;
115 c2cdf5c5 Kevin Wolf
    char *p;
116 c2cdf5c5 Kevin Wolf
    char **rval = g_malloc0(sizeof(char *));
117 c2cdf5c5 Kevin Wolf
    char **tmp;
118 c2cdf5c5 Kevin Wolf
119 c2cdf5c5 Kevin Wolf
    while (rval && (p = qemu_strsep(&input, " ")) != NULL) {
120 c2cdf5c5 Kevin Wolf
        if (!*p) {
121 c2cdf5c5 Kevin Wolf
            continue;
122 c2cdf5c5 Kevin Wolf
        }
123 c2cdf5c5 Kevin Wolf
        c++;
124 c2cdf5c5 Kevin Wolf
        tmp = g_realloc(rval, sizeof(*rval) * (c + 1));
125 c2cdf5c5 Kevin Wolf
        if (!tmp) {
126 c2cdf5c5 Kevin Wolf
            g_free(rval);
127 c2cdf5c5 Kevin Wolf
            rval = NULL;
128 c2cdf5c5 Kevin Wolf
            c = 0;
129 c2cdf5c5 Kevin Wolf
            break;
130 c2cdf5c5 Kevin Wolf
        } else {
131 c2cdf5c5 Kevin Wolf
            rval = tmp;
132 c2cdf5c5 Kevin Wolf
        }
133 c2cdf5c5 Kevin Wolf
        rval[c - 1] = p;
134 c2cdf5c5 Kevin Wolf
        rval[c] = NULL;
135 c2cdf5c5 Kevin Wolf
    }
136 c2cdf5c5 Kevin Wolf
    *count = c;
137 c2cdf5c5 Kevin Wolf
    return rval;
138 c2cdf5c5 Kevin Wolf
}
139 c2cdf5c5 Kevin Wolf
140 797ac58c Kevin Wolf
static int64_t cvtnum(const char *s)
141 797ac58c Kevin Wolf
{
142 797ac58c Kevin Wolf
    char *end;
143 797ac58c Kevin Wolf
    return strtosz_suffix(s, &end, STRTOSZ_DEFSUFFIX_B);
144 797ac58c Kevin Wolf
}
145 797ac58c Kevin Wolf
146 0b613881 Kevin Wolf
#define EXABYTES(x)     ((long long)(x) << 60)
147 0b613881 Kevin Wolf
#define PETABYTES(x)    ((long long)(x) << 50)
148 0b613881 Kevin Wolf
#define TERABYTES(x)    ((long long)(x) << 40)
149 0b613881 Kevin Wolf
#define GIGABYTES(x)    ((long long)(x) << 30)
150 0b613881 Kevin Wolf
#define MEGABYTES(x)    ((long long)(x) << 20)
151 0b613881 Kevin Wolf
#define KILOBYTES(x)    ((long long)(x) << 10)
152 0b613881 Kevin Wolf
153 0b613881 Kevin Wolf
#define TO_EXABYTES(x)  ((x) / EXABYTES(1))
154 0b613881 Kevin Wolf
#define TO_PETABYTES(x) ((x) / PETABYTES(1))
155 0b613881 Kevin Wolf
#define TO_TERABYTES(x) ((x) / TERABYTES(1))
156 0b613881 Kevin Wolf
#define TO_GIGABYTES(x) ((x) / GIGABYTES(1))
157 0b613881 Kevin Wolf
#define TO_MEGABYTES(x) ((x) / MEGABYTES(1))
158 0b613881 Kevin Wolf
#define TO_KILOBYTES(x) ((x) / KILOBYTES(1))
159 0b613881 Kevin Wolf
160 0b613881 Kevin Wolf
static void cvtstr(double value, char *str, size_t size)
161 0b613881 Kevin Wolf
{
162 0b613881 Kevin Wolf
    char *trim;
163 0b613881 Kevin Wolf
    const char *suffix;
164 0b613881 Kevin Wolf
165 0b613881 Kevin Wolf
    if (value >= EXABYTES(1)) {
166 0b613881 Kevin Wolf
        suffix = " EiB";
167 0b613881 Kevin Wolf
        snprintf(str, size - 4, "%.3f", TO_EXABYTES(value));
168 0b613881 Kevin Wolf
    } else if (value >= PETABYTES(1)) {
169 0b613881 Kevin Wolf
        suffix = " PiB";
170 0b613881 Kevin Wolf
        snprintf(str, size - 4, "%.3f", TO_PETABYTES(value));
171 0b613881 Kevin Wolf
    } else if (value >= TERABYTES(1)) {
172 0b613881 Kevin Wolf
        suffix = " TiB";
173 0b613881 Kevin Wolf
        snprintf(str, size - 4, "%.3f", TO_TERABYTES(value));
174 0b613881 Kevin Wolf
    } else if (value >= GIGABYTES(1)) {
175 0b613881 Kevin Wolf
        suffix = " GiB";
176 0b613881 Kevin Wolf
        snprintf(str, size - 4, "%.3f", TO_GIGABYTES(value));
177 0b613881 Kevin Wolf
    } else if (value >= MEGABYTES(1)) {
178 0b613881 Kevin Wolf
        suffix = " MiB";
179 0b613881 Kevin Wolf
        snprintf(str, size - 4, "%.3f", TO_MEGABYTES(value));
180 0b613881 Kevin Wolf
    } else if (value >= KILOBYTES(1)) {
181 0b613881 Kevin Wolf
        suffix = " KiB";
182 0b613881 Kevin Wolf
        snprintf(str, size - 4, "%.3f", TO_KILOBYTES(value));
183 0b613881 Kevin Wolf
    } else {
184 0b613881 Kevin Wolf
        suffix = " bytes";
185 0b613881 Kevin Wolf
        snprintf(str, size - 6, "%f", value);
186 0b613881 Kevin Wolf
    }
187 0b613881 Kevin Wolf
188 0b613881 Kevin Wolf
    trim = strstr(str, ".000");
189 0b613881 Kevin Wolf
    if (trim) {
190 0b613881 Kevin Wolf
        strcpy(trim, suffix);
191 0b613881 Kevin Wolf
    } else {
192 0b613881 Kevin Wolf
        strcat(str, suffix);
193 0b613881 Kevin Wolf
    }
194 0b613881 Kevin Wolf
}
195 0b613881 Kevin Wolf
196 0b613881 Kevin Wolf
197 0b613881 Kevin Wolf
198 0b613881 Kevin Wolf
static struct timeval tsub(struct timeval t1, struct timeval t2)
199 0b613881 Kevin Wolf
{
200 0b613881 Kevin Wolf
    t1.tv_usec -= t2.tv_usec;
201 0b613881 Kevin Wolf
    if (t1.tv_usec < 0) {
202 0b613881 Kevin Wolf
        t1.tv_usec += 1000000;
203 0b613881 Kevin Wolf
        t1.tv_sec--;
204 0b613881 Kevin Wolf
    }
205 0b613881 Kevin Wolf
    t1.tv_sec -= t2.tv_sec;
206 0b613881 Kevin Wolf
    return t1;
207 0b613881 Kevin Wolf
}
208 0b613881 Kevin Wolf
209 0b613881 Kevin Wolf
static double tdiv(double value, struct timeval tv)
210 0b613881 Kevin Wolf
{
211 0b613881 Kevin Wolf
    return value / ((double)tv.tv_sec + ((double)tv.tv_usec / 1000000.0));
212 0b613881 Kevin Wolf
}
213 0b613881 Kevin Wolf
214 0b613881 Kevin Wolf
#define HOURS(sec)      ((sec) / (60 * 60))
215 0b613881 Kevin Wolf
#define MINUTES(sec)    (((sec) % (60 * 60)) / 60)
216 0b613881 Kevin Wolf
#define SECONDS(sec)    ((sec) % 60)
217 0b613881 Kevin Wolf
218 0b613881 Kevin Wolf
enum {
219 0b613881 Kevin Wolf
    DEFAULT_TIME        = 0x0,
220 0b613881 Kevin Wolf
    TERSE_FIXED_TIME    = 0x1,
221 0b613881 Kevin Wolf
    VERBOSE_FIXED_TIME  = 0x2,
222 0b613881 Kevin Wolf
};
223 0b613881 Kevin Wolf
224 0b613881 Kevin Wolf
static void timestr(struct timeval *tv, char *ts, size_t size, int format)
225 0b613881 Kevin Wolf
{
226 0b613881 Kevin Wolf
    double usec = (double)tv->tv_usec / 1000000.0;
227 0b613881 Kevin Wolf
228 0b613881 Kevin Wolf
    if (format & TERSE_FIXED_TIME) {
229 0b613881 Kevin Wolf
        if (!HOURS(tv->tv_sec)) {
230 0b613881 Kevin Wolf
            snprintf(ts, size, "%u:%02u.%02u",
231 0b613881 Kevin Wolf
                    (unsigned int) MINUTES(tv->tv_sec),
232 0b613881 Kevin Wolf
                    (unsigned int) SECONDS(tv->tv_sec),
233 0b613881 Kevin Wolf
                    (unsigned int) (usec * 100));
234 0b613881 Kevin Wolf
            return;
235 0b613881 Kevin Wolf
        }
236 0b613881 Kevin Wolf
        format |= VERBOSE_FIXED_TIME; /* fallback if hours needed */
237 0b613881 Kevin Wolf
    }
238 0b613881 Kevin Wolf
239 0b613881 Kevin Wolf
    if ((format & VERBOSE_FIXED_TIME) || tv->tv_sec) {
240 0b613881 Kevin Wolf
        snprintf(ts, size, "%u:%02u:%02u.%02u",
241 0b613881 Kevin Wolf
                (unsigned int) HOURS(tv->tv_sec),
242 0b613881 Kevin Wolf
                (unsigned int) MINUTES(tv->tv_sec),
243 0b613881 Kevin Wolf
                (unsigned int) SECONDS(tv->tv_sec),
244 0b613881 Kevin Wolf
                (unsigned int) (usec * 100));
245 0b613881 Kevin Wolf
    } else {
246 0b613881 Kevin Wolf
        snprintf(ts, size, "0.%04u sec", (unsigned int) (usec * 10000));
247 0b613881 Kevin Wolf
    }
248 0b613881 Kevin Wolf
}
249 0b613881 Kevin Wolf
250 797ac58c Kevin Wolf
/*
251 797ac58c Kevin Wolf
 * Parse the pattern argument to various sub-commands.
252 797ac58c Kevin Wolf
 *
253 797ac58c Kevin Wolf
 * Because the pattern is used as an argument to memset it must evaluate
254 797ac58c Kevin Wolf
 * to an unsigned integer that fits into a single byte.
255 797ac58c Kevin Wolf
 */
256 797ac58c Kevin Wolf
static int parse_pattern(const char *arg)
257 797ac58c Kevin Wolf
{
258 797ac58c Kevin Wolf
    char *endptr = NULL;
259 797ac58c Kevin Wolf
    long pattern;
260 797ac58c Kevin Wolf
261 797ac58c Kevin Wolf
    pattern = strtol(arg, &endptr, 0);
262 797ac58c Kevin Wolf
    if (pattern < 0 || pattern > UCHAR_MAX || *endptr != '\0') {
263 797ac58c Kevin Wolf
        printf("%s is not a valid pattern byte\n", arg);
264 797ac58c Kevin Wolf
        return -1;
265 797ac58c Kevin Wolf
    }
266 797ac58c Kevin Wolf
267 797ac58c Kevin Wolf
    return pattern;
268 797ac58c Kevin Wolf
}
269 797ac58c Kevin Wolf
270 797ac58c Kevin Wolf
/*
271 797ac58c Kevin Wolf
 * Memory allocation helpers.
272 797ac58c Kevin Wolf
 *
273 797ac58c Kevin Wolf
 * Make sure memory is aligned by default, or purposefully misaligned if
274 797ac58c Kevin Wolf
 * that is specified on the command line.
275 797ac58c Kevin Wolf
 */
276 797ac58c Kevin Wolf
277 797ac58c Kevin Wolf
#define MISALIGN_OFFSET     16
278 797ac58c Kevin Wolf
static void *qemu_io_alloc(BlockDriverState *bs, size_t len, int pattern)
279 797ac58c Kevin Wolf
{
280 797ac58c Kevin Wolf
    void *buf;
281 797ac58c Kevin Wolf
282 797ac58c Kevin Wolf
    if (qemuio_misalign) {
283 797ac58c Kevin Wolf
        len += MISALIGN_OFFSET;
284 797ac58c Kevin Wolf
    }
285 797ac58c Kevin Wolf
    buf = qemu_blockalign(bs, len);
286 797ac58c Kevin Wolf
    memset(buf, pattern, len);
287 797ac58c Kevin Wolf
    if (qemuio_misalign) {
288 797ac58c Kevin Wolf
        buf += MISALIGN_OFFSET;
289 797ac58c Kevin Wolf
    }
290 797ac58c Kevin Wolf
    return buf;
291 797ac58c Kevin Wolf
}
292 797ac58c Kevin Wolf
293 797ac58c Kevin Wolf
static void qemu_io_free(void *p)
294 797ac58c Kevin Wolf
{
295 797ac58c Kevin Wolf
    if (qemuio_misalign) {
296 797ac58c Kevin Wolf
        p -= MISALIGN_OFFSET;
297 797ac58c Kevin Wolf
    }
298 797ac58c Kevin Wolf
    qemu_vfree(p);
299 797ac58c Kevin Wolf
}
300 797ac58c Kevin Wolf
301 797ac58c Kevin Wolf
static void dump_buffer(const void *buffer, int64_t offset, int len)
302 797ac58c Kevin Wolf
{
303 797ac58c Kevin Wolf
    int i, j;
304 797ac58c Kevin Wolf
    const uint8_t *p;
305 797ac58c Kevin Wolf
306 797ac58c Kevin Wolf
    for (i = 0, p = buffer; i < len; i += 16) {
307 797ac58c Kevin Wolf
        const uint8_t *s = p;
308 797ac58c Kevin Wolf
309 797ac58c Kevin Wolf
        printf("%08" PRIx64 ":  ", offset + i);
310 797ac58c Kevin Wolf
        for (j = 0; j < 16 && i + j < len; j++, p++) {
311 797ac58c Kevin Wolf
            printf("%02x ", *p);
312 797ac58c Kevin Wolf
        }
313 797ac58c Kevin Wolf
        printf(" ");
314 797ac58c Kevin Wolf
        for (j = 0; j < 16 && i + j < len; j++, s++) {
315 797ac58c Kevin Wolf
            if (isalnum(*s)) {
316 797ac58c Kevin Wolf
                printf("%c", *s);
317 797ac58c Kevin Wolf
            } else {
318 797ac58c Kevin Wolf
                printf(".");
319 797ac58c Kevin Wolf
            }
320 797ac58c Kevin Wolf
        }
321 797ac58c Kevin Wolf
        printf("\n");
322 797ac58c Kevin Wolf
    }
323 797ac58c Kevin Wolf
}
324 797ac58c Kevin Wolf
325 797ac58c Kevin Wolf
static void print_report(const char *op, struct timeval *t, int64_t offset,
326 797ac58c Kevin Wolf
                         int count, int total, int cnt, int Cflag)
327 797ac58c Kevin Wolf
{
328 797ac58c Kevin Wolf
    char s1[64], s2[64], ts[64];
329 797ac58c Kevin Wolf
330 797ac58c Kevin Wolf
    timestr(t, ts, sizeof(ts), Cflag ? VERBOSE_FIXED_TIME : 0);
331 797ac58c Kevin Wolf
    if (!Cflag) {
332 797ac58c Kevin Wolf
        cvtstr((double)total, s1, sizeof(s1));
333 797ac58c Kevin Wolf
        cvtstr(tdiv((double)total, *t), s2, sizeof(s2));
334 797ac58c Kevin Wolf
        printf("%s %d/%d bytes at offset %" PRId64 "\n",
335 797ac58c Kevin Wolf
               op, total, count, offset);
336 797ac58c Kevin Wolf
        printf("%s, %d ops; %s (%s/sec and %.4f ops/sec)\n",
337 797ac58c Kevin Wolf
               s1, cnt, ts, s2, tdiv((double)cnt, *t));
338 797ac58c Kevin Wolf
    } else {/* bytes,ops,time,bytes/sec,ops/sec */
339 797ac58c Kevin Wolf
        printf("%d,%d,%s,%.3f,%.3f\n",
340 797ac58c Kevin Wolf
            total, cnt, ts,
341 797ac58c Kevin Wolf
            tdiv((double)total, *t),
342 797ac58c Kevin Wolf
            tdiv((double)cnt, *t));
343 797ac58c Kevin Wolf
    }
344 797ac58c Kevin Wolf
}
345 797ac58c Kevin Wolf
346 797ac58c Kevin Wolf
/*
347 797ac58c Kevin Wolf
 * Parse multiple length statements for vectored I/O, and construct an I/O
348 797ac58c Kevin Wolf
 * vector matching it.
349 797ac58c Kevin Wolf
 */
350 797ac58c Kevin Wolf
static void *
351 797ac58c Kevin Wolf
create_iovec(BlockDriverState *bs, QEMUIOVector *qiov, char **argv, int nr_iov,
352 797ac58c Kevin Wolf
             int pattern)
353 797ac58c Kevin Wolf
{
354 797ac58c Kevin Wolf
    size_t *sizes = g_new0(size_t, nr_iov);
355 797ac58c Kevin Wolf
    size_t count = 0;
356 797ac58c Kevin Wolf
    void *buf = NULL;
357 797ac58c Kevin Wolf
    void *p;
358 797ac58c Kevin Wolf
    int i;
359 797ac58c Kevin Wolf
360 797ac58c Kevin Wolf
    for (i = 0; i < nr_iov; i++) {
361 797ac58c Kevin Wolf
        char *arg = argv[i];
362 797ac58c Kevin Wolf
        int64_t len;
363 797ac58c Kevin Wolf
364 797ac58c Kevin Wolf
        len = cvtnum(arg);
365 797ac58c Kevin Wolf
        if (len < 0) {
366 797ac58c Kevin Wolf
            printf("non-numeric length argument -- %s\n", arg);
367 797ac58c Kevin Wolf
            goto fail;
368 797ac58c Kevin Wolf
        }
369 797ac58c Kevin Wolf
370 797ac58c Kevin Wolf
        /* should be SIZE_T_MAX, but that doesn't exist */
371 797ac58c Kevin Wolf
        if (len > INT_MAX) {
372 797ac58c Kevin Wolf
            printf("too large length argument -- %s\n", arg);
373 797ac58c Kevin Wolf
            goto fail;
374 797ac58c Kevin Wolf
        }
375 797ac58c Kevin Wolf
376 797ac58c Kevin Wolf
        if (len & 0x1ff) {
377 797ac58c Kevin Wolf
            printf("length argument %" PRId64
378 797ac58c Kevin Wolf
                   " is not sector aligned\n", len);
379 797ac58c Kevin Wolf
            goto fail;
380 797ac58c Kevin Wolf
        }
381 797ac58c Kevin Wolf
382 797ac58c Kevin Wolf
        sizes[i] = len;
383 797ac58c Kevin Wolf
        count += len;
384 797ac58c Kevin Wolf
    }
385 797ac58c Kevin Wolf
386 797ac58c Kevin Wolf
    qemu_iovec_init(qiov, nr_iov);
387 797ac58c Kevin Wolf
388 797ac58c Kevin Wolf
    buf = p = qemu_io_alloc(bs, count, pattern);
389 797ac58c Kevin Wolf
390 797ac58c Kevin Wolf
    for (i = 0; i < nr_iov; i++) {
391 797ac58c Kevin Wolf
        qemu_iovec_add(qiov, p, sizes[i]);
392 797ac58c Kevin Wolf
        p += sizes[i];
393 797ac58c Kevin Wolf
    }
394 797ac58c Kevin Wolf
395 797ac58c Kevin Wolf
fail:
396 797ac58c Kevin Wolf
    g_free(sizes);
397 797ac58c Kevin Wolf
    return buf;
398 797ac58c Kevin Wolf
}
399 797ac58c Kevin Wolf
400 797ac58c Kevin Wolf
static int do_read(BlockDriverState *bs, char *buf, int64_t offset, int count,
401 797ac58c Kevin Wolf
                   int *total)
402 797ac58c Kevin Wolf
{
403 797ac58c Kevin Wolf
    int ret;
404 797ac58c Kevin Wolf
405 797ac58c Kevin Wolf
    ret = bdrv_read(bs, offset >> 9, (uint8_t *)buf, count >> 9);
406 797ac58c Kevin Wolf
    if (ret < 0) {
407 797ac58c Kevin Wolf
        return ret;
408 797ac58c Kevin Wolf
    }
409 797ac58c Kevin Wolf
    *total = count;
410 797ac58c Kevin Wolf
    return 1;
411 797ac58c Kevin Wolf
}
412 797ac58c Kevin Wolf
413 797ac58c Kevin Wolf
static int do_write(BlockDriverState *bs, char *buf, int64_t offset, int count,
414 797ac58c Kevin Wolf
                    int *total)
415 797ac58c Kevin Wolf
{
416 797ac58c Kevin Wolf
    int ret;
417 797ac58c Kevin Wolf
418 797ac58c Kevin Wolf
    ret = bdrv_write(bs, offset >> 9, (uint8_t *)buf, count >> 9);
419 797ac58c Kevin Wolf
    if (ret < 0) {
420 797ac58c Kevin Wolf
        return ret;
421 797ac58c Kevin Wolf
    }
422 797ac58c Kevin Wolf
    *total = count;
423 797ac58c Kevin Wolf
    return 1;
424 797ac58c Kevin Wolf
}
425 797ac58c Kevin Wolf
426 797ac58c Kevin Wolf
static int do_pread(BlockDriverState *bs, char *buf, int64_t offset, int count,
427 797ac58c Kevin Wolf
                    int *total)
428 797ac58c Kevin Wolf
{
429 797ac58c Kevin Wolf
    *total = bdrv_pread(bs, offset, (uint8_t *)buf, count);
430 797ac58c Kevin Wolf
    if (*total < 0) {
431 797ac58c Kevin Wolf
        return *total;
432 797ac58c Kevin Wolf
    }
433 797ac58c Kevin Wolf
    return 1;
434 797ac58c Kevin Wolf
}
435 797ac58c Kevin Wolf
436 797ac58c Kevin Wolf
static int do_pwrite(BlockDriverState *bs, char *buf, int64_t offset, int count,
437 797ac58c Kevin Wolf
                     int *total)
438 797ac58c Kevin Wolf
{
439 797ac58c Kevin Wolf
    *total = bdrv_pwrite(bs, offset, (uint8_t *)buf, count);
440 797ac58c Kevin Wolf
    if (*total < 0) {
441 797ac58c Kevin Wolf
        return *total;
442 797ac58c Kevin Wolf
    }
443 797ac58c Kevin Wolf
    return 1;
444 797ac58c Kevin Wolf
}
445 797ac58c Kevin Wolf
446 797ac58c Kevin Wolf
typedef struct {
447 797ac58c Kevin Wolf
    BlockDriverState *bs;
448 797ac58c Kevin Wolf
    int64_t offset;
449 797ac58c Kevin Wolf
    int count;
450 797ac58c Kevin Wolf
    int *total;
451 797ac58c Kevin Wolf
    int ret;
452 797ac58c Kevin Wolf
    bool done;
453 797ac58c Kevin Wolf
} CoWriteZeroes;
454 797ac58c Kevin Wolf
455 797ac58c Kevin Wolf
static void coroutine_fn co_write_zeroes_entry(void *opaque)
456 797ac58c Kevin Wolf
{
457 797ac58c Kevin Wolf
    CoWriteZeroes *data = opaque;
458 797ac58c Kevin Wolf
459 797ac58c Kevin Wolf
    data->ret = bdrv_co_write_zeroes(data->bs, data->offset / BDRV_SECTOR_SIZE,
460 aa7bfbff Peter Lieven
                                     data->count / BDRV_SECTOR_SIZE, 0);
461 797ac58c Kevin Wolf
    data->done = true;
462 797ac58c Kevin Wolf
    if (data->ret < 0) {
463 797ac58c Kevin Wolf
        *data->total = data->ret;
464 797ac58c Kevin Wolf
        return;
465 797ac58c Kevin Wolf
    }
466 797ac58c Kevin Wolf
467 797ac58c Kevin Wolf
    *data->total = data->count;
468 797ac58c Kevin Wolf
}
469 797ac58c Kevin Wolf
470 797ac58c Kevin Wolf
static int do_co_write_zeroes(BlockDriverState *bs, int64_t offset, int count,
471 797ac58c Kevin Wolf
                              int *total)
472 797ac58c Kevin Wolf
{
473 797ac58c Kevin Wolf
    Coroutine *co;
474 797ac58c Kevin Wolf
    CoWriteZeroes data = {
475 797ac58c Kevin Wolf
        .bs     = bs,
476 797ac58c Kevin Wolf
        .offset = offset,
477 797ac58c Kevin Wolf
        .count  = count,
478 797ac58c Kevin Wolf
        .total  = total,
479 797ac58c Kevin Wolf
        .done   = false,
480 797ac58c Kevin Wolf
    };
481 797ac58c Kevin Wolf
482 797ac58c Kevin Wolf
    co = qemu_coroutine_create(co_write_zeroes_entry);
483 797ac58c Kevin Wolf
    qemu_coroutine_enter(co, &data);
484 797ac58c Kevin Wolf
    while (!data.done) {
485 797ac58c Kevin Wolf
        qemu_aio_wait();
486 797ac58c Kevin Wolf
    }
487 797ac58c Kevin Wolf
    if (data.ret < 0) {
488 797ac58c Kevin Wolf
        return data.ret;
489 797ac58c Kevin Wolf
    } else {
490 797ac58c Kevin Wolf
        return 1;
491 797ac58c Kevin Wolf
    }
492 797ac58c Kevin Wolf
}
493 797ac58c Kevin Wolf
494 797ac58c Kevin Wolf
static int do_write_compressed(BlockDriverState *bs, char *buf, int64_t offset,
495 797ac58c Kevin Wolf
                               int count, int *total)
496 797ac58c Kevin Wolf
{
497 797ac58c Kevin Wolf
    int ret;
498 797ac58c Kevin Wolf
499 797ac58c Kevin Wolf
    ret = bdrv_write_compressed(bs, offset >> 9, (uint8_t *)buf, count >> 9);
500 797ac58c Kevin Wolf
    if (ret < 0) {
501 797ac58c Kevin Wolf
        return ret;
502 797ac58c Kevin Wolf
    }
503 797ac58c Kevin Wolf
    *total = count;
504 797ac58c Kevin Wolf
    return 1;
505 797ac58c Kevin Wolf
}
506 797ac58c Kevin Wolf
507 797ac58c Kevin Wolf
static int do_load_vmstate(BlockDriverState *bs, char *buf, int64_t offset,
508 797ac58c Kevin Wolf
                           int count, int *total)
509 797ac58c Kevin Wolf
{
510 797ac58c Kevin Wolf
    *total = bdrv_load_vmstate(bs, (uint8_t *)buf, offset, count);
511 797ac58c Kevin Wolf
    if (*total < 0) {
512 797ac58c Kevin Wolf
        return *total;
513 797ac58c Kevin Wolf
    }
514 797ac58c Kevin Wolf
    return 1;
515 797ac58c Kevin Wolf
}
516 797ac58c Kevin Wolf
517 797ac58c Kevin Wolf
static int do_save_vmstate(BlockDriverState *bs, char *buf, int64_t offset,
518 797ac58c Kevin Wolf
                           int count, int *total)
519 797ac58c Kevin Wolf
{
520 797ac58c Kevin Wolf
    *total = bdrv_save_vmstate(bs, (uint8_t *)buf, offset, count);
521 797ac58c Kevin Wolf
    if (*total < 0) {
522 797ac58c Kevin Wolf
        return *total;
523 797ac58c Kevin Wolf
    }
524 797ac58c Kevin Wolf
    return 1;
525 797ac58c Kevin Wolf
}
526 797ac58c Kevin Wolf
527 797ac58c Kevin Wolf
#define NOT_DONE 0x7fffffff
528 797ac58c Kevin Wolf
static void aio_rw_done(void *opaque, int ret)
529 797ac58c Kevin Wolf
{
530 797ac58c Kevin Wolf
    *(int *)opaque = ret;
531 797ac58c Kevin Wolf
}
532 797ac58c Kevin Wolf
533 797ac58c Kevin Wolf
static int do_aio_readv(BlockDriverState *bs, QEMUIOVector *qiov,
534 797ac58c Kevin Wolf
                        int64_t offset, int *total)
535 797ac58c Kevin Wolf
{
536 797ac58c Kevin Wolf
    int async_ret = NOT_DONE;
537 797ac58c Kevin Wolf
538 797ac58c Kevin Wolf
    bdrv_aio_readv(bs, offset >> 9, qiov, qiov->size >> 9,
539 797ac58c Kevin Wolf
                   aio_rw_done, &async_ret);
540 797ac58c Kevin Wolf
    while (async_ret == NOT_DONE) {
541 797ac58c Kevin Wolf
        main_loop_wait(false);
542 797ac58c Kevin Wolf
    }
543 797ac58c Kevin Wolf
544 797ac58c Kevin Wolf
    *total = qiov->size;
545 797ac58c Kevin Wolf
    return async_ret < 0 ? async_ret : 1;
546 797ac58c Kevin Wolf
}
547 797ac58c Kevin Wolf
548 797ac58c Kevin Wolf
static int do_aio_writev(BlockDriverState *bs, QEMUIOVector *qiov,
549 797ac58c Kevin Wolf
                         int64_t offset, int *total)
550 797ac58c Kevin Wolf
{
551 797ac58c Kevin Wolf
    int async_ret = NOT_DONE;
552 797ac58c Kevin Wolf
553 797ac58c Kevin Wolf
    bdrv_aio_writev(bs, offset >> 9, qiov, qiov->size >> 9,
554 797ac58c Kevin Wolf
                    aio_rw_done, &async_ret);
555 797ac58c Kevin Wolf
    while (async_ret == NOT_DONE) {
556 797ac58c Kevin Wolf
        main_loop_wait(false);
557 797ac58c Kevin Wolf
    }
558 797ac58c Kevin Wolf
559 797ac58c Kevin Wolf
    *total = qiov->size;
560 797ac58c Kevin Wolf
    return async_ret < 0 ? async_ret : 1;
561 797ac58c Kevin Wolf
}
562 797ac58c Kevin Wolf
563 797ac58c Kevin Wolf
struct multiwrite_async_ret {
564 797ac58c Kevin Wolf
    int num_done;
565 797ac58c Kevin Wolf
    int error;
566 797ac58c Kevin Wolf
};
567 797ac58c Kevin Wolf
568 797ac58c Kevin Wolf
static void multiwrite_cb(void *opaque, int ret)
569 797ac58c Kevin Wolf
{
570 797ac58c Kevin Wolf
    struct multiwrite_async_ret *async_ret = opaque;
571 797ac58c Kevin Wolf
572 797ac58c Kevin Wolf
    async_ret->num_done++;
573 797ac58c Kevin Wolf
    if (ret < 0) {
574 797ac58c Kevin Wolf
        async_ret->error = ret;
575 797ac58c Kevin Wolf
    }
576 797ac58c Kevin Wolf
}
577 797ac58c Kevin Wolf
578 797ac58c Kevin Wolf
static int do_aio_multiwrite(BlockDriverState *bs, BlockRequest* reqs,
579 797ac58c Kevin Wolf
                             int num_reqs, int *total)
580 797ac58c Kevin Wolf
{
581 797ac58c Kevin Wolf
    int i, ret;
582 797ac58c Kevin Wolf
    struct multiwrite_async_ret async_ret = {
583 797ac58c Kevin Wolf
        .num_done = 0,
584 797ac58c Kevin Wolf
        .error = 0,
585 797ac58c Kevin Wolf
    };
586 797ac58c Kevin Wolf
587 797ac58c Kevin Wolf
    *total = 0;
588 797ac58c Kevin Wolf
    for (i = 0; i < num_reqs; i++) {
589 797ac58c Kevin Wolf
        reqs[i].cb = multiwrite_cb;
590 797ac58c Kevin Wolf
        reqs[i].opaque = &async_ret;
591 797ac58c Kevin Wolf
        *total += reqs[i].qiov->size;
592 797ac58c Kevin Wolf
    }
593 797ac58c Kevin Wolf
594 797ac58c Kevin Wolf
    ret = bdrv_aio_multiwrite(bs, reqs, num_reqs);
595 797ac58c Kevin Wolf
    if (ret < 0) {
596 797ac58c Kevin Wolf
        return ret;
597 797ac58c Kevin Wolf
    }
598 797ac58c Kevin Wolf
599 797ac58c Kevin Wolf
    while (async_ret.num_done < num_reqs) {
600 797ac58c Kevin Wolf
        main_loop_wait(false);
601 797ac58c Kevin Wolf
    }
602 797ac58c Kevin Wolf
603 797ac58c Kevin Wolf
    return async_ret.error < 0 ? async_ret.error : 1;
604 797ac58c Kevin Wolf
}
605 797ac58c Kevin Wolf
606 797ac58c Kevin Wolf
static void read_help(void)
607 797ac58c Kevin Wolf
{
608 797ac58c Kevin Wolf
    printf(
609 797ac58c Kevin Wolf
"\n"
610 797ac58c Kevin Wolf
" reads a range of bytes from the given offset\n"
611 797ac58c Kevin Wolf
"\n"
612 797ac58c Kevin Wolf
" Example:\n"
613 797ac58c Kevin Wolf
" 'read -v 512 1k' - dumps 1 kilobyte read from 512 bytes into the file\n"
614 797ac58c Kevin Wolf
"\n"
615 797ac58c Kevin Wolf
" Reads a segment of the currently open file, optionally dumping it to the\n"
616 797ac58c Kevin Wolf
" standard output stream (with -v option) for subsequent inspection.\n"
617 797ac58c Kevin Wolf
" -b, -- read from the VM state rather than the virtual disk\n"
618 797ac58c Kevin Wolf
" -C, -- report statistics in a machine parsable format\n"
619 797ac58c Kevin Wolf
" -l, -- length for pattern verification (only with -P)\n"
620 797ac58c Kevin Wolf
" -p, -- use bdrv_pread to read the file\n"
621 797ac58c Kevin Wolf
" -P, -- use a pattern to verify read data\n"
622 797ac58c Kevin Wolf
" -q, -- quiet mode, do not show I/O statistics\n"
623 797ac58c Kevin Wolf
" -s, -- start offset for pattern verification (only with -P)\n"
624 797ac58c Kevin Wolf
" -v, -- dump buffer to standard output\n"
625 797ac58c Kevin Wolf
"\n");
626 797ac58c Kevin Wolf
}
627 797ac58c Kevin Wolf
628 797ac58c Kevin Wolf
static int read_f(BlockDriverState *bs, int argc, char **argv);
629 797ac58c Kevin Wolf
630 797ac58c Kevin Wolf
static const cmdinfo_t read_cmd = {
631 797ac58c Kevin Wolf
    .name       = "read",
632 797ac58c Kevin Wolf
    .altname    = "r",
633 797ac58c Kevin Wolf
    .cfunc      = read_f,
634 797ac58c Kevin Wolf
    .argmin     = 2,
635 797ac58c Kevin Wolf
    .argmax     = -1,
636 797ac58c Kevin Wolf
    .args       = "[-abCpqv] [-P pattern [-s off] [-l len]] off len",
637 797ac58c Kevin Wolf
    .oneline    = "reads a number of bytes at a specified offset",
638 797ac58c Kevin Wolf
    .help       = read_help,
639 797ac58c Kevin Wolf
};
640 797ac58c Kevin Wolf
641 797ac58c Kevin Wolf
static int read_f(BlockDriverState *bs, int argc, char **argv)
642 797ac58c Kevin Wolf
{
643 797ac58c Kevin Wolf
    struct timeval t1, t2;
644 797ac58c Kevin Wolf
    int Cflag = 0, pflag = 0, qflag = 0, vflag = 0;
645 797ac58c Kevin Wolf
    int Pflag = 0, sflag = 0, lflag = 0, bflag = 0;
646 797ac58c Kevin Wolf
    int c, cnt;
647 797ac58c Kevin Wolf
    char *buf;
648 797ac58c Kevin Wolf
    int64_t offset;
649 797ac58c Kevin Wolf
    int count;
650 797ac58c Kevin Wolf
    /* Some compilers get confused and warn if this is not initialized.  */
651 797ac58c Kevin Wolf
    int total = 0;
652 797ac58c Kevin Wolf
    int pattern = 0, pattern_offset = 0, pattern_count = 0;
653 797ac58c Kevin Wolf
654 797ac58c Kevin Wolf
    while ((c = getopt(argc, argv, "bCl:pP:qs:v")) != EOF) {
655 797ac58c Kevin Wolf
        switch (c) {
656 797ac58c Kevin Wolf
        case 'b':
657 797ac58c Kevin Wolf
            bflag = 1;
658 797ac58c Kevin Wolf
            break;
659 797ac58c Kevin Wolf
        case 'C':
660 797ac58c Kevin Wolf
            Cflag = 1;
661 797ac58c Kevin Wolf
            break;
662 797ac58c Kevin Wolf
        case 'l':
663 797ac58c Kevin Wolf
            lflag = 1;
664 797ac58c Kevin Wolf
            pattern_count = cvtnum(optarg);
665 797ac58c Kevin Wolf
            if (pattern_count < 0) {
666 797ac58c Kevin Wolf
                printf("non-numeric length argument -- %s\n", optarg);
667 797ac58c Kevin Wolf
                return 0;
668 797ac58c Kevin Wolf
            }
669 797ac58c Kevin Wolf
            break;
670 797ac58c Kevin Wolf
        case 'p':
671 797ac58c Kevin Wolf
            pflag = 1;
672 797ac58c Kevin Wolf
            break;
673 797ac58c Kevin Wolf
        case 'P':
674 797ac58c Kevin Wolf
            Pflag = 1;
675 797ac58c Kevin Wolf
            pattern = parse_pattern(optarg);
676 797ac58c Kevin Wolf
            if (pattern < 0) {
677 797ac58c Kevin Wolf
                return 0;
678 797ac58c Kevin Wolf
            }
679 797ac58c Kevin Wolf
            break;
680 797ac58c Kevin Wolf
        case 'q':
681 797ac58c Kevin Wolf
            qflag = 1;
682 797ac58c Kevin Wolf
            break;
683 797ac58c Kevin Wolf
        case 's':
684 797ac58c Kevin Wolf
            sflag = 1;
685 797ac58c Kevin Wolf
            pattern_offset = cvtnum(optarg);
686 797ac58c Kevin Wolf
            if (pattern_offset < 0) {
687 797ac58c Kevin Wolf
                printf("non-numeric length argument -- %s\n", optarg);
688 797ac58c Kevin Wolf
                return 0;
689 797ac58c Kevin Wolf
            }
690 797ac58c Kevin Wolf
            break;
691 797ac58c Kevin Wolf
        case 'v':
692 797ac58c Kevin Wolf
            vflag = 1;
693 797ac58c Kevin Wolf
            break;
694 797ac58c Kevin Wolf
        default:
695 c2cdf5c5 Kevin Wolf
            return qemuio_command_usage(&read_cmd);
696 797ac58c Kevin Wolf
        }
697 797ac58c Kevin Wolf
    }
698 797ac58c Kevin Wolf
699 797ac58c Kevin Wolf
    if (optind != argc - 2) {
700 c2cdf5c5 Kevin Wolf
        return qemuio_command_usage(&read_cmd);
701 797ac58c Kevin Wolf
    }
702 797ac58c Kevin Wolf
703 797ac58c Kevin Wolf
    if (bflag && pflag) {
704 797ac58c Kevin Wolf
        printf("-b and -p cannot be specified at the same time\n");
705 797ac58c Kevin Wolf
        return 0;
706 797ac58c Kevin Wolf
    }
707 797ac58c Kevin Wolf
708 797ac58c Kevin Wolf
    offset = cvtnum(argv[optind]);
709 797ac58c Kevin Wolf
    if (offset < 0) {
710 797ac58c Kevin Wolf
        printf("non-numeric length argument -- %s\n", argv[optind]);
711 797ac58c Kevin Wolf
        return 0;
712 797ac58c Kevin Wolf
    }
713 797ac58c Kevin Wolf
714 797ac58c Kevin Wolf
    optind++;
715 797ac58c Kevin Wolf
    count = cvtnum(argv[optind]);
716 797ac58c Kevin Wolf
    if (count < 0) {
717 797ac58c Kevin Wolf
        printf("non-numeric length argument -- %s\n", argv[optind]);
718 797ac58c Kevin Wolf
        return 0;
719 797ac58c Kevin Wolf
    }
720 797ac58c Kevin Wolf
721 797ac58c Kevin Wolf
    if (!Pflag && (lflag || sflag)) {
722 c2cdf5c5 Kevin Wolf
        return qemuio_command_usage(&read_cmd);
723 797ac58c Kevin Wolf
    }
724 797ac58c Kevin Wolf
725 797ac58c Kevin Wolf
    if (!lflag) {
726 797ac58c Kevin Wolf
        pattern_count = count - pattern_offset;
727 797ac58c Kevin Wolf
    }
728 797ac58c Kevin Wolf
729 797ac58c Kevin Wolf
    if ((pattern_count < 0) || (pattern_count + pattern_offset > count))  {
730 797ac58c Kevin Wolf
        printf("pattern verification range exceeds end of read data\n");
731 797ac58c Kevin Wolf
        return 0;
732 797ac58c Kevin Wolf
    }
733 797ac58c Kevin Wolf
734 797ac58c Kevin Wolf
    if (!pflag) {
735 797ac58c Kevin Wolf
        if (offset & 0x1ff) {
736 797ac58c Kevin Wolf
            printf("offset %" PRId64 " is not sector aligned\n",
737 797ac58c Kevin Wolf
                   offset);
738 797ac58c Kevin Wolf
            return 0;
739 797ac58c Kevin Wolf
        }
740 797ac58c Kevin Wolf
        if (count & 0x1ff) {
741 797ac58c Kevin Wolf
            printf("count %d is not sector aligned\n",
742 797ac58c Kevin Wolf
                   count);
743 797ac58c Kevin Wolf
            return 0;
744 797ac58c Kevin Wolf
        }
745 797ac58c Kevin Wolf
    }
746 797ac58c Kevin Wolf
747 797ac58c Kevin Wolf
    buf = qemu_io_alloc(bs, count, 0xab);
748 797ac58c Kevin Wolf
749 797ac58c Kevin Wolf
    gettimeofday(&t1, NULL);
750 797ac58c Kevin Wolf
    if (pflag) {
751 797ac58c Kevin Wolf
        cnt = do_pread(bs, buf, offset, count, &total);
752 797ac58c Kevin Wolf
    } else if (bflag) {
753 797ac58c Kevin Wolf
        cnt = do_load_vmstate(bs, buf, offset, count, &total);
754 797ac58c Kevin Wolf
    } else {
755 797ac58c Kevin Wolf
        cnt = do_read(bs, buf, offset, count, &total);
756 797ac58c Kevin Wolf
    }
757 797ac58c Kevin Wolf
    gettimeofday(&t2, NULL);
758 797ac58c Kevin Wolf
759 797ac58c Kevin Wolf
    if (cnt < 0) {
760 797ac58c Kevin Wolf
        printf("read failed: %s\n", strerror(-cnt));
761 797ac58c Kevin Wolf
        goto out;
762 797ac58c Kevin Wolf
    }
763 797ac58c Kevin Wolf
764 797ac58c Kevin Wolf
    if (Pflag) {
765 797ac58c Kevin Wolf
        void *cmp_buf = g_malloc(pattern_count);
766 797ac58c Kevin Wolf
        memset(cmp_buf, pattern, pattern_count);
767 797ac58c Kevin Wolf
        if (memcmp(buf + pattern_offset, cmp_buf, pattern_count)) {
768 797ac58c Kevin Wolf
            printf("Pattern verification failed at offset %"
769 797ac58c Kevin Wolf
                   PRId64 ", %d bytes\n",
770 797ac58c Kevin Wolf
                   offset + pattern_offset, pattern_count);
771 797ac58c Kevin Wolf
        }
772 797ac58c Kevin Wolf
        g_free(cmp_buf);
773 797ac58c Kevin Wolf
    }
774 797ac58c Kevin Wolf
775 797ac58c Kevin Wolf
    if (qflag) {
776 797ac58c Kevin Wolf
        goto out;
777 797ac58c Kevin Wolf
    }
778 797ac58c Kevin Wolf
779 797ac58c Kevin Wolf
    if (vflag) {
780 797ac58c Kevin Wolf
        dump_buffer(buf, offset, count);
781 797ac58c Kevin Wolf
    }
782 797ac58c Kevin Wolf
783 797ac58c Kevin Wolf
    /* Finally, report back -- -C gives a parsable format */
784 797ac58c Kevin Wolf
    t2 = tsub(t2, t1);
785 797ac58c Kevin Wolf
    print_report("read", &t2, offset, count, total, cnt, Cflag);
786 797ac58c Kevin Wolf
787 797ac58c Kevin Wolf
out:
788 797ac58c Kevin Wolf
    qemu_io_free(buf);
789 797ac58c Kevin Wolf
790 797ac58c Kevin Wolf
    return 0;
791 797ac58c Kevin Wolf
}
792 797ac58c Kevin Wolf
793 797ac58c Kevin Wolf
static void readv_help(void)
794 797ac58c Kevin Wolf
{
795 797ac58c Kevin Wolf
    printf(
796 797ac58c Kevin Wolf
"\n"
797 797ac58c Kevin Wolf
" reads a range of bytes from the given offset into multiple buffers\n"
798 797ac58c Kevin Wolf
"\n"
799 797ac58c Kevin Wolf
" Example:\n"
800 797ac58c Kevin Wolf
" 'readv -v 512 1k 1k ' - dumps 2 kilobytes read from 512 bytes into the file\n"
801 797ac58c Kevin Wolf
"\n"
802 797ac58c Kevin Wolf
" Reads a segment of the currently open file, optionally dumping it to the\n"
803 797ac58c Kevin Wolf
" standard output stream (with -v option) for subsequent inspection.\n"
804 797ac58c Kevin Wolf
" Uses multiple iovec buffers if more than one byte range is specified.\n"
805 797ac58c Kevin Wolf
" -C, -- report statistics in a machine parsable format\n"
806 797ac58c Kevin Wolf
" -P, -- use a pattern to verify read data\n"
807 797ac58c Kevin Wolf
" -v, -- dump buffer to standard output\n"
808 797ac58c Kevin Wolf
" -q, -- quiet mode, do not show I/O statistics\n"
809 797ac58c Kevin Wolf
"\n");
810 797ac58c Kevin Wolf
}
811 797ac58c Kevin Wolf
812 797ac58c Kevin Wolf
static int readv_f(BlockDriverState *bs, int argc, char **argv);
813 797ac58c Kevin Wolf
814 797ac58c Kevin Wolf
static const cmdinfo_t readv_cmd = {
815 797ac58c Kevin Wolf
    .name       = "readv",
816 797ac58c Kevin Wolf
    .cfunc      = readv_f,
817 797ac58c Kevin Wolf
    .argmin     = 2,
818 797ac58c Kevin Wolf
    .argmax     = -1,
819 797ac58c Kevin Wolf
    .args       = "[-Cqv] [-P pattern ] off len [len..]",
820 797ac58c Kevin Wolf
    .oneline    = "reads a number of bytes at a specified offset",
821 797ac58c Kevin Wolf
    .help       = readv_help,
822 797ac58c Kevin Wolf
};
823 797ac58c Kevin Wolf
824 797ac58c Kevin Wolf
static int readv_f(BlockDriverState *bs, int argc, char **argv)
825 797ac58c Kevin Wolf
{
826 797ac58c Kevin Wolf
    struct timeval t1, t2;
827 797ac58c Kevin Wolf
    int Cflag = 0, qflag = 0, vflag = 0;
828 797ac58c Kevin Wolf
    int c, cnt;
829 797ac58c Kevin Wolf
    char *buf;
830 797ac58c Kevin Wolf
    int64_t offset;
831 797ac58c Kevin Wolf
    /* Some compilers get confused and warn if this is not initialized.  */
832 797ac58c Kevin Wolf
    int total = 0;
833 797ac58c Kevin Wolf
    int nr_iov;
834 797ac58c Kevin Wolf
    QEMUIOVector qiov;
835 797ac58c Kevin Wolf
    int pattern = 0;
836 797ac58c Kevin Wolf
    int Pflag = 0;
837 797ac58c Kevin Wolf
838 797ac58c Kevin Wolf
    while ((c = getopt(argc, argv, "CP:qv")) != EOF) {
839 797ac58c Kevin Wolf
        switch (c) {
840 797ac58c Kevin Wolf
        case 'C':
841 797ac58c Kevin Wolf
            Cflag = 1;
842 797ac58c Kevin Wolf
            break;
843 797ac58c Kevin Wolf
        case 'P':
844 797ac58c Kevin Wolf
            Pflag = 1;
845 797ac58c Kevin Wolf
            pattern = parse_pattern(optarg);
846 797ac58c Kevin Wolf
            if (pattern < 0) {
847 797ac58c Kevin Wolf
                return 0;
848 797ac58c Kevin Wolf
            }
849 797ac58c Kevin Wolf
            break;
850 797ac58c Kevin Wolf
        case 'q':
851 797ac58c Kevin Wolf
            qflag = 1;
852 797ac58c Kevin Wolf
            break;
853 797ac58c Kevin Wolf
        case 'v':
854 797ac58c Kevin Wolf
            vflag = 1;
855 797ac58c Kevin Wolf
            break;
856 797ac58c Kevin Wolf
        default:
857 c2cdf5c5 Kevin Wolf
            return qemuio_command_usage(&readv_cmd);
858 797ac58c Kevin Wolf
        }
859 797ac58c Kevin Wolf
    }
860 797ac58c Kevin Wolf
861 797ac58c Kevin Wolf
    if (optind > argc - 2) {
862 c2cdf5c5 Kevin Wolf
        return qemuio_command_usage(&readv_cmd);
863 797ac58c Kevin Wolf
    }
864 797ac58c Kevin Wolf
865 797ac58c Kevin Wolf
866 797ac58c Kevin Wolf
    offset = cvtnum(argv[optind]);
867 797ac58c Kevin Wolf
    if (offset < 0) {
868 797ac58c Kevin Wolf
        printf("non-numeric length argument -- %s\n", argv[optind]);
869 797ac58c Kevin Wolf
        return 0;
870 797ac58c Kevin Wolf
    }
871 797ac58c Kevin Wolf
    optind++;
872 797ac58c Kevin Wolf
873 797ac58c Kevin Wolf
    if (offset & 0x1ff) {
874 797ac58c Kevin Wolf
        printf("offset %" PRId64 " is not sector aligned\n",
875 797ac58c Kevin Wolf
               offset);
876 797ac58c Kevin Wolf
        return 0;
877 797ac58c Kevin Wolf
    }
878 797ac58c Kevin Wolf
879 797ac58c Kevin Wolf
    nr_iov = argc - optind;
880 797ac58c Kevin Wolf
    buf = create_iovec(bs, &qiov, &argv[optind], nr_iov, 0xab);
881 797ac58c Kevin Wolf
    if (buf == NULL) {
882 797ac58c Kevin Wolf
        return 0;
883 797ac58c Kevin Wolf
    }
884 797ac58c Kevin Wolf
885 797ac58c Kevin Wolf
    gettimeofday(&t1, NULL);
886 797ac58c Kevin Wolf
    cnt = do_aio_readv(bs, &qiov, offset, &total);
887 797ac58c Kevin Wolf
    gettimeofday(&t2, NULL);
888 797ac58c Kevin Wolf
889 797ac58c Kevin Wolf
    if (cnt < 0) {
890 797ac58c Kevin Wolf
        printf("readv failed: %s\n", strerror(-cnt));
891 797ac58c Kevin Wolf
        goto out;
892 797ac58c Kevin Wolf
    }
893 797ac58c Kevin Wolf
894 797ac58c Kevin Wolf
    if (Pflag) {
895 797ac58c Kevin Wolf
        void *cmp_buf = g_malloc(qiov.size);
896 797ac58c Kevin Wolf
        memset(cmp_buf, pattern, qiov.size);
897 797ac58c Kevin Wolf
        if (memcmp(buf, cmp_buf, qiov.size)) {
898 797ac58c Kevin Wolf
            printf("Pattern verification failed at offset %"
899 797ac58c Kevin Wolf
                   PRId64 ", %zd bytes\n", offset, qiov.size);
900 797ac58c Kevin Wolf
        }
901 797ac58c Kevin Wolf
        g_free(cmp_buf);
902 797ac58c Kevin Wolf
    }
903 797ac58c Kevin Wolf
904 797ac58c Kevin Wolf
    if (qflag) {
905 797ac58c Kevin Wolf
        goto out;
906 797ac58c Kevin Wolf
    }
907 797ac58c Kevin Wolf
908 797ac58c Kevin Wolf
    if (vflag) {
909 797ac58c Kevin Wolf
        dump_buffer(buf, offset, qiov.size);
910 797ac58c Kevin Wolf
    }
911 797ac58c Kevin Wolf
912 797ac58c Kevin Wolf
    /* Finally, report back -- -C gives a parsable format */
913 797ac58c Kevin Wolf
    t2 = tsub(t2, t1);
914 797ac58c Kevin Wolf
    print_report("read", &t2, offset, qiov.size, total, cnt, Cflag);
915 797ac58c Kevin Wolf
916 797ac58c Kevin Wolf
out:
917 797ac58c Kevin Wolf
    qemu_iovec_destroy(&qiov);
918 797ac58c Kevin Wolf
    qemu_io_free(buf);
919 797ac58c Kevin Wolf
    return 0;
920 797ac58c Kevin Wolf
}
921 797ac58c Kevin Wolf
922 797ac58c Kevin Wolf
static void write_help(void)
923 797ac58c Kevin Wolf
{
924 797ac58c Kevin Wolf
    printf(
925 797ac58c Kevin Wolf
"\n"
926 797ac58c Kevin Wolf
" writes a range of bytes from the given offset\n"
927 797ac58c Kevin Wolf
"\n"
928 797ac58c Kevin Wolf
" Example:\n"
929 797ac58c Kevin Wolf
" 'write 512 1k' - writes 1 kilobyte at 512 bytes into the open file\n"
930 797ac58c Kevin Wolf
"\n"
931 797ac58c Kevin Wolf
" Writes into a segment of the currently open file, using a buffer\n"
932 797ac58c Kevin Wolf
" filled with a set pattern (0xcdcdcdcd).\n"
933 797ac58c Kevin Wolf
" -b, -- write to the VM state rather than the virtual disk\n"
934 797ac58c Kevin Wolf
" -c, -- write compressed data with bdrv_write_compressed\n"
935 797ac58c Kevin Wolf
" -p, -- use bdrv_pwrite to write the file\n"
936 797ac58c Kevin Wolf
" -P, -- use different pattern to fill file\n"
937 797ac58c Kevin Wolf
" -C, -- report statistics in a machine parsable format\n"
938 797ac58c Kevin Wolf
" -q, -- quiet mode, do not show I/O statistics\n"
939 797ac58c Kevin Wolf
" -z, -- write zeroes using bdrv_co_write_zeroes\n"
940 797ac58c Kevin Wolf
"\n");
941 797ac58c Kevin Wolf
}
942 797ac58c Kevin Wolf
943 797ac58c Kevin Wolf
static int write_f(BlockDriverState *bs, int argc, char **argv);
944 797ac58c Kevin Wolf
945 797ac58c Kevin Wolf
static const cmdinfo_t write_cmd = {
946 797ac58c Kevin Wolf
    .name       = "write",
947 797ac58c Kevin Wolf
    .altname    = "w",
948 797ac58c Kevin Wolf
    .cfunc      = write_f,
949 797ac58c Kevin Wolf
    .argmin     = 2,
950 797ac58c Kevin Wolf
    .argmax     = -1,
951 797ac58c Kevin Wolf
    .args       = "[-bcCpqz] [-P pattern ] off len",
952 797ac58c Kevin Wolf
    .oneline    = "writes a number of bytes at a specified offset",
953 797ac58c Kevin Wolf
    .help       = write_help,
954 797ac58c Kevin Wolf
};
955 797ac58c Kevin Wolf
956 797ac58c Kevin Wolf
static int write_f(BlockDriverState *bs, int argc, char **argv)
957 797ac58c Kevin Wolf
{
958 797ac58c Kevin Wolf
    struct timeval t1, t2;
959 797ac58c Kevin Wolf
    int Cflag = 0, pflag = 0, qflag = 0, bflag = 0, Pflag = 0, zflag = 0;
960 797ac58c Kevin Wolf
    int cflag = 0;
961 797ac58c Kevin Wolf
    int c, cnt;
962 797ac58c Kevin Wolf
    char *buf = NULL;
963 797ac58c Kevin Wolf
    int64_t offset;
964 797ac58c Kevin Wolf
    int count;
965 797ac58c Kevin Wolf
    /* Some compilers get confused and warn if this is not initialized.  */
966 797ac58c Kevin Wolf
    int total = 0;
967 797ac58c Kevin Wolf
    int pattern = 0xcd;
968 797ac58c Kevin Wolf
969 797ac58c Kevin Wolf
    while ((c = getopt(argc, argv, "bcCpP:qz")) != EOF) {
970 797ac58c Kevin Wolf
        switch (c) {
971 797ac58c Kevin Wolf
        case 'b':
972 797ac58c Kevin Wolf
            bflag = 1;
973 797ac58c Kevin Wolf
            break;
974 797ac58c Kevin Wolf
        case 'c':
975 797ac58c Kevin Wolf
            cflag = 1;
976 797ac58c Kevin Wolf
            break;
977 797ac58c Kevin Wolf
        case 'C':
978 797ac58c Kevin Wolf
            Cflag = 1;
979 797ac58c Kevin Wolf
            break;
980 797ac58c Kevin Wolf
        case 'p':
981 797ac58c Kevin Wolf
            pflag = 1;
982 797ac58c Kevin Wolf
            break;
983 797ac58c Kevin Wolf
        case 'P':
984 797ac58c Kevin Wolf
            Pflag = 1;
985 797ac58c Kevin Wolf
            pattern = parse_pattern(optarg);
986 797ac58c Kevin Wolf
            if (pattern < 0) {
987 797ac58c Kevin Wolf
                return 0;
988 797ac58c Kevin Wolf
            }
989 797ac58c Kevin Wolf
            break;
990 797ac58c Kevin Wolf
        case 'q':
991 797ac58c Kevin Wolf
            qflag = 1;
992 797ac58c Kevin Wolf
            break;
993 797ac58c Kevin Wolf
        case 'z':
994 797ac58c Kevin Wolf
            zflag = 1;
995 797ac58c Kevin Wolf
            break;
996 797ac58c Kevin Wolf
        default:
997 c2cdf5c5 Kevin Wolf
            return qemuio_command_usage(&write_cmd);
998 797ac58c Kevin Wolf
        }
999 797ac58c Kevin Wolf
    }
1000 797ac58c Kevin Wolf
1001 797ac58c Kevin Wolf
    if (optind != argc - 2) {
1002 c2cdf5c5 Kevin Wolf
        return qemuio_command_usage(&write_cmd);
1003 797ac58c Kevin Wolf
    }
1004 797ac58c Kevin Wolf
1005 797ac58c Kevin Wolf
    if (bflag + pflag + zflag > 1) {
1006 797ac58c Kevin Wolf
        printf("-b, -p, or -z cannot be specified at the same time\n");
1007 797ac58c Kevin Wolf
        return 0;
1008 797ac58c Kevin Wolf
    }
1009 797ac58c Kevin Wolf
1010 797ac58c Kevin Wolf
    if (zflag && Pflag) {
1011 797ac58c Kevin Wolf
        printf("-z and -P cannot be specified at the same time\n");
1012 797ac58c Kevin Wolf
        return 0;
1013 797ac58c Kevin Wolf
    }
1014 797ac58c Kevin Wolf
1015 797ac58c Kevin Wolf
    offset = cvtnum(argv[optind]);
1016 797ac58c Kevin Wolf
    if (offset < 0) {
1017 797ac58c Kevin Wolf
        printf("non-numeric length argument -- %s\n", argv[optind]);
1018 797ac58c Kevin Wolf
        return 0;
1019 797ac58c Kevin Wolf
    }
1020 797ac58c Kevin Wolf
1021 797ac58c Kevin Wolf
    optind++;
1022 797ac58c Kevin Wolf
    count = cvtnum(argv[optind]);
1023 797ac58c Kevin Wolf
    if (count < 0) {
1024 797ac58c Kevin Wolf
        printf("non-numeric length argument -- %s\n", argv[optind]);
1025 797ac58c Kevin Wolf
        return 0;
1026 797ac58c Kevin Wolf
    }
1027 797ac58c Kevin Wolf
1028 797ac58c Kevin Wolf
    if (!pflag) {
1029 797ac58c Kevin Wolf
        if (offset & 0x1ff) {
1030 797ac58c Kevin Wolf
            printf("offset %" PRId64 " is not sector aligned\n",
1031 797ac58c Kevin Wolf
                   offset);
1032 797ac58c Kevin Wolf
            return 0;
1033 797ac58c Kevin Wolf
        }
1034 797ac58c Kevin Wolf
1035 797ac58c Kevin Wolf
        if (count & 0x1ff) {
1036 797ac58c Kevin Wolf
            printf("count %d is not sector aligned\n",
1037 797ac58c Kevin Wolf
                   count);
1038 797ac58c Kevin Wolf
            return 0;
1039 797ac58c Kevin Wolf
        }
1040 797ac58c Kevin Wolf
    }
1041 797ac58c Kevin Wolf
1042 797ac58c Kevin Wolf
    if (!zflag) {
1043 797ac58c Kevin Wolf
        buf = qemu_io_alloc(bs, count, pattern);
1044 797ac58c Kevin Wolf
    }
1045 797ac58c Kevin Wolf
1046 797ac58c Kevin Wolf
    gettimeofday(&t1, NULL);
1047 797ac58c Kevin Wolf
    if (pflag) {
1048 797ac58c Kevin Wolf
        cnt = do_pwrite(bs, buf, offset, count, &total);
1049 797ac58c Kevin Wolf
    } else if (bflag) {
1050 797ac58c Kevin Wolf
        cnt = do_save_vmstate(bs, buf, offset, count, &total);
1051 797ac58c Kevin Wolf
    } else if (zflag) {
1052 797ac58c Kevin Wolf
        cnt = do_co_write_zeroes(bs, offset, count, &total);
1053 797ac58c Kevin Wolf
    } else if (cflag) {
1054 797ac58c Kevin Wolf
        cnt = do_write_compressed(bs, buf, offset, count, &total);
1055 797ac58c Kevin Wolf
    } else {
1056 797ac58c Kevin Wolf
        cnt = do_write(bs, buf, offset, count, &total);
1057 797ac58c Kevin Wolf
    }
1058 797ac58c Kevin Wolf
    gettimeofday(&t2, NULL);
1059 797ac58c Kevin Wolf
1060 797ac58c Kevin Wolf
    if (cnt < 0) {
1061 797ac58c Kevin Wolf
        printf("write failed: %s\n", strerror(-cnt));
1062 797ac58c Kevin Wolf
        goto out;
1063 797ac58c Kevin Wolf
    }
1064 797ac58c Kevin Wolf
1065 797ac58c Kevin Wolf
    if (qflag) {
1066 797ac58c Kevin Wolf
        goto out;
1067 797ac58c Kevin Wolf
    }
1068 797ac58c Kevin Wolf
1069 797ac58c Kevin Wolf
    /* Finally, report back -- -C gives a parsable format */
1070 797ac58c Kevin Wolf
    t2 = tsub(t2, t1);
1071 797ac58c Kevin Wolf
    print_report("wrote", &t2, offset, count, total, cnt, Cflag);
1072 797ac58c Kevin Wolf
1073 797ac58c Kevin Wolf
out:
1074 797ac58c Kevin Wolf
    if (!zflag) {
1075 797ac58c Kevin Wolf
        qemu_io_free(buf);
1076 797ac58c Kevin Wolf
    }
1077 797ac58c Kevin Wolf
1078 797ac58c Kevin Wolf
    return 0;
1079 797ac58c Kevin Wolf
}
1080 797ac58c Kevin Wolf
1081 797ac58c Kevin Wolf
static void
1082 797ac58c Kevin Wolf
writev_help(void)
1083 797ac58c Kevin Wolf
{
1084 797ac58c Kevin Wolf
    printf(
1085 797ac58c Kevin Wolf
"\n"
1086 797ac58c Kevin Wolf
" writes a range of bytes from the given offset source from multiple buffers\n"
1087 797ac58c Kevin Wolf
"\n"
1088 797ac58c Kevin Wolf
" Example:\n"
1089 797ac58c Kevin Wolf
" 'write 512 1k 1k' - writes 2 kilobytes at 512 bytes into the open file\n"
1090 797ac58c Kevin Wolf
"\n"
1091 797ac58c Kevin Wolf
" Writes into a segment of the currently open file, using a buffer\n"
1092 797ac58c Kevin Wolf
" filled with a set pattern (0xcdcdcdcd).\n"
1093 797ac58c Kevin Wolf
" -P, -- use different pattern to fill file\n"
1094 797ac58c Kevin Wolf
" -C, -- report statistics in a machine parsable format\n"
1095 797ac58c Kevin Wolf
" -q, -- quiet mode, do not show I/O statistics\n"
1096 797ac58c Kevin Wolf
"\n");
1097 797ac58c Kevin Wolf
}
1098 797ac58c Kevin Wolf
1099 797ac58c Kevin Wolf
static int writev_f(BlockDriverState *bs, int argc, char **argv);
1100 797ac58c Kevin Wolf
1101 797ac58c Kevin Wolf
static const cmdinfo_t writev_cmd = {
1102 797ac58c Kevin Wolf
    .name       = "writev",
1103 797ac58c Kevin Wolf
    .cfunc      = writev_f,
1104 797ac58c Kevin Wolf
    .argmin     = 2,
1105 797ac58c Kevin Wolf
    .argmax     = -1,
1106 797ac58c Kevin Wolf
    .args       = "[-Cq] [-P pattern ] off len [len..]",
1107 797ac58c Kevin Wolf
    .oneline    = "writes a number of bytes at a specified offset",
1108 797ac58c Kevin Wolf
    .help       = writev_help,
1109 797ac58c Kevin Wolf
};
1110 797ac58c Kevin Wolf
1111 797ac58c Kevin Wolf
static int writev_f(BlockDriverState *bs, int argc, char **argv)
1112 797ac58c Kevin Wolf
{
1113 797ac58c Kevin Wolf
    struct timeval t1, t2;
1114 797ac58c Kevin Wolf
    int Cflag = 0, qflag = 0;
1115 797ac58c Kevin Wolf
    int c, cnt;
1116 797ac58c Kevin Wolf
    char *buf;
1117 797ac58c Kevin Wolf
    int64_t offset;
1118 797ac58c Kevin Wolf
    /* Some compilers get confused and warn if this is not initialized.  */
1119 797ac58c Kevin Wolf
    int total = 0;
1120 797ac58c Kevin Wolf
    int nr_iov;
1121 797ac58c Kevin Wolf
    int pattern = 0xcd;
1122 797ac58c Kevin Wolf
    QEMUIOVector qiov;
1123 797ac58c Kevin Wolf
1124 797ac58c Kevin Wolf
    while ((c = getopt(argc, argv, "CqP:")) != EOF) {
1125 797ac58c Kevin Wolf
        switch (c) {
1126 797ac58c Kevin Wolf
        case 'C':
1127 797ac58c Kevin Wolf
            Cflag = 1;
1128 797ac58c Kevin Wolf
            break;
1129 797ac58c Kevin Wolf
        case 'q':
1130 797ac58c Kevin Wolf
            qflag = 1;
1131 797ac58c Kevin Wolf
            break;
1132 797ac58c Kevin Wolf
        case 'P':
1133 797ac58c Kevin Wolf
            pattern = parse_pattern(optarg);
1134 797ac58c Kevin Wolf
            if (pattern < 0) {
1135 797ac58c Kevin Wolf
                return 0;
1136 797ac58c Kevin Wolf
            }
1137 797ac58c Kevin Wolf
            break;
1138 797ac58c Kevin Wolf
        default:
1139 c2cdf5c5 Kevin Wolf
            return qemuio_command_usage(&writev_cmd);
1140 797ac58c Kevin Wolf
        }
1141 797ac58c Kevin Wolf
    }
1142 797ac58c Kevin Wolf
1143 797ac58c Kevin Wolf
    if (optind > argc - 2) {
1144 c2cdf5c5 Kevin Wolf
        return qemuio_command_usage(&writev_cmd);
1145 797ac58c Kevin Wolf
    }
1146 797ac58c Kevin Wolf
1147 797ac58c Kevin Wolf
    offset = cvtnum(argv[optind]);
1148 797ac58c Kevin Wolf
    if (offset < 0) {
1149 797ac58c Kevin Wolf
        printf("non-numeric length argument -- %s\n", argv[optind]);
1150 797ac58c Kevin Wolf
        return 0;
1151 797ac58c Kevin Wolf
    }
1152 797ac58c Kevin Wolf
    optind++;
1153 797ac58c Kevin Wolf
1154 797ac58c Kevin Wolf
    if (offset & 0x1ff) {
1155 797ac58c Kevin Wolf
        printf("offset %" PRId64 " is not sector aligned\n",
1156 797ac58c Kevin Wolf
               offset);
1157 797ac58c Kevin Wolf
        return 0;
1158 797ac58c Kevin Wolf
    }
1159 797ac58c Kevin Wolf
1160 797ac58c Kevin Wolf
    nr_iov = argc - optind;
1161 797ac58c Kevin Wolf
    buf = create_iovec(bs, &qiov, &argv[optind], nr_iov, pattern);
1162 797ac58c Kevin Wolf
    if (buf == NULL) {
1163 797ac58c Kevin Wolf
        return 0;
1164 797ac58c Kevin Wolf
    }
1165 797ac58c Kevin Wolf
1166 797ac58c Kevin Wolf
    gettimeofday(&t1, NULL);
1167 797ac58c Kevin Wolf
    cnt = do_aio_writev(bs, &qiov, offset, &total);
1168 797ac58c Kevin Wolf
    gettimeofday(&t2, NULL);
1169 797ac58c Kevin Wolf
1170 797ac58c Kevin Wolf
    if (cnt < 0) {
1171 797ac58c Kevin Wolf
        printf("writev failed: %s\n", strerror(-cnt));
1172 797ac58c Kevin Wolf
        goto out;
1173 797ac58c Kevin Wolf
    }
1174 797ac58c Kevin Wolf
1175 797ac58c Kevin Wolf
    if (qflag) {
1176 797ac58c Kevin Wolf
        goto out;
1177 797ac58c Kevin Wolf
    }
1178 797ac58c Kevin Wolf
1179 797ac58c Kevin Wolf
    /* Finally, report back -- -C gives a parsable format */
1180 797ac58c Kevin Wolf
    t2 = tsub(t2, t1);
1181 797ac58c Kevin Wolf
    print_report("wrote", &t2, offset, qiov.size, total, cnt, Cflag);
1182 797ac58c Kevin Wolf
out:
1183 797ac58c Kevin Wolf
    qemu_iovec_destroy(&qiov);
1184 797ac58c Kevin Wolf
    qemu_io_free(buf);
1185 797ac58c Kevin Wolf
    return 0;
1186 797ac58c Kevin Wolf
}
1187 797ac58c Kevin Wolf
1188 797ac58c Kevin Wolf
static void multiwrite_help(void)
1189 797ac58c Kevin Wolf
{
1190 797ac58c Kevin Wolf
    printf(
1191 797ac58c Kevin Wolf
"\n"
1192 797ac58c Kevin Wolf
" writes a range of bytes from the given offset source from multiple buffers,\n"
1193 797ac58c Kevin Wolf
" in a batch of requests that may be merged by qemu\n"
1194 797ac58c Kevin Wolf
"\n"
1195 797ac58c Kevin Wolf
" Example:\n"
1196 797ac58c Kevin Wolf
" 'multiwrite 512 1k 1k ; 4k 1k'\n"
1197 797ac58c Kevin Wolf
"  writes 2 kB at 512 bytes and 1 kB at 4 kB into the open file\n"
1198 797ac58c Kevin Wolf
"\n"
1199 797ac58c Kevin Wolf
" Writes into a segment of the currently open file, using a buffer\n"
1200 797ac58c Kevin Wolf
" filled with a set pattern (0xcdcdcdcd). The pattern byte is increased\n"
1201 797ac58c Kevin Wolf
" by one for each request contained in the multiwrite command.\n"
1202 797ac58c Kevin Wolf
" -P, -- use different pattern to fill file\n"
1203 797ac58c Kevin Wolf
" -C, -- report statistics in a machine parsable format\n"
1204 797ac58c Kevin Wolf
" -q, -- quiet mode, do not show I/O statistics\n"
1205 797ac58c Kevin Wolf
"\n");
1206 797ac58c Kevin Wolf
}
1207 797ac58c Kevin Wolf
1208 797ac58c Kevin Wolf
static int multiwrite_f(BlockDriverState *bs, int argc, char **argv);
1209 797ac58c Kevin Wolf
1210 797ac58c Kevin Wolf
static const cmdinfo_t multiwrite_cmd = {
1211 797ac58c Kevin Wolf
    .name       = "multiwrite",
1212 797ac58c Kevin Wolf
    .cfunc      = multiwrite_f,
1213 797ac58c Kevin Wolf
    .argmin     = 2,
1214 797ac58c Kevin Wolf
    .argmax     = -1,
1215 797ac58c Kevin Wolf
    .args       = "[-Cq] [-P pattern ] off len [len..] [; off len [len..]..]",
1216 797ac58c Kevin Wolf
    .oneline    = "issues multiple write requests at once",
1217 797ac58c Kevin Wolf
    .help       = multiwrite_help,
1218 797ac58c Kevin Wolf
};
1219 797ac58c Kevin Wolf
1220 797ac58c Kevin Wolf
static int multiwrite_f(BlockDriverState *bs, int argc, char **argv)
1221 797ac58c Kevin Wolf
{
1222 797ac58c Kevin Wolf
    struct timeval t1, t2;
1223 797ac58c Kevin Wolf
    int Cflag = 0, qflag = 0;
1224 797ac58c Kevin Wolf
    int c, cnt;
1225 797ac58c Kevin Wolf
    char **buf;
1226 797ac58c Kevin Wolf
    int64_t offset, first_offset = 0;
1227 797ac58c Kevin Wolf
    /* Some compilers get confused and warn if this is not initialized.  */
1228 797ac58c Kevin Wolf
    int total = 0;
1229 797ac58c Kevin Wolf
    int nr_iov;
1230 797ac58c Kevin Wolf
    int nr_reqs;
1231 797ac58c Kevin Wolf
    int pattern = 0xcd;
1232 797ac58c Kevin Wolf
    QEMUIOVector *qiovs;
1233 797ac58c Kevin Wolf
    int i;
1234 797ac58c Kevin Wolf
    BlockRequest *reqs;
1235 797ac58c Kevin Wolf
1236 797ac58c Kevin Wolf
    while ((c = getopt(argc, argv, "CqP:")) != EOF) {
1237 797ac58c Kevin Wolf
        switch (c) {
1238 797ac58c Kevin Wolf
        case 'C':
1239 797ac58c Kevin Wolf
            Cflag = 1;
1240 797ac58c Kevin Wolf
            break;
1241 797ac58c Kevin Wolf
        case 'q':
1242 797ac58c Kevin Wolf
            qflag = 1;
1243 797ac58c Kevin Wolf
            break;
1244 797ac58c Kevin Wolf
        case 'P':
1245 797ac58c Kevin Wolf
            pattern = parse_pattern(optarg);
1246 797ac58c Kevin Wolf
            if (pattern < 0) {
1247 797ac58c Kevin Wolf
                return 0;
1248 797ac58c Kevin Wolf
            }
1249 797ac58c Kevin Wolf
            break;
1250 797ac58c Kevin Wolf
        default:
1251 c2cdf5c5 Kevin Wolf
            return qemuio_command_usage(&writev_cmd);
1252 797ac58c Kevin Wolf
        }
1253 797ac58c Kevin Wolf
    }
1254 797ac58c Kevin Wolf
1255 797ac58c Kevin Wolf
    if (optind > argc - 2) {
1256 c2cdf5c5 Kevin Wolf
        return qemuio_command_usage(&writev_cmd);
1257 797ac58c Kevin Wolf
    }
1258 797ac58c Kevin Wolf
1259 797ac58c Kevin Wolf
    nr_reqs = 1;
1260 797ac58c Kevin Wolf
    for (i = optind; i < argc; i++) {
1261 797ac58c Kevin Wolf
        if (!strcmp(argv[i], ";")) {
1262 797ac58c Kevin Wolf
            nr_reqs++;
1263 797ac58c Kevin Wolf
        }
1264 797ac58c Kevin Wolf
    }
1265 797ac58c Kevin Wolf
1266 797ac58c Kevin Wolf
    reqs = g_malloc0(nr_reqs * sizeof(*reqs));
1267 797ac58c Kevin Wolf
    buf = g_malloc0(nr_reqs * sizeof(*buf));
1268 797ac58c Kevin Wolf
    qiovs = g_malloc(nr_reqs * sizeof(*qiovs));
1269 797ac58c Kevin Wolf
1270 797ac58c Kevin Wolf
    for (i = 0; i < nr_reqs && optind < argc; i++) {
1271 797ac58c Kevin Wolf
        int j;
1272 797ac58c Kevin Wolf
1273 797ac58c Kevin Wolf
        /* Read the offset of the request */
1274 797ac58c Kevin Wolf
        offset = cvtnum(argv[optind]);
1275 797ac58c Kevin Wolf
        if (offset < 0) {
1276 797ac58c Kevin Wolf
            printf("non-numeric offset argument -- %s\n", argv[optind]);
1277 797ac58c Kevin Wolf
            goto out;
1278 797ac58c Kevin Wolf
        }
1279 797ac58c Kevin Wolf
        optind++;
1280 797ac58c Kevin Wolf
1281 797ac58c Kevin Wolf
        if (offset & 0x1ff) {
1282 797ac58c Kevin Wolf
            printf("offset %lld is not sector aligned\n",
1283 797ac58c Kevin Wolf
                   (long long)offset);
1284 797ac58c Kevin Wolf
            goto out;
1285 797ac58c Kevin Wolf
        }
1286 797ac58c Kevin Wolf
1287 797ac58c Kevin Wolf
        if (i == 0) {
1288 797ac58c Kevin Wolf
            first_offset = offset;
1289 797ac58c Kevin Wolf
        }
1290 797ac58c Kevin Wolf
1291 797ac58c Kevin Wolf
        /* Read lengths for qiov entries */
1292 797ac58c Kevin Wolf
        for (j = optind; j < argc; j++) {
1293 797ac58c Kevin Wolf
            if (!strcmp(argv[j], ";")) {
1294 797ac58c Kevin Wolf
                break;
1295 797ac58c Kevin Wolf
            }
1296 797ac58c Kevin Wolf
        }
1297 797ac58c Kevin Wolf
1298 797ac58c Kevin Wolf
        nr_iov = j - optind;
1299 797ac58c Kevin Wolf
1300 797ac58c Kevin Wolf
        /* Build request */
1301 797ac58c Kevin Wolf
        buf[i] = create_iovec(bs, &qiovs[i], &argv[optind], nr_iov, pattern);
1302 797ac58c Kevin Wolf
        if (buf[i] == NULL) {
1303 797ac58c Kevin Wolf
            goto out;
1304 797ac58c Kevin Wolf
        }
1305 797ac58c Kevin Wolf
1306 797ac58c Kevin Wolf
        reqs[i].qiov = &qiovs[i];
1307 797ac58c Kevin Wolf
        reqs[i].sector = offset >> 9;
1308 797ac58c Kevin Wolf
        reqs[i].nb_sectors = reqs[i].qiov->size >> 9;
1309 797ac58c Kevin Wolf
1310 797ac58c Kevin Wolf
        optind = j + 1;
1311 797ac58c Kevin Wolf
1312 797ac58c Kevin Wolf
        pattern++;
1313 797ac58c Kevin Wolf
    }
1314 797ac58c Kevin Wolf
1315 797ac58c Kevin Wolf
    /* If there were empty requests at the end, ignore them */
1316 797ac58c Kevin Wolf
    nr_reqs = i;
1317 797ac58c Kevin Wolf
1318 797ac58c Kevin Wolf
    gettimeofday(&t1, NULL);
1319 797ac58c Kevin Wolf
    cnt = do_aio_multiwrite(bs, reqs, nr_reqs, &total);
1320 797ac58c Kevin Wolf
    gettimeofday(&t2, NULL);
1321 797ac58c Kevin Wolf
1322 797ac58c Kevin Wolf
    if (cnt < 0) {
1323 797ac58c Kevin Wolf
        printf("aio_multiwrite failed: %s\n", strerror(-cnt));
1324 797ac58c Kevin Wolf
        goto out;
1325 797ac58c Kevin Wolf
    }
1326 797ac58c Kevin Wolf
1327 797ac58c Kevin Wolf
    if (qflag) {
1328 797ac58c Kevin Wolf
        goto out;
1329 797ac58c Kevin Wolf
    }
1330 797ac58c Kevin Wolf
1331 797ac58c Kevin Wolf
    /* Finally, report back -- -C gives a parsable format */
1332 797ac58c Kevin Wolf
    t2 = tsub(t2, t1);
1333 797ac58c Kevin Wolf
    print_report("wrote", &t2, first_offset, total, total, cnt, Cflag);
1334 797ac58c Kevin Wolf
out:
1335 797ac58c Kevin Wolf
    for (i = 0; i < nr_reqs; i++) {
1336 797ac58c Kevin Wolf
        qemu_io_free(buf[i]);
1337 797ac58c Kevin Wolf
        if (reqs[i].qiov != NULL) {
1338 797ac58c Kevin Wolf
            qemu_iovec_destroy(&qiovs[i]);
1339 797ac58c Kevin Wolf
        }
1340 797ac58c Kevin Wolf
    }
1341 797ac58c Kevin Wolf
    g_free(buf);
1342 797ac58c Kevin Wolf
    g_free(reqs);
1343 797ac58c Kevin Wolf
    g_free(qiovs);
1344 797ac58c Kevin Wolf
    return 0;
1345 797ac58c Kevin Wolf
}
1346 797ac58c Kevin Wolf
1347 797ac58c Kevin Wolf
struct aio_ctx {
1348 797ac58c Kevin Wolf
    QEMUIOVector qiov;
1349 797ac58c Kevin Wolf
    int64_t offset;
1350 797ac58c Kevin Wolf
    char *buf;
1351 797ac58c Kevin Wolf
    int qflag;
1352 797ac58c Kevin Wolf
    int vflag;
1353 797ac58c Kevin Wolf
    int Cflag;
1354 797ac58c Kevin Wolf
    int Pflag;
1355 797ac58c Kevin Wolf
    int pattern;
1356 797ac58c Kevin Wolf
    struct timeval t1;
1357 797ac58c Kevin Wolf
};
1358 797ac58c Kevin Wolf
1359 797ac58c Kevin Wolf
static void aio_write_done(void *opaque, int ret)
1360 797ac58c Kevin Wolf
{
1361 797ac58c Kevin Wolf
    struct aio_ctx *ctx = opaque;
1362 797ac58c Kevin Wolf
    struct timeval t2;
1363 797ac58c Kevin Wolf
1364 797ac58c Kevin Wolf
    gettimeofday(&t2, NULL);
1365 797ac58c Kevin Wolf
1366 797ac58c Kevin Wolf
1367 797ac58c Kevin Wolf
    if (ret < 0) {
1368 797ac58c Kevin Wolf
        printf("aio_write failed: %s\n", strerror(-ret));
1369 797ac58c Kevin Wolf
        goto out;
1370 797ac58c Kevin Wolf
    }
1371 797ac58c Kevin Wolf
1372 797ac58c Kevin Wolf
    if (ctx->qflag) {
1373 797ac58c Kevin Wolf
        goto out;
1374 797ac58c Kevin Wolf
    }
1375 797ac58c Kevin Wolf
1376 797ac58c Kevin Wolf
    /* Finally, report back -- -C gives a parsable format */
1377 797ac58c Kevin Wolf
    t2 = tsub(t2, ctx->t1);
1378 797ac58c Kevin Wolf
    print_report("wrote", &t2, ctx->offset, ctx->qiov.size,
1379 797ac58c Kevin Wolf
                 ctx->qiov.size, 1, ctx->Cflag);
1380 797ac58c Kevin Wolf
out:
1381 797ac58c Kevin Wolf
    qemu_io_free(ctx->buf);
1382 797ac58c Kevin Wolf
    qemu_iovec_destroy(&ctx->qiov);
1383 797ac58c Kevin Wolf
    g_free(ctx);
1384 797ac58c Kevin Wolf
}
1385 797ac58c Kevin Wolf
1386 797ac58c Kevin Wolf
static void aio_read_done(void *opaque, int ret)
1387 797ac58c Kevin Wolf
{
1388 797ac58c Kevin Wolf
    struct aio_ctx *ctx = opaque;
1389 797ac58c Kevin Wolf
    struct timeval t2;
1390 797ac58c Kevin Wolf
1391 797ac58c Kevin Wolf
    gettimeofday(&t2, NULL);
1392 797ac58c Kevin Wolf
1393 797ac58c Kevin Wolf
    if (ret < 0) {
1394 797ac58c Kevin Wolf
        printf("readv failed: %s\n", strerror(-ret));
1395 797ac58c Kevin Wolf
        goto out;
1396 797ac58c Kevin Wolf
    }
1397 797ac58c Kevin Wolf
1398 797ac58c Kevin Wolf
    if (ctx->Pflag) {
1399 797ac58c Kevin Wolf
        void *cmp_buf = g_malloc(ctx->qiov.size);
1400 797ac58c Kevin Wolf
1401 797ac58c Kevin Wolf
        memset(cmp_buf, ctx->pattern, ctx->qiov.size);
1402 797ac58c Kevin Wolf
        if (memcmp(ctx->buf, cmp_buf, ctx->qiov.size)) {
1403 797ac58c Kevin Wolf
            printf("Pattern verification failed at offset %"
1404 797ac58c Kevin Wolf
                   PRId64 ", %zd bytes\n", ctx->offset, ctx->qiov.size);
1405 797ac58c Kevin Wolf
        }
1406 797ac58c Kevin Wolf
        g_free(cmp_buf);
1407 797ac58c Kevin Wolf
    }
1408 797ac58c Kevin Wolf
1409 797ac58c Kevin Wolf
    if (ctx->qflag) {
1410 797ac58c Kevin Wolf
        goto out;
1411 797ac58c Kevin Wolf
    }
1412 797ac58c Kevin Wolf
1413 797ac58c Kevin Wolf
    if (ctx->vflag) {
1414 797ac58c Kevin Wolf
        dump_buffer(ctx->buf, ctx->offset, ctx->qiov.size);
1415 797ac58c Kevin Wolf
    }
1416 797ac58c Kevin Wolf
1417 797ac58c Kevin Wolf
    /* Finally, report back -- -C gives a parsable format */
1418 797ac58c Kevin Wolf
    t2 = tsub(t2, ctx->t1);
1419 797ac58c Kevin Wolf
    print_report("read", &t2, ctx->offset, ctx->qiov.size,
1420 797ac58c Kevin Wolf
                 ctx->qiov.size, 1, ctx->Cflag);
1421 797ac58c Kevin Wolf
out:
1422 797ac58c Kevin Wolf
    qemu_io_free(ctx->buf);
1423 797ac58c Kevin Wolf
    qemu_iovec_destroy(&ctx->qiov);
1424 797ac58c Kevin Wolf
    g_free(ctx);
1425 797ac58c Kevin Wolf
}
1426 797ac58c Kevin Wolf
1427 797ac58c Kevin Wolf
static void aio_read_help(void)
1428 797ac58c Kevin Wolf
{
1429 797ac58c Kevin Wolf
    printf(
1430 797ac58c Kevin Wolf
"\n"
1431 797ac58c Kevin Wolf
" asynchronously reads a range of bytes from the given offset\n"
1432 797ac58c Kevin Wolf
"\n"
1433 797ac58c Kevin Wolf
" Example:\n"
1434 797ac58c Kevin Wolf
" 'aio_read -v 512 1k 1k ' - dumps 2 kilobytes read from 512 bytes into the file\n"
1435 797ac58c Kevin Wolf
"\n"
1436 797ac58c Kevin Wolf
" Reads a segment of the currently open file, optionally dumping it to the\n"
1437 797ac58c Kevin Wolf
" standard output stream (with -v option) for subsequent inspection.\n"
1438 797ac58c Kevin Wolf
" The read is performed asynchronously and the aio_flush command must be\n"
1439 797ac58c Kevin Wolf
" used to ensure all outstanding aio requests have been completed.\n"
1440 797ac58c Kevin Wolf
" -C, -- report statistics in a machine parsable format\n"
1441 797ac58c Kevin Wolf
" -P, -- use a pattern to verify read data\n"
1442 797ac58c Kevin Wolf
" -v, -- dump buffer to standard output\n"
1443 797ac58c Kevin Wolf
" -q, -- quiet mode, do not show I/O statistics\n"
1444 797ac58c Kevin Wolf
"\n");
1445 797ac58c Kevin Wolf
}
1446 797ac58c Kevin Wolf
1447 797ac58c Kevin Wolf
static int aio_read_f(BlockDriverState *bs, int argc, char **argv);
1448 797ac58c Kevin Wolf
1449 797ac58c Kevin Wolf
static const cmdinfo_t aio_read_cmd = {
1450 797ac58c Kevin Wolf
    .name       = "aio_read",
1451 797ac58c Kevin Wolf
    .cfunc      = aio_read_f,
1452 797ac58c Kevin Wolf
    .argmin     = 2,
1453 797ac58c Kevin Wolf
    .argmax     = -1,
1454 797ac58c Kevin Wolf
    .args       = "[-Cqv] [-P pattern ] off len [len..]",
1455 797ac58c Kevin Wolf
    .oneline    = "asynchronously reads a number of bytes",
1456 797ac58c Kevin Wolf
    .help       = aio_read_help,
1457 797ac58c Kevin Wolf
};
1458 797ac58c Kevin Wolf
1459 797ac58c Kevin Wolf
static int aio_read_f(BlockDriverState *bs, int argc, char **argv)
1460 797ac58c Kevin Wolf
{
1461 797ac58c Kevin Wolf
    int nr_iov, c;
1462 797ac58c Kevin Wolf
    struct aio_ctx *ctx = g_new0(struct aio_ctx, 1);
1463 797ac58c Kevin Wolf
1464 797ac58c Kevin Wolf
    while ((c = getopt(argc, argv, "CP:qv")) != EOF) {
1465 797ac58c Kevin Wolf
        switch (c) {
1466 797ac58c Kevin Wolf
        case 'C':
1467 797ac58c Kevin Wolf
            ctx->Cflag = 1;
1468 797ac58c Kevin Wolf
            break;
1469 797ac58c Kevin Wolf
        case 'P':
1470 797ac58c Kevin Wolf
            ctx->Pflag = 1;
1471 797ac58c Kevin Wolf
            ctx->pattern = parse_pattern(optarg);
1472 797ac58c Kevin Wolf
            if (ctx->pattern < 0) {
1473 797ac58c Kevin Wolf
                g_free(ctx);
1474 797ac58c Kevin Wolf
                return 0;
1475 797ac58c Kevin Wolf
            }
1476 797ac58c Kevin Wolf
            break;
1477 797ac58c Kevin Wolf
        case 'q':
1478 797ac58c Kevin Wolf
            ctx->qflag = 1;
1479 797ac58c Kevin Wolf
            break;
1480 797ac58c Kevin Wolf
        case 'v':
1481 797ac58c Kevin Wolf
            ctx->vflag = 1;
1482 797ac58c Kevin Wolf
            break;
1483 797ac58c Kevin Wolf
        default:
1484 797ac58c Kevin Wolf
            g_free(ctx);
1485 c2cdf5c5 Kevin Wolf
            return qemuio_command_usage(&aio_read_cmd);
1486 797ac58c Kevin Wolf
        }
1487 797ac58c Kevin Wolf
    }
1488 797ac58c Kevin Wolf
1489 797ac58c Kevin Wolf
    if (optind > argc - 2) {
1490 797ac58c Kevin Wolf
        g_free(ctx);
1491 c2cdf5c5 Kevin Wolf
        return qemuio_command_usage(&aio_read_cmd);
1492 797ac58c Kevin Wolf
    }
1493 797ac58c Kevin Wolf
1494 797ac58c Kevin Wolf
    ctx->offset = cvtnum(argv[optind]);
1495 797ac58c Kevin Wolf
    if (ctx->offset < 0) {
1496 797ac58c Kevin Wolf
        printf("non-numeric length argument -- %s\n", argv[optind]);
1497 797ac58c Kevin Wolf
        g_free(ctx);
1498 797ac58c Kevin Wolf
        return 0;
1499 797ac58c Kevin Wolf
    }
1500 797ac58c Kevin Wolf
    optind++;
1501 797ac58c Kevin Wolf
1502 797ac58c Kevin Wolf
    if (ctx->offset & 0x1ff) {
1503 797ac58c Kevin Wolf
        printf("offset %" PRId64 " is not sector aligned\n",
1504 797ac58c Kevin Wolf
               ctx->offset);
1505 797ac58c Kevin Wolf
        g_free(ctx);
1506 797ac58c Kevin Wolf
        return 0;
1507 797ac58c Kevin Wolf
    }
1508 797ac58c Kevin Wolf
1509 797ac58c Kevin Wolf
    nr_iov = argc - optind;
1510 797ac58c Kevin Wolf
    ctx->buf = create_iovec(bs, &ctx->qiov, &argv[optind], nr_iov, 0xab);
1511 797ac58c Kevin Wolf
    if (ctx->buf == NULL) {
1512 797ac58c Kevin Wolf
        g_free(ctx);
1513 797ac58c Kevin Wolf
        return 0;
1514 797ac58c Kevin Wolf
    }
1515 797ac58c Kevin Wolf
1516 797ac58c Kevin Wolf
    gettimeofday(&ctx->t1, NULL);
1517 797ac58c Kevin Wolf
    bdrv_aio_readv(bs, ctx->offset >> 9, &ctx->qiov,
1518 797ac58c Kevin Wolf
                   ctx->qiov.size >> 9, aio_read_done, ctx);
1519 797ac58c Kevin Wolf
    return 0;
1520 797ac58c Kevin Wolf
}
1521 797ac58c Kevin Wolf
1522 797ac58c Kevin Wolf
static void aio_write_help(void)
1523 797ac58c Kevin Wolf
{
1524 797ac58c Kevin Wolf
    printf(
1525 797ac58c Kevin Wolf
"\n"
1526 797ac58c Kevin Wolf
" asynchronously writes a range of bytes from the given offset source\n"
1527 797ac58c Kevin Wolf
" from multiple buffers\n"
1528 797ac58c Kevin Wolf
"\n"
1529 797ac58c Kevin Wolf
" Example:\n"
1530 797ac58c Kevin Wolf
" 'aio_write 512 1k 1k' - writes 2 kilobytes at 512 bytes into the open file\n"
1531 797ac58c Kevin Wolf
"\n"
1532 797ac58c Kevin Wolf
" Writes into a segment of the currently open file, using a buffer\n"
1533 797ac58c Kevin Wolf
" filled with a set pattern (0xcdcdcdcd).\n"
1534 797ac58c Kevin Wolf
" The write is performed asynchronously and the aio_flush command must be\n"
1535 797ac58c Kevin Wolf
" used to ensure all outstanding aio requests have been completed.\n"
1536 797ac58c Kevin Wolf
" -P, -- use different pattern to fill file\n"
1537 797ac58c Kevin Wolf
" -C, -- report statistics in a machine parsable format\n"
1538 797ac58c Kevin Wolf
" -q, -- quiet mode, do not show I/O statistics\n"
1539 797ac58c Kevin Wolf
"\n");
1540 797ac58c Kevin Wolf
}
1541 797ac58c Kevin Wolf
1542 797ac58c Kevin Wolf
static int aio_write_f(BlockDriverState *bs, int argc, char **argv);
1543 797ac58c Kevin Wolf
1544 797ac58c Kevin Wolf
static const cmdinfo_t aio_write_cmd = {
1545 797ac58c Kevin Wolf
    .name       = "aio_write",
1546 797ac58c Kevin Wolf
    .cfunc      = aio_write_f,
1547 797ac58c Kevin Wolf
    .argmin     = 2,
1548 797ac58c Kevin Wolf
    .argmax     = -1,
1549 797ac58c Kevin Wolf
    .args       = "[-Cq] [-P pattern ] off len [len..]",
1550 797ac58c Kevin Wolf
    .oneline    = "asynchronously writes a number of bytes",
1551 797ac58c Kevin Wolf
    .help       = aio_write_help,
1552 797ac58c Kevin Wolf
};
1553 797ac58c Kevin Wolf
1554 797ac58c Kevin Wolf
static int aio_write_f(BlockDriverState *bs, int argc, char **argv)
1555 797ac58c Kevin Wolf
{
1556 797ac58c Kevin Wolf
    int nr_iov, c;
1557 797ac58c Kevin Wolf
    int pattern = 0xcd;
1558 797ac58c Kevin Wolf
    struct aio_ctx *ctx = g_new0(struct aio_ctx, 1);
1559 797ac58c Kevin Wolf
1560 797ac58c Kevin Wolf
    while ((c = getopt(argc, argv, "CqP:")) != EOF) {
1561 797ac58c Kevin Wolf
        switch (c) {
1562 797ac58c Kevin Wolf
        case 'C':
1563 797ac58c Kevin Wolf
            ctx->Cflag = 1;
1564 797ac58c Kevin Wolf
            break;
1565 797ac58c Kevin Wolf
        case 'q':
1566 797ac58c Kevin Wolf
            ctx->qflag = 1;
1567 797ac58c Kevin Wolf
            break;
1568 797ac58c Kevin Wolf
        case 'P':
1569 797ac58c Kevin Wolf
            pattern = parse_pattern(optarg);
1570 797ac58c Kevin Wolf
            if (pattern < 0) {
1571 797ac58c Kevin Wolf
                g_free(ctx);
1572 797ac58c Kevin Wolf
                return 0;
1573 797ac58c Kevin Wolf
            }
1574 797ac58c Kevin Wolf
            break;
1575 797ac58c Kevin Wolf
        default:
1576 797ac58c Kevin Wolf
            g_free(ctx);
1577 c2cdf5c5 Kevin Wolf
            return qemuio_command_usage(&aio_write_cmd);
1578 797ac58c Kevin Wolf
        }
1579 797ac58c Kevin Wolf
    }
1580 797ac58c Kevin Wolf
1581 797ac58c Kevin Wolf
    if (optind > argc - 2) {
1582 797ac58c Kevin Wolf
        g_free(ctx);
1583 c2cdf5c5 Kevin Wolf
        return qemuio_command_usage(&aio_write_cmd);
1584 797ac58c Kevin Wolf
    }
1585 797ac58c Kevin Wolf
1586 797ac58c Kevin Wolf
    ctx->offset = cvtnum(argv[optind]);
1587 797ac58c Kevin Wolf
    if (ctx->offset < 0) {
1588 797ac58c Kevin Wolf
        printf("non-numeric length argument -- %s\n", argv[optind]);
1589 797ac58c Kevin Wolf
        g_free(ctx);
1590 797ac58c Kevin Wolf
        return 0;
1591 797ac58c Kevin Wolf
    }
1592 797ac58c Kevin Wolf
    optind++;
1593 797ac58c Kevin Wolf
1594 797ac58c Kevin Wolf
    if (ctx->offset & 0x1ff) {
1595 797ac58c Kevin Wolf
        printf("offset %" PRId64 " is not sector aligned\n",
1596 797ac58c Kevin Wolf
               ctx->offset);
1597 797ac58c Kevin Wolf
        g_free(ctx);
1598 797ac58c Kevin Wolf
        return 0;
1599 797ac58c Kevin Wolf
    }
1600 797ac58c Kevin Wolf
1601 797ac58c Kevin Wolf
    nr_iov = argc - optind;
1602 797ac58c Kevin Wolf
    ctx->buf = create_iovec(bs, &ctx->qiov, &argv[optind], nr_iov, pattern);
1603 797ac58c Kevin Wolf
    if (ctx->buf == NULL) {
1604 797ac58c Kevin Wolf
        g_free(ctx);
1605 797ac58c Kevin Wolf
        return 0;
1606 797ac58c Kevin Wolf
    }
1607 797ac58c Kevin Wolf
1608 797ac58c Kevin Wolf
    gettimeofday(&ctx->t1, NULL);
1609 797ac58c Kevin Wolf
    bdrv_aio_writev(bs, ctx->offset >> 9, &ctx->qiov,
1610 797ac58c Kevin Wolf
                    ctx->qiov.size >> 9, aio_write_done, ctx);
1611 797ac58c Kevin Wolf
    return 0;
1612 797ac58c Kevin Wolf
}
1613 797ac58c Kevin Wolf
1614 797ac58c Kevin Wolf
static int aio_flush_f(BlockDriverState *bs, int argc, char **argv)
1615 797ac58c Kevin Wolf
{
1616 797ac58c Kevin Wolf
    bdrv_drain_all();
1617 797ac58c Kevin Wolf
    return 0;
1618 797ac58c Kevin Wolf
}
1619 797ac58c Kevin Wolf
1620 797ac58c Kevin Wolf
static const cmdinfo_t aio_flush_cmd = {
1621 797ac58c Kevin Wolf
    .name       = "aio_flush",
1622 797ac58c Kevin Wolf
    .cfunc      = aio_flush_f,
1623 797ac58c Kevin Wolf
    .oneline    = "completes all outstanding aio requests"
1624 797ac58c Kevin Wolf
};
1625 797ac58c Kevin Wolf
1626 797ac58c Kevin Wolf
static int flush_f(BlockDriverState *bs, int argc, char **argv)
1627 797ac58c Kevin Wolf
{
1628 797ac58c Kevin Wolf
    bdrv_flush(bs);
1629 797ac58c Kevin Wolf
    return 0;
1630 797ac58c Kevin Wolf
}
1631 797ac58c Kevin Wolf
1632 797ac58c Kevin Wolf
static const cmdinfo_t flush_cmd = {
1633 797ac58c Kevin Wolf
    .name       = "flush",
1634 797ac58c Kevin Wolf
    .altname    = "f",
1635 797ac58c Kevin Wolf
    .cfunc      = flush_f,
1636 797ac58c Kevin Wolf
    .oneline    = "flush all in-core file state to disk",
1637 797ac58c Kevin Wolf
};
1638 797ac58c Kevin Wolf
1639 797ac58c Kevin Wolf
static int truncate_f(BlockDriverState *bs, int argc, char **argv)
1640 797ac58c Kevin Wolf
{
1641 797ac58c Kevin Wolf
    int64_t offset;
1642 797ac58c Kevin Wolf
    int ret;
1643 797ac58c Kevin Wolf
1644 797ac58c Kevin Wolf
    offset = cvtnum(argv[1]);
1645 797ac58c Kevin Wolf
    if (offset < 0) {
1646 797ac58c Kevin Wolf
        printf("non-numeric truncate argument -- %s\n", argv[1]);
1647 797ac58c Kevin Wolf
        return 0;
1648 797ac58c Kevin Wolf
    }
1649 797ac58c Kevin Wolf
1650 797ac58c Kevin Wolf
    ret = bdrv_truncate(bs, offset);
1651 797ac58c Kevin Wolf
    if (ret < 0) {
1652 797ac58c Kevin Wolf
        printf("truncate: %s\n", strerror(-ret));
1653 797ac58c Kevin Wolf
        return 0;
1654 797ac58c Kevin Wolf
    }
1655 797ac58c Kevin Wolf
1656 797ac58c Kevin Wolf
    return 0;
1657 797ac58c Kevin Wolf
}
1658 797ac58c Kevin Wolf
1659 797ac58c Kevin Wolf
static const cmdinfo_t truncate_cmd = {
1660 797ac58c Kevin Wolf
    .name       = "truncate",
1661 797ac58c Kevin Wolf
    .altname    = "t",
1662 797ac58c Kevin Wolf
    .cfunc      = truncate_f,
1663 797ac58c Kevin Wolf
    .argmin     = 1,
1664 797ac58c Kevin Wolf
    .argmax     = 1,
1665 797ac58c Kevin Wolf
    .args       = "off",
1666 797ac58c Kevin Wolf
    .oneline    = "truncates the current file at the given offset",
1667 797ac58c Kevin Wolf
};
1668 797ac58c Kevin Wolf
1669 797ac58c Kevin Wolf
static int length_f(BlockDriverState *bs, int argc, char **argv)
1670 797ac58c Kevin Wolf
{
1671 797ac58c Kevin Wolf
    int64_t size;
1672 797ac58c Kevin Wolf
    char s1[64];
1673 797ac58c Kevin Wolf
1674 797ac58c Kevin Wolf
    size = bdrv_getlength(bs);
1675 797ac58c Kevin Wolf
    if (size < 0) {
1676 797ac58c Kevin Wolf
        printf("getlength: %s\n", strerror(-size));
1677 797ac58c Kevin Wolf
        return 0;
1678 797ac58c Kevin Wolf
    }
1679 797ac58c Kevin Wolf
1680 797ac58c Kevin Wolf
    cvtstr(size, s1, sizeof(s1));
1681 797ac58c Kevin Wolf
    printf("%s\n", s1);
1682 797ac58c Kevin Wolf
    return 0;
1683 797ac58c Kevin Wolf
}
1684 797ac58c Kevin Wolf
1685 797ac58c Kevin Wolf
1686 797ac58c Kevin Wolf
static const cmdinfo_t length_cmd = {
1687 797ac58c Kevin Wolf
    .name   = "length",
1688 797ac58c Kevin Wolf
    .altname    = "l",
1689 797ac58c Kevin Wolf
    .cfunc      = length_f,
1690 797ac58c Kevin Wolf
    .oneline    = "gets the length of the current file",
1691 797ac58c Kevin Wolf
};
1692 797ac58c Kevin Wolf
1693 797ac58c Kevin Wolf
1694 797ac58c Kevin Wolf
static int info_f(BlockDriverState *bs, int argc, char **argv)
1695 797ac58c Kevin Wolf
{
1696 797ac58c Kevin Wolf
    BlockDriverInfo bdi;
1697 a8d8ecb7 Max Reitz
    ImageInfoSpecific *spec_info;
1698 797ac58c Kevin Wolf
    char s1[64], s2[64];
1699 797ac58c Kevin Wolf
    int ret;
1700 797ac58c Kevin Wolf
1701 797ac58c Kevin Wolf
    if (bs->drv && bs->drv->format_name) {
1702 797ac58c Kevin Wolf
        printf("format name: %s\n", bs->drv->format_name);
1703 797ac58c Kevin Wolf
    }
1704 797ac58c Kevin Wolf
    if (bs->drv && bs->drv->protocol_name) {
1705 797ac58c Kevin Wolf
        printf("format name: %s\n", bs->drv->protocol_name);
1706 797ac58c Kevin Wolf
    }
1707 797ac58c Kevin Wolf
1708 797ac58c Kevin Wolf
    ret = bdrv_get_info(bs, &bdi);
1709 797ac58c Kevin Wolf
    if (ret) {
1710 797ac58c Kevin Wolf
        return 0;
1711 797ac58c Kevin Wolf
    }
1712 797ac58c Kevin Wolf
1713 797ac58c Kevin Wolf
    cvtstr(bdi.cluster_size, s1, sizeof(s1));
1714 797ac58c Kevin Wolf
    cvtstr(bdi.vm_state_offset, s2, sizeof(s2));
1715 797ac58c Kevin Wolf
1716 797ac58c Kevin Wolf
    printf("cluster size: %s\n", s1);
1717 797ac58c Kevin Wolf
    printf("vm state offset: %s\n", s2);
1718 797ac58c Kevin Wolf
1719 a8d8ecb7 Max Reitz
    spec_info = bdrv_get_specific_info(bs);
1720 a8d8ecb7 Max Reitz
    if (spec_info) {
1721 a8d8ecb7 Max Reitz
        printf("Format specific information:\n");
1722 a8d8ecb7 Max Reitz
        bdrv_image_info_specific_dump(fprintf, stdout, spec_info);
1723 a8d8ecb7 Max Reitz
        qapi_free_ImageInfoSpecific(spec_info);
1724 a8d8ecb7 Max Reitz
    }
1725 a8d8ecb7 Max Reitz
1726 797ac58c Kevin Wolf
    return 0;
1727 797ac58c Kevin Wolf
}
1728 797ac58c Kevin Wolf
1729 797ac58c Kevin Wolf
1730 797ac58c Kevin Wolf
1731 797ac58c Kevin Wolf
static const cmdinfo_t info_cmd = {
1732 797ac58c Kevin Wolf
    .name       = "info",
1733 797ac58c Kevin Wolf
    .altname    = "i",
1734 797ac58c Kevin Wolf
    .cfunc      = info_f,
1735 797ac58c Kevin Wolf
    .oneline    = "prints information about the current file",
1736 797ac58c Kevin Wolf
};
1737 797ac58c Kevin Wolf
1738 797ac58c Kevin Wolf
static void discard_help(void)
1739 797ac58c Kevin Wolf
{
1740 797ac58c Kevin Wolf
    printf(
1741 797ac58c Kevin Wolf
"\n"
1742 797ac58c Kevin Wolf
" discards a range of bytes from the given offset\n"
1743 797ac58c Kevin Wolf
"\n"
1744 797ac58c Kevin Wolf
" Example:\n"
1745 797ac58c Kevin Wolf
" 'discard 512 1k' - discards 1 kilobyte from 512 bytes into the file\n"
1746 797ac58c Kevin Wolf
"\n"
1747 797ac58c Kevin Wolf
" Discards a segment of the currently open file.\n"
1748 797ac58c Kevin Wolf
" -C, -- report statistics in a machine parsable format\n"
1749 797ac58c Kevin Wolf
" -q, -- quiet mode, do not show I/O statistics\n"
1750 797ac58c Kevin Wolf
"\n");
1751 797ac58c Kevin Wolf
}
1752 797ac58c Kevin Wolf
1753 797ac58c Kevin Wolf
static int discard_f(BlockDriverState *bs, int argc, char **argv);
1754 797ac58c Kevin Wolf
1755 797ac58c Kevin Wolf
static const cmdinfo_t discard_cmd = {
1756 797ac58c Kevin Wolf
    .name       = "discard",
1757 797ac58c Kevin Wolf
    .altname    = "d",
1758 797ac58c Kevin Wolf
    .cfunc      = discard_f,
1759 797ac58c Kevin Wolf
    .argmin     = 2,
1760 797ac58c Kevin Wolf
    .argmax     = -1,
1761 797ac58c Kevin Wolf
    .args       = "[-Cq] off len",
1762 797ac58c Kevin Wolf
    .oneline    = "discards a number of bytes at a specified offset",
1763 797ac58c Kevin Wolf
    .help       = discard_help,
1764 797ac58c Kevin Wolf
};
1765 797ac58c Kevin Wolf
1766 797ac58c Kevin Wolf
static int discard_f(BlockDriverState *bs, int argc, char **argv)
1767 797ac58c Kevin Wolf
{
1768 797ac58c Kevin Wolf
    struct timeval t1, t2;
1769 797ac58c Kevin Wolf
    int Cflag = 0, qflag = 0;
1770 797ac58c Kevin Wolf
    int c, ret;
1771 797ac58c Kevin Wolf
    int64_t offset;
1772 797ac58c Kevin Wolf
    int count;
1773 797ac58c Kevin Wolf
1774 797ac58c Kevin Wolf
    while ((c = getopt(argc, argv, "Cq")) != EOF) {
1775 797ac58c Kevin Wolf
        switch (c) {
1776 797ac58c Kevin Wolf
        case 'C':
1777 797ac58c Kevin Wolf
            Cflag = 1;
1778 797ac58c Kevin Wolf
            break;
1779 797ac58c Kevin Wolf
        case 'q':
1780 797ac58c Kevin Wolf
            qflag = 1;
1781 797ac58c Kevin Wolf
            break;
1782 797ac58c Kevin Wolf
        default:
1783 c2cdf5c5 Kevin Wolf
            return qemuio_command_usage(&discard_cmd);
1784 797ac58c Kevin Wolf
        }
1785 797ac58c Kevin Wolf
    }
1786 797ac58c Kevin Wolf
1787 797ac58c Kevin Wolf
    if (optind != argc - 2) {
1788 c2cdf5c5 Kevin Wolf
        return qemuio_command_usage(&discard_cmd);
1789 797ac58c Kevin Wolf
    }
1790 797ac58c Kevin Wolf
1791 797ac58c Kevin Wolf
    offset = cvtnum(argv[optind]);
1792 797ac58c Kevin Wolf
    if (offset < 0) {
1793 797ac58c Kevin Wolf
        printf("non-numeric length argument -- %s\n", argv[optind]);
1794 797ac58c Kevin Wolf
        return 0;
1795 797ac58c Kevin Wolf
    }
1796 797ac58c Kevin Wolf
1797 797ac58c Kevin Wolf
    optind++;
1798 797ac58c Kevin Wolf
    count = cvtnum(argv[optind]);
1799 797ac58c Kevin Wolf
    if (count < 0) {
1800 797ac58c Kevin Wolf
        printf("non-numeric length argument -- %s\n", argv[optind]);
1801 797ac58c Kevin Wolf
        return 0;
1802 797ac58c Kevin Wolf
    }
1803 797ac58c Kevin Wolf
1804 797ac58c Kevin Wolf
    gettimeofday(&t1, NULL);
1805 797ac58c Kevin Wolf
    ret = bdrv_discard(bs, offset >> BDRV_SECTOR_BITS,
1806 797ac58c Kevin Wolf
                       count >> BDRV_SECTOR_BITS);
1807 797ac58c Kevin Wolf
    gettimeofday(&t2, NULL);
1808 797ac58c Kevin Wolf
1809 797ac58c Kevin Wolf
    if (ret < 0) {
1810 797ac58c Kevin Wolf
        printf("discard failed: %s\n", strerror(-ret));
1811 797ac58c Kevin Wolf
        goto out;
1812 797ac58c Kevin Wolf
    }
1813 797ac58c Kevin Wolf
1814 797ac58c Kevin Wolf
    /* Finally, report back -- -C gives a parsable format */
1815 797ac58c Kevin Wolf
    if (!qflag) {
1816 797ac58c Kevin Wolf
        t2 = tsub(t2, t1);
1817 797ac58c Kevin Wolf
        print_report("discard", &t2, offset, count, count, 1, Cflag);
1818 797ac58c Kevin Wolf
    }
1819 797ac58c Kevin Wolf
1820 797ac58c Kevin Wolf
out:
1821 797ac58c Kevin Wolf
    return 0;
1822 797ac58c Kevin Wolf
}
1823 797ac58c Kevin Wolf
1824 797ac58c Kevin Wolf
static int alloc_f(BlockDriverState *bs, int argc, char **argv)
1825 797ac58c Kevin Wolf
{
1826 797ac58c Kevin Wolf
    int64_t offset, sector_num;
1827 797ac58c Kevin Wolf
    int nb_sectors, remaining;
1828 797ac58c Kevin Wolf
    char s1[64];
1829 797ac58c Kevin Wolf
    int num, sum_alloc;
1830 797ac58c Kevin Wolf
    int ret;
1831 797ac58c Kevin Wolf
1832 797ac58c Kevin Wolf
    offset = cvtnum(argv[1]);
1833 797ac58c Kevin Wolf
    if (offset < 0) {
1834 797ac58c Kevin Wolf
        printf("non-numeric offset argument -- %s\n", argv[1]);
1835 797ac58c Kevin Wolf
        return 0;
1836 797ac58c Kevin Wolf
    } else if (offset & 0x1ff) {
1837 797ac58c Kevin Wolf
        printf("offset %" PRId64 " is not sector aligned\n",
1838 797ac58c Kevin Wolf
               offset);
1839 797ac58c Kevin Wolf
        return 0;
1840 797ac58c Kevin Wolf
    }
1841 797ac58c Kevin Wolf
1842 797ac58c Kevin Wolf
    if (argc == 3) {
1843 797ac58c Kevin Wolf
        nb_sectors = cvtnum(argv[2]);
1844 797ac58c Kevin Wolf
        if (nb_sectors < 0) {
1845 797ac58c Kevin Wolf
            printf("non-numeric length argument -- %s\n", argv[2]);
1846 797ac58c Kevin Wolf
            return 0;
1847 797ac58c Kevin Wolf
        }
1848 797ac58c Kevin Wolf
    } else {
1849 797ac58c Kevin Wolf
        nb_sectors = 1;
1850 797ac58c Kevin Wolf
    }
1851 797ac58c Kevin Wolf
1852 797ac58c Kevin Wolf
    remaining = nb_sectors;
1853 797ac58c Kevin Wolf
    sum_alloc = 0;
1854 797ac58c Kevin Wolf
    sector_num = offset >> 9;
1855 797ac58c Kevin Wolf
    while (remaining) {
1856 797ac58c Kevin Wolf
        ret = bdrv_is_allocated(bs, sector_num, remaining, &num);
1857 d663640c Paolo Bonzini
        if (ret < 0) {
1858 d663640c Paolo Bonzini
            printf("is_allocated failed: %s\n", strerror(-ret));
1859 d663640c Paolo Bonzini
            return 0;
1860 d663640c Paolo Bonzini
        }
1861 797ac58c Kevin Wolf
        sector_num += num;
1862 797ac58c Kevin Wolf
        remaining -= num;
1863 797ac58c Kevin Wolf
        if (ret) {
1864 797ac58c Kevin Wolf
            sum_alloc += num;
1865 797ac58c Kevin Wolf
        }
1866 797ac58c Kevin Wolf
        if (num == 0) {
1867 797ac58c Kevin Wolf
            nb_sectors -= remaining;
1868 797ac58c Kevin Wolf
            remaining = 0;
1869 797ac58c Kevin Wolf
        }
1870 797ac58c Kevin Wolf
    }
1871 797ac58c Kevin Wolf
1872 797ac58c Kevin Wolf
    cvtstr(offset, s1, sizeof(s1));
1873 797ac58c Kevin Wolf
1874 797ac58c Kevin Wolf
    printf("%d/%d sectors allocated at offset %s\n",
1875 797ac58c Kevin Wolf
           sum_alloc, nb_sectors, s1);
1876 797ac58c Kevin Wolf
    return 0;
1877 797ac58c Kevin Wolf
}
1878 797ac58c Kevin Wolf
1879 797ac58c Kevin Wolf
static const cmdinfo_t alloc_cmd = {
1880 797ac58c Kevin Wolf
    .name       = "alloc",
1881 797ac58c Kevin Wolf
    .altname    = "a",
1882 797ac58c Kevin Wolf
    .argmin     = 1,
1883 797ac58c Kevin Wolf
    .argmax     = 2,
1884 797ac58c Kevin Wolf
    .cfunc      = alloc_f,
1885 797ac58c Kevin Wolf
    .args       = "off [sectors]",
1886 797ac58c Kevin Wolf
    .oneline    = "checks if a sector is present in the file",
1887 797ac58c Kevin Wolf
};
1888 797ac58c Kevin Wolf
1889 797ac58c Kevin Wolf
1890 797ac58c Kevin Wolf
static int map_is_allocated(BlockDriverState *bs, int64_t sector_num,
1891 797ac58c Kevin Wolf
                            int64_t nb_sectors, int64_t *pnum)
1892 797ac58c Kevin Wolf
{
1893 797ac58c Kevin Wolf
    int num, num_checked;
1894 797ac58c Kevin Wolf
    int ret, firstret;
1895 797ac58c Kevin Wolf
1896 797ac58c Kevin Wolf
    num_checked = MIN(nb_sectors, INT_MAX);
1897 797ac58c Kevin Wolf
    ret = bdrv_is_allocated(bs, sector_num, num_checked, &num);
1898 797ac58c Kevin Wolf
    if (ret < 0) {
1899 797ac58c Kevin Wolf
        return ret;
1900 797ac58c Kevin Wolf
    }
1901 797ac58c Kevin Wolf
1902 797ac58c Kevin Wolf
    firstret = ret;
1903 797ac58c Kevin Wolf
    *pnum = num;
1904 797ac58c Kevin Wolf
1905 797ac58c Kevin Wolf
    while (nb_sectors > 0 && ret == firstret) {
1906 797ac58c Kevin Wolf
        sector_num += num;
1907 797ac58c Kevin Wolf
        nb_sectors -= num;
1908 797ac58c Kevin Wolf
1909 797ac58c Kevin Wolf
        num_checked = MIN(nb_sectors, INT_MAX);
1910 797ac58c Kevin Wolf
        ret = bdrv_is_allocated(bs, sector_num, num_checked, &num);
1911 797ac58c Kevin Wolf
        if (ret == firstret) {
1912 797ac58c Kevin Wolf
            *pnum += num;
1913 797ac58c Kevin Wolf
        } else {
1914 797ac58c Kevin Wolf
            break;
1915 797ac58c Kevin Wolf
        }
1916 797ac58c Kevin Wolf
    }
1917 797ac58c Kevin Wolf
1918 797ac58c Kevin Wolf
    return firstret;
1919 797ac58c Kevin Wolf
}
1920 797ac58c Kevin Wolf
1921 797ac58c Kevin Wolf
static int map_f(BlockDriverState *bs, int argc, char **argv)
1922 797ac58c Kevin Wolf
{
1923 797ac58c Kevin Wolf
    int64_t offset;
1924 797ac58c Kevin Wolf
    int64_t nb_sectors;
1925 797ac58c Kevin Wolf
    char s1[64];
1926 797ac58c Kevin Wolf
    int64_t num;
1927 797ac58c Kevin Wolf
    int ret;
1928 797ac58c Kevin Wolf
    const char *retstr;
1929 797ac58c Kevin Wolf
1930 797ac58c Kevin Wolf
    offset = 0;
1931 797ac58c Kevin Wolf
    nb_sectors = bs->total_sectors;
1932 797ac58c Kevin Wolf
1933 797ac58c Kevin Wolf
    do {
1934 797ac58c Kevin Wolf
        ret = map_is_allocated(bs, offset, nb_sectors, &num);
1935 797ac58c Kevin Wolf
        if (ret < 0) {
1936 797ac58c Kevin Wolf
            error_report("Failed to get allocation status: %s", strerror(-ret));
1937 797ac58c Kevin Wolf
            return 0;
1938 797ac58c Kevin Wolf
        }
1939 797ac58c Kevin Wolf
1940 797ac58c Kevin Wolf
        retstr = ret ? "    allocated" : "not allocated";
1941 797ac58c Kevin Wolf
        cvtstr(offset << 9ULL, s1, sizeof(s1));
1942 797ac58c Kevin Wolf
        printf("[% 24" PRId64 "] % 8" PRId64 "/% 8" PRId64 " sectors %s "
1943 797ac58c Kevin Wolf
               "at offset %s (%d)\n",
1944 797ac58c Kevin Wolf
               offset << 9ULL, num, nb_sectors, retstr, s1, ret);
1945 797ac58c Kevin Wolf
1946 797ac58c Kevin Wolf
        offset += num;
1947 797ac58c Kevin Wolf
        nb_sectors -= num;
1948 797ac58c Kevin Wolf
    } while (offset < bs->total_sectors);
1949 797ac58c Kevin Wolf
1950 797ac58c Kevin Wolf
    return 0;
1951 797ac58c Kevin Wolf
}
1952 797ac58c Kevin Wolf
1953 797ac58c Kevin Wolf
static const cmdinfo_t map_cmd = {
1954 797ac58c Kevin Wolf
       .name           = "map",
1955 797ac58c Kevin Wolf
       .argmin         = 0,
1956 797ac58c Kevin Wolf
       .argmax         = 0,
1957 797ac58c Kevin Wolf
       .cfunc          = map_f,
1958 797ac58c Kevin Wolf
       .args           = "",
1959 797ac58c Kevin Wolf
       .oneline        = "prints the allocated areas of a file",
1960 797ac58c Kevin Wolf
};
1961 797ac58c Kevin Wolf
1962 797ac58c Kevin Wolf
static int break_f(BlockDriverState *bs, int argc, char **argv)
1963 797ac58c Kevin Wolf
{
1964 797ac58c Kevin Wolf
    int ret;
1965 797ac58c Kevin Wolf
1966 797ac58c Kevin Wolf
    ret = bdrv_debug_breakpoint(bs, argv[1], argv[2]);
1967 797ac58c Kevin Wolf
    if (ret < 0) {
1968 797ac58c Kevin Wolf
        printf("Could not set breakpoint: %s\n", strerror(-ret));
1969 797ac58c Kevin Wolf
    }
1970 797ac58c Kevin Wolf
1971 797ac58c Kevin Wolf
    return 0;
1972 797ac58c Kevin Wolf
}
1973 797ac58c Kevin Wolf
1974 4cc70e93 Fam Zheng
static int remove_break_f(BlockDriverState *bs, int argc, char **argv)
1975 4cc70e93 Fam Zheng
{
1976 4cc70e93 Fam Zheng
    int ret;
1977 4cc70e93 Fam Zheng
1978 4cc70e93 Fam Zheng
    ret = bdrv_debug_remove_breakpoint(bs, argv[1]);
1979 4cc70e93 Fam Zheng
    if (ret < 0) {
1980 4cc70e93 Fam Zheng
        printf("Could not remove breakpoint %s: %s\n", argv[1], strerror(-ret));
1981 4cc70e93 Fam Zheng
    }
1982 4cc70e93 Fam Zheng
1983 4cc70e93 Fam Zheng
    return 0;
1984 4cc70e93 Fam Zheng
}
1985 4cc70e93 Fam Zheng
1986 797ac58c Kevin Wolf
static const cmdinfo_t break_cmd = {
1987 797ac58c Kevin Wolf
       .name           = "break",
1988 797ac58c Kevin Wolf
       .argmin         = 2,
1989 797ac58c Kevin Wolf
       .argmax         = 2,
1990 797ac58c Kevin Wolf
       .cfunc          = break_f,
1991 797ac58c Kevin Wolf
       .args           = "event tag",
1992 797ac58c Kevin Wolf
       .oneline        = "sets a breakpoint on event and tags the stopped "
1993 797ac58c Kevin Wolf
                         "request as tag",
1994 797ac58c Kevin Wolf
};
1995 797ac58c Kevin Wolf
1996 4cc70e93 Fam Zheng
static const cmdinfo_t remove_break_cmd = {
1997 4cc70e93 Fam Zheng
       .name           = "remove_break",
1998 4cc70e93 Fam Zheng
       .argmin         = 1,
1999 4cc70e93 Fam Zheng
       .argmax         = 1,
2000 4cc70e93 Fam Zheng
       .cfunc          = remove_break_f,
2001 4cc70e93 Fam Zheng
       .args           = "tag",
2002 4cc70e93 Fam Zheng
       .oneline        = "remove a breakpoint by tag",
2003 4cc70e93 Fam Zheng
};
2004 4cc70e93 Fam Zheng
2005 797ac58c Kevin Wolf
static int resume_f(BlockDriverState *bs, int argc, char **argv)
2006 797ac58c Kevin Wolf
{
2007 797ac58c Kevin Wolf
    int ret;
2008 797ac58c Kevin Wolf
2009 797ac58c Kevin Wolf
    ret = bdrv_debug_resume(bs, argv[1]);
2010 797ac58c Kevin Wolf
    if (ret < 0) {
2011 797ac58c Kevin Wolf
        printf("Could not resume request: %s\n", strerror(-ret));
2012 797ac58c Kevin Wolf
    }
2013 797ac58c Kevin Wolf
2014 797ac58c Kevin Wolf
    return 0;
2015 797ac58c Kevin Wolf
}
2016 797ac58c Kevin Wolf
2017 797ac58c Kevin Wolf
static const cmdinfo_t resume_cmd = {
2018 797ac58c Kevin Wolf
       .name           = "resume",
2019 797ac58c Kevin Wolf
       .argmin         = 1,
2020 797ac58c Kevin Wolf
       .argmax         = 1,
2021 797ac58c Kevin Wolf
       .cfunc          = resume_f,
2022 797ac58c Kevin Wolf
       .args           = "tag",
2023 797ac58c Kevin Wolf
       .oneline        = "resumes the request tagged as tag",
2024 797ac58c Kevin Wolf
};
2025 797ac58c Kevin Wolf
2026 797ac58c Kevin Wolf
static int wait_break_f(BlockDriverState *bs, int argc, char **argv)
2027 797ac58c Kevin Wolf
{
2028 797ac58c Kevin Wolf
    while (!bdrv_debug_is_suspended(bs, argv[1])) {
2029 797ac58c Kevin Wolf
        qemu_aio_wait();
2030 797ac58c Kevin Wolf
    }
2031 797ac58c Kevin Wolf
2032 797ac58c Kevin Wolf
    return 0;
2033 797ac58c Kevin Wolf
}
2034 797ac58c Kevin Wolf
2035 797ac58c Kevin Wolf
static const cmdinfo_t wait_break_cmd = {
2036 797ac58c Kevin Wolf
       .name           = "wait_break",
2037 797ac58c Kevin Wolf
       .argmin         = 1,
2038 797ac58c Kevin Wolf
       .argmax         = 1,
2039 797ac58c Kevin Wolf
       .cfunc          = wait_break_f,
2040 797ac58c Kevin Wolf
       .args           = "tag",
2041 797ac58c Kevin Wolf
       .oneline        = "waits for the suspension of a request",
2042 797ac58c Kevin Wolf
};
2043 797ac58c Kevin Wolf
2044 797ac58c Kevin Wolf
static int abort_f(BlockDriverState *bs, int argc, char **argv)
2045 797ac58c Kevin Wolf
{
2046 797ac58c Kevin Wolf
    abort();
2047 797ac58c Kevin Wolf
}
2048 797ac58c Kevin Wolf
2049 797ac58c Kevin Wolf
static const cmdinfo_t abort_cmd = {
2050 797ac58c Kevin Wolf
       .name           = "abort",
2051 797ac58c Kevin Wolf
       .cfunc          = abort_f,
2052 797ac58c Kevin Wolf
       .flags          = CMD_NOFILE_OK,
2053 797ac58c Kevin Wolf
       .oneline        = "simulate a program crash using abort(3)",
2054 797ac58c Kevin Wolf
};
2055 797ac58c Kevin Wolf
2056 f18a834a Kevin Wolf
static void help_oneline(const char *cmd, const cmdinfo_t *ct)
2057 f18a834a Kevin Wolf
{
2058 f18a834a Kevin Wolf
    if (cmd) {
2059 f18a834a Kevin Wolf
        printf("%s ", cmd);
2060 f18a834a Kevin Wolf
    } else {
2061 f18a834a Kevin Wolf
        printf("%s ", ct->name);
2062 f18a834a Kevin Wolf
        if (ct->altname) {
2063 f18a834a Kevin Wolf
            printf("(or %s) ", ct->altname);
2064 f18a834a Kevin Wolf
        }
2065 f18a834a Kevin Wolf
    }
2066 f18a834a Kevin Wolf
2067 f18a834a Kevin Wolf
    if (ct->args) {
2068 f18a834a Kevin Wolf
        printf("%s ", ct->args);
2069 f18a834a Kevin Wolf
    }
2070 f18a834a Kevin Wolf
    printf("-- %s\n", ct->oneline);
2071 f18a834a Kevin Wolf
}
2072 f18a834a Kevin Wolf
2073 f18a834a Kevin Wolf
static void help_onecmd(const char *cmd, const cmdinfo_t *ct)
2074 f18a834a Kevin Wolf
{
2075 f18a834a Kevin Wolf
    help_oneline(cmd, ct);
2076 f18a834a Kevin Wolf
    if (ct->help) {
2077 f18a834a Kevin Wolf
        ct->help();
2078 f18a834a Kevin Wolf
    }
2079 f18a834a Kevin Wolf
}
2080 f18a834a Kevin Wolf
2081 f18a834a Kevin Wolf
static void help_all(void)
2082 f18a834a Kevin Wolf
{
2083 f18a834a Kevin Wolf
    const cmdinfo_t *ct;
2084 f18a834a Kevin Wolf
2085 f18a834a Kevin Wolf
    for (ct = cmdtab; ct < &cmdtab[ncmds]; ct++) {
2086 f18a834a Kevin Wolf
        help_oneline(ct->name, ct);
2087 f18a834a Kevin Wolf
    }
2088 f18a834a Kevin Wolf
    printf("\nUse 'help commandname' for extended help.\n");
2089 f18a834a Kevin Wolf
}
2090 f18a834a Kevin Wolf
2091 f18a834a Kevin Wolf
static int help_f(BlockDriverState *bs, int argc, char **argv)
2092 f18a834a Kevin Wolf
{
2093 f18a834a Kevin Wolf
    const cmdinfo_t *ct;
2094 f18a834a Kevin Wolf
2095 f18a834a Kevin Wolf
    if (argc == 1) {
2096 f18a834a Kevin Wolf
        help_all();
2097 f18a834a Kevin Wolf
        return 0;
2098 f18a834a Kevin Wolf
    }
2099 f18a834a Kevin Wolf
2100 f18a834a Kevin Wolf
    ct = find_command(argv[1]);
2101 f18a834a Kevin Wolf
    if (ct == NULL) {
2102 f18a834a Kevin Wolf
        printf("command %s not found\n", argv[1]);
2103 f18a834a Kevin Wolf
        return 0;
2104 f18a834a Kevin Wolf
    }
2105 f18a834a Kevin Wolf
2106 f18a834a Kevin Wolf
    help_onecmd(argv[1], ct);
2107 f18a834a Kevin Wolf
    return 0;
2108 f18a834a Kevin Wolf
}
2109 f18a834a Kevin Wolf
2110 f18a834a Kevin Wolf
static const cmdinfo_t help_cmd = {
2111 f18a834a Kevin Wolf
    .name       = "help",
2112 f18a834a Kevin Wolf
    .altname    = "?",
2113 f18a834a Kevin Wolf
    .cfunc      = help_f,
2114 f18a834a Kevin Wolf
    .argmin     = 0,
2115 f18a834a Kevin Wolf
    .argmax     = 1,
2116 f18a834a Kevin Wolf
    .flags      = CMD_FLAG_GLOBAL,
2117 f18a834a Kevin Wolf
    .args       = "[command]",
2118 f18a834a Kevin Wolf
    .oneline    = "help for one or all commands",
2119 f18a834a Kevin Wolf
};
2120 f18a834a Kevin Wolf
2121 3d21994f Kevin Wolf
bool qemuio_command(BlockDriverState *bs, const char *cmd)
2122 dd583296 Kevin Wolf
{
2123 dd583296 Kevin Wolf
    char *input;
2124 dd583296 Kevin Wolf
    const cmdinfo_t *ct;
2125 dd583296 Kevin Wolf
    char **v;
2126 dd583296 Kevin Wolf
    int c;
2127 dd583296 Kevin Wolf
    bool done = false;
2128 dd583296 Kevin Wolf
2129 dd583296 Kevin Wolf
    input = g_strdup(cmd);
2130 dd583296 Kevin Wolf
    v = breakline(input, &c);
2131 dd583296 Kevin Wolf
    if (c) {
2132 dd583296 Kevin Wolf
        ct = find_command(v[0]);
2133 dd583296 Kevin Wolf
        if (ct) {
2134 3d21994f Kevin Wolf
            done = command(bs, ct, c, v);
2135 dd583296 Kevin Wolf
        } else {
2136 dd583296 Kevin Wolf
            fprintf(stderr, "command \"%s\" not found\n", v[0]);
2137 dd583296 Kevin Wolf
        }
2138 dd583296 Kevin Wolf
    }
2139 dd583296 Kevin Wolf
    g_free(input);
2140 dd583296 Kevin Wolf
    g_free(v);
2141 dd583296 Kevin Wolf
2142 dd583296 Kevin Wolf
    return done;
2143 dd583296 Kevin Wolf
}
2144 dd583296 Kevin Wolf
2145 797ac58c Kevin Wolf
static void __attribute((constructor)) init_qemuio_commands(void)
2146 797ac58c Kevin Wolf
{
2147 797ac58c Kevin Wolf
    /* initialize commands */
2148 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&help_cmd);
2149 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&read_cmd);
2150 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&readv_cmd);
2151 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&write_cmd);
2152 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&writev_cmd);
2153 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&multiwrite_cmd);
2154 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&aio_read_cmd);
2155 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&aio_write_cmd);
2156 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&aio_flush_cmd);
2157 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&flush_cmd);
2158 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&truncate_cmd);
2159 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&length_cmd);
2160 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&info_cmd);
2161 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&discard_cmd);
2162 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&alloc_cmd);
2163 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&map_cmd);
2164 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&break_cmd);
2165 4cc70e93 Fam Zheng
    qemuio_add_command(&remove_break_cmd);
2166 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&resume_cmd);
2167 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&wait_break_cmd);
2168 c2cdf5c5 Kevin Wolf
    qemuio_add_command(&abort_cmd);
2169 797ac58c Kevin Wolf
}