提交 9ea2ea71 编写于 作者: K Kevin Wolf 提交者: Anthony Liguori

Convert qemu-img create to new bdrv_create

This patch changes qemu-img to actually use the new bdrv_create interface. It
translates the old-style qemu-img options which have been bdrv_create2
parameters or flags so far to option structures. As the generic approach, it
introduces an -o option which accepts any parameter the driver knows.
Signed-off-by: NKevin Wolf <kwolf@redhat.com>
Signed-off-by: NAnthony Liguori <aliguori@us.ibm.com>
上级 0e7e1989
...@@ -22,6 +22,7 @@ ...@@ -22,6 +22,7 @@
* THE SOFTWARE. * THE SOFTWARE.
*/ */
#include "qemu-common.h" #include "qemu-common.h"
#include "qemu-option.h"
#include "osdep.h" #include "osdep.h"
#include "block_int.h" #include "block_int.h"
#include <stdio.h> #include <stdio.h>
...@@ -221,14 +222,13 @@ static int img_create(int argc, char **argv) ...@@ -221,14 +222,13 @@ static int img_create(int argc, char **argv)
const char *base_fmt = NULL; const char *base_fmt = NULL;
const char *filename; const char *filename;
const char *base_filename = NULL; const char *base_filename = NULL;
uint64_t size;
double sizef;
const char *p;
BlockDriver *drv; BlockDriver *drv;
QEMUOptionParameter *param = NULL;
char *options = NULL;
flags = 0; flags = 0;
for(;;) { for(;;) {
c = getopt(argc, argv, "F:b:f:he6"); c = getopt(argc, argv, "F:b:f:he6o:");
if (c == -1) if (c == -1)
break; break;
switch(c) { switch(c) {
...@@ -250,59 +250,100 @@ static int img_create(int argc, char **argv) ...@@ -250,59 +250,100 @@ static int img_create(int argc, char **argv)
case '6': case '6':
flags |= BLOCK_FLAG_COMPAT6; flags |= BLOCK_FLAG_COMPAT6;
break; break;
case 'o':
options = optarg;
break;
} }
} }
if (optind >= argc) if (optind >= argc)
help(); help();
filename = argv[optind++]; filename = argv[optind++];
size = 0;
if (base_filename) {
BlockDriverState *bs;
BlockDriver *base_drv = NULL;
if (base_fmt) { /* Find driver and parse its options */
base_drv = bdrv_find_format(base_fmt); drv = bdrv_find_format(fmt);
if (base_drv == NULL) if (!drv)
error("Unknown basefile format '%s'", base_fmt); error("Unknown file format '%s'", fmt);
}
bs = bdrv_new_open(base_filename, base_fmt); if (options) {
bdrv_get_geometry(bs, &size); param = parse_option_parameters(options, drv->create_options, param);
size *= 512; if (param == NULL) {
bdrv_delete(bs); error("Invalid options for file format '%s'.", fmt);
}
} else { } else {
if (optind >= argc) param = parse_option_parameters("", drv->create_options, param);
help(); }
p = argv[optind];
sizef = strtod(p, (char **)&p); /* Add size to parameters */
if (*p == 'M') { if (optind < argc) {
size = (uint64_t)(sizef * 1024 * 1024); set_option_parameter(param, BLOCK_OPT_SIZE, argv[optind++]);
} else if (*p == 'G') { }
size = (uint64_t)(sizef * 1024 * 1024 * 1024);
} else if (*p == 'k' || *p == 'K' || *p == '\0') { /* Add old-style options to parameters */
size = (uint64_t)(sizef * 1024); if (flags & BLOCK_FLAG_ENCRYPT) {
} else { if (set_option_parameter(param, BLOCK_OPT_ENCRYPT, "on")) {
help(); error("Encryption not supported for file format '%s'", fmt);
} }
} }
drv = bdrv_find_format(fmt); if (flags & BLOCK_FLAG_COMPAT6) {
if (!drv) if (set_option_parameter(param, BLOCK_OPT_COMPAT6, "on")) {
error("Unknown file format '%s'", fmt); error("VMDK version 6 not supported for file format '%s'", fmt);
printf("Formatting '%s', fmt=%s", }
filename, fmt); }
if (flags & BLOCK_FLAG_ENCRYPT)
printf(", encrypted");
if (flags & BLOCK_FLAG_COMPAT6)
printf(", compatibility level=6");
if (base_filename) { if (base_filename) {
printf(", backing_file=%s", if (set_option_parameter(param, BLOCK_OPT_BACKING_FILE, base_filename)) {
base_filename); error("Backing file not supported for file format '%s'", fmt);
if (base_fmt) }
printf(", backing_fmt=%s", }
base_fmt); if (base_fmt) {
} if (set_option_parameter(param, BLOCK_OPT_BACKING_FMT, base_fmt)) {
printf(", size=%" PRIu64 " kB\n", size / 1024); error("Backing file format not supported for file format '%s'", fmt);
ret = bdrv_create2(drv, filename, size / 512, base_filename, base_fmt, flags); }
}
// The size for the image must always be specified, with one exception:
// If we are using a backing file, we can obtain the size from there
if (get_option_parameter(param, BLOCK_OPT_SIZE)->value.n == 0) {
QEMUOptionParameter *backing_file =
get_option_parameter(param, BLOCK_OPT_BACKING_FILE);
QEMUOptionParameter *backing_fmt =
get_option_parameter(param, BLOCK_OPT_BACKING_FMT);
if (backing_file && backing_file->value.s) {
BlockDriverState *bs;
uint64_t size;
const char *fmt = NULL;
char buf[32];
if (backing_fmt && backing_fmt->value.s) {
if (bdrv_find_format(backing_fmt->value.s)) {
fmt = backing_fmt->value.s;
} else {
error("Unknown backing file format '%s'",
backing_fmt->value.s);
}
}
bs = bdrv_new_open(backing_file->value.s, fmt);
bdrv_get_geometry(bs, &size);
size *= 512;
bdrv_delete(bs);
snprintf(buf, sizeof(buf), "%" PRId64, size);
set_option_parameter(param, BLOCK_OPT_SIZE, buf);
} else {
error("Image creation needs a size parameter");
}
}
printf("Formatting '%s', fmt=%s ", filename, fmt);
print_option_parameters(param);
puts("");
ret = bdrv_create(drv, filename, param);
free_option_parameters(param);
if (ret < 0) { if (ret < 0) {
if (ret == -ENOTSUP) { if (ret == -ENOTSUP) {
error("Formatting or formatting option not supported for file format '%s'", fmt); error("Formatting or formatting option not supported for file format '%s'", fmt);
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册