Block patches for 2.0.0-rc1

-----BEGIN PGP SIGNATURE-----
 Version: GnuPG v2.0.22 (GNU/Linux)
 
 iQIcBAABAgAGBQJTKZWfAAoJEH8JsnLIjy/WPGkP/1DibTfX6kXXwRNtt4GDsQFL
 1l5lyazUlYP9q5Y7nOWcGrshzCZUX0SmktOble1TlWWlKK7g0iPcSnC6uq7bxS3p
 oqg3bG6I8rKFmy1/YILwl+Dx8tbA5KarUW8Fv36GFDmR/jn9TEQAjAcJqqBty/lG
 vVu01rf1dv1qhM37Rd9TZDyXTLTaAE0UeXjnafcppN+P6XRlQkC9JvFWIq+uRWiY
 oPRgHnLkuffsWuHCt0qqlM4OewIM9f7axX7DQmUiWdRRbxZeWKxQG5E4Tws9RsUa
 QG7A2NDRmisyry9qaJMKMrnOwBBtJdAGkMqWEynSfY5pampIugNt9dy6A8OP5q9z
 3hmZWjulxN8ZfFn71ZM4nwC1MeR9WW/AiIOCIOr40u/5Nb+GhGSvEvRvewwKz7j9
 nCCIJKlRE6XP88shQMyxUfPKutYc4BeSCMOmpyDa8hVczljLsU8rBGFGWv4mmafM
 o7KWo5Lea2//Y6t28IacjZ2awSF3Yzc8e95JgxcMap2P/bWSQuC8IhxTj4JGqXgn
 nx7WV0gvr6N6/wPw3vbhdJzYOY1vdGYcqmlrZGSl9tYYd8iP7VFucsaiML1r8vT+
 Ye96EuhGVUo0poG0mQzCZSv3KjU9tlr4P+KDMGPEbDAEyFDywySjcHeWuiymHRGL
 BfNLCl95GuQN3y2iCkNo
 =W6Lu
 -----END PGP SIGNATURE-----

Merge remote-tracking branch 'remotes/kevin/tags/for-upstream' into staging

Block patches for 2.0.0-rc1

# gpg: Signature made Wed 19 Mar 2014 13:03:27 GMT using RSA key ID C88F2FD6
# gpg: Good signature from "Kevin Wolf <kwolf@redhat.com>"

* remotes/kevin/tags/for-upstream:
  dataplane: fix implicit IOThread refcount
  block/nfs: report errors from libnfs
  block/nfs: bump libnfs requirement to 1.9.3
  qcow2: Fix fail path in realloc_refcount_block()
  qcow2: Correct comment for realloc_refcount_block()
  qemu-io: Extended "--cmd" description in usage text
  qemu-io-cmds: Fixed typo in example for writev.
  block: Add error handling to bdrv_invalidate_cache()

Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
Peter Maydell 2014-03-19 13:47:22 +00:00
commit c1b94a0ed2
13 changed files with 107 additions and 41 deletions

28
block.c
View File

@ -4781,27 +4781,43 @@ flush_parent:
return bdrv_co_flush(bs->file); return bdrv_co_flush(bs->file);
} }
void bdrv_invalidate_cache(BlockDriverState *bs) void bdrv_invalidate_cache(BlockDriverState *bs, Error **errp)
{ {
Error *local_err = NULL;
int ret;
if (!bs->drv) { if (!bs->drv) {
return; return;
} }
if (bs->drv->bdrv_invalidate_cache) { if (bs->drv->bdrv_invalidate_cache) {
bs->drv->bdrv_invalidate_cache(bs); bs->drv->bdrv_invalidate_cache(bs, &local_err);
} else if (bs->file) { } else if (bs->file) {
bdrv_invalidate_cache(bs->file); bdrv_invalidate_cache(bs->file, &local_err);
}
if (local_err) {
error_propagate(errp, local_err);
return;
} }
refresh_total_sectors(bs, bs->total_sectors); ret = refresh_total_sectors(bs, bs->total_sectors);
if (ret < 0) {
error_setg_errno(errp, -ret, "Could not refresh total sector count");
return;
}
} }
void bdrv_invalidate_cache_all(void) void bdrv_invalidate_cache_all(Error **errp)
{ {
BlockDriverState *bs; BlockDriverState *bs;
Error *local_err = NULL;
QTAILQ_FOREACH(bs, &bdrv_states, device_list) { QTAILQ_FOREACH(bs, &bdrv_states, device_list) {
bdrv_invalidate_cache(bs); bdrv_invalidate_cache(bs, &local_err);
if (local_err) {
error_propagate(errp, local_err);
return;
}
} }
} }

View File

@ -112,6 +112,9 @@ nfs_co_generic_cb(int ret, struct nfs_context *nfs, void *data,
if (task->ret == 0 && task->st) { if (task->ret == 0 && task->st) {
memcpy(task->st, data, sizeof(struct stat)); memcpy(task->st, data, sizeof(struct stat));
} }
if (task->ret < 0) {
error_report("NFS Error: %s", nfs_get_error(nfs));
}
if (task->co) { if (task->co) {
task->bh = qemu_bh_new(nfs_co_generic_bh_cb, task); task->bh = qemu_bh_new(nfs_co_generic_bh_cb, task);
qemu_bh_schedule(task->bh); qemu_bh_schedule(task->bh);

View File

@ -1383,7 +1383,7 @@ static int write_reftable_entry(BlockDriverState *bs, int rt_index)
* does _not_ decrement the reference count for the currently occupied cluster. * does _not_ decrement the reference count for the currently occupied cluster.
* *
* This function prints an informative message to stderr on error (and returns * This function prints an informative message to stderr on error (and returns
* -errno); on success, 0 is returned. * -errno); on success, the offset of the newly allocated cluster is returned.
*/ */
static int64_t realloc_refcount_block(BlockDriverState *bs, int reftable_index, static int64_t realloc_refcount_block(BlockDriverState *bs, int reftable_index,
uint64_t offset) uint64_t offset)
@ -1399,14 +1399,14 @@ static int64_t realloc_refcount_block(BlockDriverState *bs, int reftable_index,
fprintf(stderr, "Could not allocate new cluster: %s\n", fprintf(stderr, "Could not allocate new cluster: %s\n",
strerror(-new_offset)); strerror(-new_offset));
ret = new_offset; ret = new_offset;
goto fail; goto done;
} }
/* fetch current refcount block content */ /* fetch current refcount block content */
ret = qcow2_cache_get(bs, s->refcount_block_cache, offset, &refcount_block); ret = qcow2_cache_get(bs, s->refcount_block_cache, offset, &refcount_block);
if (ret < 0) { if (ret < 0) {
fprintf(stderr, "Could not fetch refcount block: %s\n", strerror(-ret)); fprintf(stderr, "Could not fetch refcount block: %s\n", strerror(-ret));
goto fail; goto fail_free_cluster;
} }
/* new block has not yet been entered into refcount table, therefore it is /* new block has not yet been entered into refcount table, therefore it is
@ -1417,8 +1417,7 @@ static int64_t realloc_refcount_block(BlockDriverState *bs, int reftable_index,
"check failed: %s\n", strerror(-ret)); "check failed: %s\n", strerror(-ret));
/* the image will be marked corrupt, so don't even attempt on freeing /* the image will be marked corrupt, so don't even attempt on freeing
* the cluster */ * the cluster */
new_offset = 0; goto done;
goto fail;
} }
/* write to new block */ /* write to new block */
@ -1426,7 +1425,7 @@ static int64_t realloc_refcount_block(BlockDriverState *bs, int reftable_index,
s->cluster_sectors); s->cluster_sectors);
if (ret < 0) { if (ret < 0) {
fprintf(stderr, "Could not write refcount block: %s\n", strerror(-ret)); fprintf(stderr, "Could not write refcount block: %s\n", strerror(-ret));
goto fail; goto fail_free_cluster;
} }
/* update refcount table */ /* update refcount table */
@ -1436,24 +1435,27 @@ static int64_t realloc_refcount_block(BlockDriverState *bs, int reftable_index,
if (ret < 0) { if (ret < 0) {
fprintf(stderr, "Could not update refcount table: %s\n", fprintf(stderr, "Could not update refcount table: %s\n",
strerror(-ret)); strerror(-ret));
goto fail; goto fail_free_cluster;
} }
fail: goto done;
if (new_offset && (ret < 0)) {
qcow2_free_clusters(bs, new_offset, s->cluster_size, fail_free_cluster:
QCOW2_DISCARD_ALWAYS); qcow2_free_clusters(bs, new_offset, s->cluster_size, QCOW2_DISCARD_OTHER);
}
done:
if (refcount_block) { if (refcount_block) {
if (ret < 0) { /* This should never fail, as it would only do so if the given refcount
qcow2_cache_put(bs, s->refcount_block_cache, &refcount_block); * block cannot be found in the cache. As this is impossible as long as
} else { * there are no bugs, assert the success. */
ret = qcow2_cache_put(bs, s->refcount_block_cache, &refcount_block); int tmp = qcow2_cache_put(bs, s->refcount_block_cache, &refcount_block);
} assert(tmp == 0);
} }
if (ret < 0) { if (ret < 0) {
return ret; return ret;
} }
return new_offset; return new_offset;
} }

View File

@ -1156,7 +1156,7 @@ static void qcow2_close(BlockDriverState *bs)
qcow2_free_snapshots(bs); qcow2_free_snapshots(bs);
} }
static void qcow2_invalidate_cache(BlockDriverState *bs) static void qcow2_invalidate_cache(BlockDriverState *bs, Error **errp)
{ {
BDRVQcowState *s = bs->opaque; BDRVQcowState *s = bs->opaque;
int flags = s->flags; int flags = s->flags;
@ -1164,6 +1164,8 @@ static void qcow2_invalidate_cache(BlockDriverState *bs)
AES_KEY aes_decrypt_key; AES_KEY aes_decrypt_key;
uint32_t crypt_method = 0; uint32_t crypt_method = 0;
QDict *options; QDict *options;
Error *local_err = NULL;
int ret;
/* /*
* Backing files are read-only which makes all of their metadata immutable, * Backing files are read-only which makes all of their metadata immutable,
@ -1178,11 +1180,25 @@ static void qcow2_invalidate_cache(BlockDriverState *bs)
qcow2_close(bs); qcow2_close(bs);
bdrv_invalidate_cache(bs->file); bdrv_invalidate_cache(bs->file, &local_err);
if (local_err) {
error_propagate(errp, local_err);
return;
}
memset(s, 0, sizeof(BDRVQcowState)); memset(s, 0, sizeof(BDRVQcowState));
options = qdict_clone_shallow(bs->options); options = qdict_clone_shallow(bs->options);
qcow2_open(bs, options, flags, NULL);
ret = qcow2_open(bs, options, flags, &local_err);
if (local_err) {
error_setg(errp, "Could not reopen qcow2 layer: %s",
error_get_pretty(local_err));
error_free(local_err);
return;
} else if (ret < 0) {
error_setg_errno(errp, -ret, "Could not reopen qcow2 layer");
return;
}
QDECREF(options); QDECREF(options);

View File

@ -1558,16 +1558,31 @@ static int bdrv_qed_change_backing_file(BlockDriverState *bs,
return ret; return ret;
} }
static void bdrv_qed_invalidate_cache(BlockDriverState *bs) static void bdrv_qed_invalidate_cache(BlockDriverState *bs, Error **errp)
{ {
BDRVQEDState *s = bs->opaque; BDRVQEDState *s = bs->opaque;
Error *local_err = NULL;
int ret;
bdrv_qed_close(bs); bdrv_qed_close(bs);
bdrv_invalidate_cache(bs->file); bdrv_invalidate_cache(bs->file, &local_err);
if (local_err) {
error_propagate(errp, local_err);
return;
}
memset(s, 0, sizeof(BDRVQEDState)); memset(s, 0, sizeof(BDRVQEDState));
bdrv_qed_open(bs, NULL, bs->open_flags, NULL); ret = bdrv_qed_open(bs, NULL, bs->open_flags, &local_err);
if (local_err) {
error_setg(errp, "Could not reopen qed layer: %s",
error_get_pretty(local_err));
error_free(local_err);
return;
} else if (ret < 0) {
error_setg_errno(errp, -ret, "Could not reopen qed layer");
return;
}
} }
static int bdrv_qed_check(BlockDriverState *bs, BdrvCheckResult *result, static int bdrv_qed_check(BlockDriverState *bs, BdrvCheckResult *result,

View File

@ -625,13 +625,18 @@ static int64_t quorum_getlength(BlockDriverState *bs)
return result; return result;
} }
static void quorum_invalidate_cache(BlockDriverState *bs) static void quorum_invalidate_cache(BlockDriverState *bs, Error **errp)
{ {
BDRVQuorumState *s = bs->opaque; BDRVQuorumState *s = bs->opaque;
Error *local_err = NULL;
int i; int i;
for (i = 0; i < s->num_children; i++) { for (i = 0; i < s->num_children; i++) {
bdrv_invalidate_cache(s->bs[i]); bdrv_invalidate_cache(s->bs[i], &local_err);
if (local_err) {
error_propagate(errp, local_err);
return;
}
} }
} }

2
configure vendored
View File

@ -3868,7 +3868,7 @@ fi
########################################## ##########################################
# Do we have libnfs # Do we have libnfs
if test "$libnfs" != "no" ; then if test "$libnfs" != "no" ; then
if $pkg_config --atleast-version=1.9.2 libnfs; then if $pkg_config --atleast-version=1.9.3 libnfs; then
libnfs="yes" libnfs="yes"
libnfs_libs=$($pkg_config --libs libnfs) libnfs_libs=$($pkg_config --libs libnfs)
LIBS="$LIBS $libnfs_libs" LIBS="$LIBS $libnfs_libs"

View File

@ -393,7 +393,6 @@ void virtio_blk_data_plane_create(VirtIODevice *vdev, VirtIOBlkConf *blk,
if (blk->iothread) { if (blk->iothread) {
s->internal_iothread = false; s->internal_iothread = false;
s->iothread = blk->iothread; s->iothread = blk->iothread;
object_ref(OBJECT(s->iothread));
} else { } else {
/* Create per-device IOThread if none specified */ /* Create per-device IOThread if none specified */
Error *local_err = NULL; Error *local_err = NULL;
@ -408,6 +407,7 @@ void virtio_blk_data_plane_create(VirtIODevice *vdev, VirtIOBlkConf *blk,
s->iothread = iothread_find(vdev->name); s->iothread = iothread_find(vdev->name);
assert(s->iothread); assert(s->iothread);
} }
object_ref(OBJECT(s->iothread));
s->ctx = iothread_get_aio_context(s->iothread); s->ctx = iothread_get_aio_context(s->iothread);
/* Prevent block operations that conflict with data plane thread */ /* Prevent block operations that conflict with data plane thread */

View File

@ -329,8 +329,8 @@ BlockDriverAIOCB *bdrv_aio_ioctl(BlockDriverState *bs,
BlockDriverCompletionFunc *cb, void *opaque); BlockDriverCompletionFunc *cb, void *opaque);
/* Invalidate any cached metadata used by image formats */ /* Invalidate any cached metadata used by image formats */
void bdrv_invalidate_cache(BlockDriverState *bs); void bdrv_invalidate_cache(BlockDriverState *bs, Error **errp);
void bdrv_invalidate_cache_all(void); void bdrv_invalidate_cache_all(Error **errp);
void bdrv_clear_incoming_migration_all(void); void bdrv_clear_incoming_migration_all(void);

View File

@ -153,7 +153,7 @@ struct BlockDriver {
/* /*
* Invalidate any cached meta-data. * Invalidate any cached meta-data.
*/ */
void (*bdrv_invalidate_cache)(BlockDriverState *bs); void (*bdrv_invalidate_cache)(BlockDriverState *bs, Error **errp);
/* /*
* Flushes all data that was already written to the OS all the way down to * Flushes all data that was already written to the OS all the way down to

View File

@ -101,6 +101,7 @@ void qemu_start_incoming_migration(const char *uri, Error **errp)
static void process_incoming_migration_co(void *opaque) static void process_incoming_migration_co(void *opaque)
{ {
QEMUFile *f = opaque; QEMUFile *f = opaque;
Error *local_err = NULL;
int ret; int ret;
ret = qemu_loadvm_state(f); ret = qemu_loadvm_state(f);
@ -115,7 +116,12 @@ static void process_incoming_migration_co(void *opaque)
bdrv_clear_incoming_migration_all(); bdrv_clear_incoming_migration_all();
/* Make sure all file formats flush their mutable metadata */ /* Make sure all file formats flush their mutable metadata */
bdrv_invalidate_cache_all(); bdrv_invalidate_cache_all(&local_err);
if (local_err) {
qerror_report_err(local_err);
error_free(local_err);
exit(EXIT_FAILURE);
}
if (autostart) { if (autostart) {
vm_start(); vm_start();

View File

@ -1087,7 +1087,7 @@ writev_help(void)
" writes a range of bytes from the given offset source from multiple buffers\n" " writes a range of bytes from the given offset source from multiple buffers\n"
"\n" "\n"
" Example:\n" " Example:\n"
" 'write 512 1k 1k' - writes 2 kilobytes at 512 bytes into the open file\n" " 'writev 512 1k 1k' - writes 2 kilobytes at 512 bytes into the open file\n"
"\n" "\n"
" Writes into a segment of the currently open file, using a buffer\n" " Writes into a segment of the currently open file, using a buffer\n"
" filled with a set pattern (0xcdcdcdcd).\n" " filled with a set pattern (0xcdcdcdcd).\n"

View File

@ -193,10 +193,11 @@ static const cmdinfo_t quit_cmd = {
static void usage(const char *name) static void usage(const char *name)
{ {
printf( printf(
"Usage: %s [-h] [-V] [-rsnm] [-c cmd] ... [file]\n" "Usage: %s [-h] [-V] [-rsnm] [-c STRING] ... [file]\n"
"QEMU Disk exerciser\n" "QEMU Disk exerciser\n"
"\n" "\n"
" -c, --cmd command to execute\n" " -c, --cmd STRING execute command with its arguments\n"
" from the given string\n"
" -r, --read-only export read-only\n" " -r, --read-only export read-only\n"
" -s, --snapshot use snapshot file\n" " -s, --snapshot use snapshot file\n"
" -n, --nocache disable host cache\n" " -n, --nocache disable host cache\n"
@ -207,8 +208,10 @@ static void usage(const char *name)
" -T, --trace FILE enable trace events listed in the given file\n" " -T, --trace FILE enable trace events listed in the given file\n"
" -h, --help display this help and exit\n" " -h, --help display this help and exit\n"
" -V, --version output version information and exit\n" " -V, --version output version information and exit\n"
"\n"
"See '%s -c help' for information on available commands."
"\n", "\n",
name); name, name);
} }
static char *get_prompt(void) static char *get_prompt(void)