5b15e2ecaf
Notable changes: * The only big change is the switch to using a custom QIOChannel for savevm-async, because the previously used QEMUFileOps was dropped. Changes to the current implementation: * Switch to vector based methods as required for an IO channel. For short reads the passed-in IO vector is stuffed with zeroes at the end, just to be sure. * For reading: The documentation in include/io/channel.h states that at least one byte should be read, so also error out when whe are at the very end instead of returning 0. * For reading: Fix off-by-one error when request goes beyond end. The wrong code piece was: if ((pos + size) > maxlen) { size = maxlen - pos - 1; } Previously, the last byte would not be read. It's actually possible to get a snapshot .raw file that has content all the way up the final 512 byte (= BDRV_SECTOR_SIZE) boundary without any trailing zero bytes (I wrote a script to do it). Luckily, it didn't cause a real issue, because qemu_loadvm_state() is not interested in the final (i.e. QEMU_VM_VMDESCRIPTION) section. The buffer for reading it is simply freed up afterwards and the function will assume that it read the whole section, even if that's not the case. * For writing: Make use of the generated blk_pwritev() wrapper instead of manually wrapping the coroutine to simplify and save a few lines. * Adapt to changed interfaces for blk_{pread,pwrite}: * a9262f551e ("block: Change blk_{pread,pwrite}() param order") * 3b35d4542c ("block: Add a 'flags' param to blk_pread()") * bf5b16fa40 ("block: Make blk_{pread,pwrite}() return 0 on success") Those changes especially affected the qemu-img dd patches, because the context also changed, but also some of our block drivers used the functions. * Drop qemu-common.h include: it got renamed after essentially everything was moved to other headers. The only remaining user I could find for things dropped from the header between 7.0 and 7.1 was qemu_get_vm_name() in the iscsi-initiatorname patch, but it already includes the header to which the function was moved. Signed-off-by: Fiona Ebner <f.ebner@proxmox.com>
329 lines
11 KiB
Diff
329 lines
11 KiB
Diff
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
From: Wolfgang Bumiller <w.bumiller@proxmox.com>
|
|
Date: Mon, 6 Apr 2020 12:16:40 +0200
|
|
Subject: [PATCH] PVE: [Up] qemu-img dd: add osize and read from/to
|
|
stdin/stdout
|
|
|
|
Neither convert nor dd were previously able to write to or
|
|
read from a pipe. Particularly serializing an image file
|
|
into a raw stream or vice versa can be useful, but using
|
|
`qemu-img convert -f qcow2 -O raw foo.qcow2 /dev/stdout` in
|
|
a pipe will fail trying to seek.
|
|
|
|
While dd and convert have overlapping use cases, `dd` is a
|
|
simple read/write loop while convert is much more
|
|
sophisticated and has ways to dealing with holes and blocks
|
|
of zeroes.
|
|
Since these typically can't be detected in pipes via
|
|
SEEK_DATA/HOLE or skipped while writing, dd seems to be the
|
|
better choice for implementing stdin/stdout streams.
|
|
|
|
This patch causes "if" and "of" to default to stdin and
|
|
stdout respectively, allowing only the "raw" format to be
|
|
used in these cases.
|
|
Since the input can now be a pipe we have no way of
|
|
detecting the size of the output image to create. Since we
|
|
also want to support images with a size not matching the
|
|
dd command's "bs" parameter (which, together with "count"
|
|
could be used to calculate the desired size, and is already
|
|
used to limit it), the "osize" option is added to explicitly
|
|
override the output file's size.
|
|
|
|
Signed-off-by: Wolfgang Bumiller <w.bumiller@proxmox.com>
|
|
Signed-off-by: Thomas Lamprecht <t.lamprecht@proxmox.com>
|
|
Signed-off-by: Fiona Ebner <f.ebner@proxmox.com>
|
|
---
|
|
qemu-img-cmds.hx | 4 +-
|
|
qemu-img.c | 202 ++++++++++++++++++++++++++++++-----------------
|
|
2 files changed, 133 insertions(+), 73 deletions(-)
|
|
|
|
diff --git a/qemu-img-cmds.hx b/qemu-img-cmds.hx
|
|
index 1b1dab5b17..d1616c045a 100644
|
|
--- a/qemu-img-cmds.hx
|
|
+++ b/qemu-img-cmds.hx
|
|
@@ -58,9 +58,9 @@ SRST
|
|
ERST
|
|
|
|
DEF("dd", img_dd,
|
|
- "dd [--image-opts] [-U] [-f fmt] [-O output_fmt] [bs=block_size] [count=blocks] [skip=blocks] if=input of=output")
|
|
+ "dd [--image-opts] [-U] [-f fmt] [-O output_fmt] [bs=block_size] [count=blocks] [skip=blocks] [osize=output_size] if=input of=output")
|
|
SRST
|
|
-.. option:: dd [--image-opts] [-U] [-f FMT] [-O OUTPUT_FMT] [bs=BLOCK_SIZE] [count=BLOCKS] [skip=BLOCKS] if=INPUT of=OUTPUT
|
|
+.. option:: dd [--image-opts] [-U] [-f FMT] [-O OUTPUT_FMT] [bs=BLOCK_SIZE] [count=BLOCKS] [skip=BLOCKS] [osize=OUTPUT_SIZE] if=INPUT of=OUTPUT
|
|
ERST
|
|
|
|
DEF("info", img_info,
|
|
diff --git a/qemu-img.c b/qemu-img.c
|
|
index bb36f42dd2..74afcb79ef 100644
|
|
--- a/qemu-img.c
|
|
+++ b/qemu-img.c
|
|
@@ -4826,10 +4826,12 @@ static int img_bitmap(int argc, char **argv)
|
|
#define C_IF 04
|
|
#define C_OF 010
|
|
#define C_SKIP 020
|
|
+#define C_OSIZE 040
|
|
|
|
struct DdInfo {
|
|
unsigned int flags;
|
|
int64_t count;
|
|
+ int64_t osize;
|
|
};
|
|
|
|
struct DdIo {
|
|
@@ -4905,6 +4907,19 @@ static int img_dd_skip(const char *arg,
|
|
return 0;
|
|
}
|
|
|
|
+static int img_dd_osize(const char *arg,
|
|
+ struct DdIo *in, struct DdIo *out,
|
|
+ struct DdInfo *dd)
|
|
+{
|
|
+ dd->osize = cvtnum("size", arg);
|
|
+
|
|
+ if (dd->osize < 0) {
|
|
+ return 1;
|
|
+ }
|
|
+
|
|
+ return 0;
|
|
+}
|
|
+
|
|
static int img_dd(int argc, char **argv)
|
|
{
|
|
int ret = 0;
|
|
@@ -4945,6 +4960,7 @@ static int img_dd(int argc, char **argv)
|
|
{ "if", img_dd_if, C_IF },
|
|
{ "of", img_dd_of, C_OF },
|
|
{ "skip", img_dd_skip, C_SKIP },
|
|
+ { "osize", img_dd_osize, C_OSIZE },
|
|
{ NULL, NULL, 0 }
|
|
};
|
|
const struct option long_options[] = {
|
|
@@ -5020,91 +5036,112 @@ static int img_dd(int argc, char **argv)
|
|
arg = NULL;
|
|
}
|
|
|
|
- if (!(dd.flags & C_IF && dd.flags & C_OF)) {
|
|
- error_report("Must specify both input and output files");
|
|
+ if (!(dd.flags & C_IF) && (!fmt || strcmp(fmt, "raw") != 0)) {
|
|
+ error_report("Input format must be raw when readin from stdin");
|
|
ret = -1;
|
|
goto out;
|
|
}
|
|
-
|
|
- blk1 = img_open(image_opts, in.filename, fmt, 0, false, false,
|
|
- force_share);
|
|
-
|
|
- if (!blk1) {
|
|
+ if (!(dd.flags & C_OF) && strcmp(out_fmt, "raw") != 0) {
|
|
+ error_report("Output format must be raw when writing to stdout");
|
|
ret = -1;
|
|
goto out;
|
|
}
|
|
|
|
- drv = bdrv_find_format(out_fmt);
|
|
- if (!drv) {
|
|
- error_report("Unknown file format");
|
|
- ret = -1;
|
|
- goto out;
|
|
- }
|
|
- proto_drv = bdrv_find_protocol(out.filename, true, &local_err);
|
|
+ if (dd.flags & C_IF) {
|
|
+ blk1 = img_open(image_opts, in.filename, fmt, 0, false, false,
|
|
+ force_share);
|
|
|
|
- if (!proto_drv) {
|
|
- error_report_err(local_err);
|
|
- ret = -1;
|
|
- goto out;
|
|
- }
|
|
- if (!drv->create_opts) {
|
|
- error_report("Format driver '%s' does not support image creation",
|
|
- drv->format_name);
|
|
- ret = -1;
|
|
- goto out;
|
|
- }
|
|
- if (!proto_drv->create_opts) {
|
|
- error_report("Protocol driver '%s' does not support image creation",
|
|
- proto_drv->format_name);
|
|
- ret = -1;
|
|
- goto out;
|
|
+ if (!blk1) {
|
|
+ ret = -1;
|
|
+ goto out;
|
|
+ }
|
|
}
|
|
- create_opts = qemu_opts_append(create_opts, drv->create_opts);
|
|
- create_opts = qemu_opts_append(create_opts, proto_drv->create_opts);
|
|
-
|
|
- opts = qemu_opts_create(create_opts, NULL, 0, &error_abort);
|
|
|
|
- size = blk_getlength(blk1);
|
|
- if (size < 0) {
|
|
- error_report("Failed to get size for '%s'", in.filename);
|
|
+ if (dd.flags & C_OSIZE) {
|
|
+ size = dd.osize;
|
|
+ } else if (dd.flags & C_IF) {
|
|
+ size = blk_getlength(blk1);
|
|
+ if (size < 0) {
|
|
+ error_report("Failed to get size for '%s'", in.filename);
|
|
+ ret = -1;
|
|
+ goto out;
|
|
+ }
|
|
+ } else if (dd.flags & C_COUNT) {
|
|
+ size = dd.count * in.bsz;
|
|
+ } else {
|
|
+ error_report("Output size must be known when reading from stdin");
|
|
ret = -1;
|
|
goto out;
|
|
}
|
|
|
|
- if (dd.flags & C_COUNT && dd.count <= INT64_MAX / in.bsz &&
|
|
+ if (!(dd.flags & C_OSIZE) && dd.flags & C_COUNT && dd.count <= INT64_MAX / in.bsz &&
|
|
dd.count * in.bsz < size) {
|
|
size = dd.count * in.bsz;
|
|
}
|
|
|
|
- /* Overflow means the specified offset is beyond input image's size */
|
|
- if (dd.flags & C_SKIP && (in.offset > INT64_MAX / in.bsz ||
|
|
- size < in.bsz * in.offset)) {
|
|
- qemu_opt_set_number(opts, BLOCK_OPT_SIZE, 0, &error_abort);
|
|
- } else {
|
|
- qemu_opt_set_number(opts, BLOCK_OPT_SIZE,
|
|
- size - in.bsz * in.offset, &error_abort);
|
|
- }
|
|
+ if (dd.flags & C_OF) {
|
|
+ drv = bdrv_find_format(out_fmt);
|
|
+ if (!drv) {
|
|
+ error_report("Unknown file format");
|
|
+ ret = -1;
|
|
+ goto out;
|
|
+ }
|
|
+ proto_drv = bdrv_find_protocol(out.filename, true, &local_err);
|
|
|
|
- ret = bdrv_create(drv, out.filename, opts, &local_err);
|
|
- if (ret < 0) {
|
|
- error_reportf_err(local_err,
|
|
- "%s: error while creating output image: ",
|
|
- out.filename);
|
|
- ret = -1;
|
|
- goto out;
|
|
- }
|
|
+ if (!proto_drv) {
|
|
+ error_report_err(local_err);
|
|
+ ret = -1;
|
|
+ goto out;
|
|
+ }
|
|
+ if (!drv->create_opts) {
|
|
+ error_report("Format driver '%s' does not support image creation",
|
|
+ drv->format_name);
|
|
+ ret = -1;
|
|
+ goto out;
|
|
+ }
|
|
+ if (!proto_drv->create_opts) {
|
|
+ error_report("Protocol driver '%s' does not support image creation",
|
|
+ proto_drv->format_name);
|
|
+ ret = -1;
|
|
+ goto out;
|
|
+ }
|
|
+ create_opts = qemu_opts_append(create_opts, drv->create_opts);
|
|
+ create_opts = qemu_opts_append(create_opts, proto_drv->create_opts);
|
|
|
|
- /* TODO, we can't honour --image-opts for the target,
|
|
- * since it needs to be given in a format compatible
|
|
- * with the bdrv_create() call above which does not
|
|
- * support image-opts style.
|
|
- */
|
|
- blk2 = img_open_file(out.filename, NULL, out_fmt, BDRV_O_RDWR,
|
|
- false, false, false);
|
|
+ opts = qemu_opts_create(create_opts, NULL, 0, &error_abort);
|
|
|
|
- if (!blk2) {
|
|
- ret = -1;
|
|
- goto out;
|
|
+ /* Overflow means the specified offset is beyond input image's size */
|
|
+ if (dd.flags & C_OSIZE) {
|
|
+ qemu_opt_set_number(opts, BLOCK_OPT_SIZE, size, &error_abort);
|
|
+ } else if (dd.flags & C_SKIP && (in.offset > INT64_MAX / in.bsz ||
|
|
+ size < in.bsz * in.offset)) {
|
|
+ qemu_opt_set_number(opts, BLOCK_OPT_SIZE, 0, &error_abort);
|
|
+ } else {
|
|
+ qemu_opt_set_number(opts, BLOCK_OPT_SIZE,
|
|
+ size - in.bsz * in.offset, &error_abort);
|
|
+ }
|
|
+
|
|
+ ret = bdrv_create(drv, out.filename, opts, &local_err);
|
|
+ if (ret < 0) {
|
|
+ error_reportf_err(local_err,
|
|
+ "%s: error while creating output image: ",
|
|
+ out.filename);
|
|
+ ret = -1;
|
|
+ goto out;
|
|
+ }
|
|
+
|
|
+ /* TODO, we can't honour --image-opts for the target,
|
|
+ * since it needs to be given in a format compatible
|
|
+ * with the bdrv_create() call above which does not
|
|
+ * support image-opts style.
|
|
+ */
|
|
+ blk2 = img_open_file(out.filename, NULL, out_fmt, BDRV_O_RDWR,
|
|
+ false, false, false);
|
|
+
|
|
+ if (!blk2) {
|
|
+ ret = -1;
|
|
+ goto out;
|
|
+ }
|
|
}
|
|
|
|
if (dd.flags & C_SKIP && (in.offset > INT64_MAX / in.bsz ||
|
|
@@ -5121,20 +5158,43 @@ static int img_dd(int argc, char **argv)
|
|
in.buf = g_new(uint8_t, in.bsz);
|
|
|
|
for (out_pos = 0; in_pos < size; block_count++) {
|
|
+ int in_ret, out_ret;
|
|
int bytes = (in_pos + in.bsz > size) ? size - in_pos : in.bsz;
|
|
-
|
|
- ret = blk_pread(blk1, in_pos, bytes, in.buf, 0);
|
|
- if (ret < 0) {
|
|
+ if (blk1) {
|
|
+ in_ret = blk_pread(blk1, in_pos, bytes, in.buf, 0);
|
|
+ if (in_ret == 0) {
|
|
+ in_ret = bytes;
|
|
+ }
|
|
+ } else {
|
|
+ in_ret = read(STDIN_FILENO, in.buf, bytes);
|
|
+ if (in_ret == 0) {
|
|
+ /* early EOF is considered an error */
|
|
+ error_report("Input ended unexpectedly");
|
|
+ ret = -1;
|
|
+ goto out;
|
|
+ }
|
|
+ }
|
|
+ if (in_ret < 0) {
|
|
error_report("error while reading from input image file: %s",
|
|
- strerror(-ret));
|
|
+ strerror(-in_ret));
|
|
+ ret = -1;
|
|
goto out;
|
|
}
|
|
in_pos += bytes;
|
|
|
|
- ret = blk_pwrite(blk2, out_pos, bytes, in.buf, 0);
|
|
- if (ret < 0) {
|
|
+ if (blk2) {
|
|
+ out_ret = blk_pwrite(blk2, out_pos, in_ret, in.buf, 0);
|
|
+ if (out_ret == 0) {
|
|
+ out_ret = in_ret;
|
|
+ }
|
|
+ } else {
|
|
+ out_ret = write(STDOUT_FILENO, in.buf, in_ret);
|
|
+ }
|
|
+
|
|
+ if (out_ret != in_ret) {
|
|
error_report("error while writing to output image file: %s",
|
|
- strerror(-ret));
|
|
+ strerror(-out_ret));
|
|
+ ret = -1;
|
|
goto out;
|
|
}
|
|
out_pos += bytes;
|