block: Guard against NULL bs->drv
We currently do not guard everywhere against a NULL bs->drv where we should be doing so. Most of the places fixed here just do not care about that case at all. Some care implicitly, e.g. through a prior function call to bdrv_getlength() which would always fail for an ejected BDS. Add an assert there to make it more obvious. Other places seem to care, but do so insufficiently: Freeing clusters in a qcow2 image is an error-free operation, but it may leave the image in an unusable state anyway. Giving qcow2_free_clusters() an error code is not really viable, it is much easier to note that bs->drv may be NULL even after a successful driver call. This concerns bdrv_co_flush(), and the way the check is added to bdrv_co_pdiscard() (in every iteration instead of only once). Finally, some places employ at least an assert(bs->drv); somewhere, that may be reasonable (such as in the reopen code), but in bdrv_has_zero_init(), it is definitely not. Returning 0 there in case of an ejected BDS saves us much headache instead. Reported-by: R. Nageswara Sastry <nasastry@in.ibm.com> Buglink: https://bugs.launchpad.net/qemu/+bug/1728660 Signed-off-by: Max Reitz <mreitz@redhat.com> Message-id: 20171110203111.7666-4-mreitz@redhat.com Reviewed-by: Eric Blake <eblake@redhat.com> Signed-off-by: Max Reitz <mreitz@redhat.com>
This commit is contained in:
parent
93bbaf03ff
commit
d470ad42ac
19
block.c
19
block.c
@ -720,6 +720,10 @@ static int refresh_total_sectors(BlockDriverState *bs, int64_t hint)
|
|||||||
{
|
{
|
||||||
BlockDriver *drv = bs->drv;
|
BlockDriver *drv = bs->drv;
|
||||||
|
|
||||||
|
if (!drv) {
|
||||||
|
return -ENOMEDIUM;
|
||||||
|
}
|
||||||
|
|
||||||
/* Do not attempt drv->bdrv_getlength() on scsi-generic devices */
|
/* Do not attempt drv->bdrv_getlength() on scsi-generic devices */
|
||||||
if (bdrv_is_sg(bs))
|
if (bdrv_is_sg(bs))
|
||||||
return 0;
|
return 0;
|
||||||
@ -3431,6 +3435,10 @@ int bdrv_change_backing_file(BlockDriverState *bs,
|
|||||||
BlockDriver *drv = bs->drv;
|
BlockDriver *drv = bs->drv;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
if (!drv) {
|
||||||
|
return -ENOMEDIUM;
|
||||||
|
}
|
||||||
|
|
||||||
/* Backing file format doesn't make sense without a backing file */
|
/* Backing file format doesn't make sense without a backing file */
|
||||||
if (backing_fmt && !backing_file) {
|
if (backing_fmt && !backing_file) {
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
@ -3916,7 +3924,9 @@ int bdrv_has_zero_init_1(BlockDriverState *bs)
|
|||||||
|
|
||||||
int bdrv_has_zero_init(BlockDriverState *bs)
|
int bdrv_has_zero_init(BlockDriverState *bs)
|
||||||
{
|
{
|
||||||
assert(bs->drv);
|
if (!bs->drv) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/* If BS is a copy on write image, it is initialized to
|
/* If BS is a copy on write image, it is initialized to
|
||||||
the contents of the base image, which may not be zeroes. */
|
the contents of the base image, which may not be zeroes. */
|
||||||
@ -4256,6 +4266,10 @@ static int bdrv_inactivate_recurse(BlockDriverState *bs,
|
|||||||
BdrvChild *child, *parent;
|
BdrvChild *child, *parent;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
if (!bs->drv) {
|
||||||
|
return -ENOMEDIUM;
|
||||||
|
}
|
||||||
|
|
||||||
if (!setting_flag && bs->drv->bdrv_inactivate) {
|
if (!setting_flag && bs->drv->bdrv_inactivate) {
|
||||||
ret = bs->drv->bdrv_inactivate(bs);
|
ret = bs->drv->bdrv_inactivate(bs);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
@ -4790,6 +4804,9 @@ void bdrv_remove_aio_context_notifier(BlockDriverState *bs,
|
|||||||
int bdrv_amend_options(BlockDriverState *bs, QemuOpts *opts,
|
int bdrv_amend_options(BlockDriverState *bs, QemuOpts *opts,
|
||||||
BlockDriverAmendStatusCB *status_cb, void *cb_opaque)
|
BlockDriverAmendStatusCB *status_cb, void *cb_opaque)
|
||||||
{
|
{
|
||||||
|
if (!bs->drv) {
|
||||||
|
return -ENOMEDIUM;
|
||||||
|
}
|
||||||
if (!bs->drv->bdrv_amend_options) {
|
if (!bs->drv->bdrv_amend_options) {
|
||||||
return -ENOTSUP;
|
return -ENOTSUP;
|
||||||
}
|
}
|
||||||
|
36
block/io.c
36
block/io.c
@ -853,6 +853,10 @@ static int coroutine_fn bdrv_driver_preadv(BlockDriverState *bs,
|
|||||||
|
|
||||||
assert(!(flags & ~BDRV_REQ_MASK));
|
assert(!(flags & ~BDRV_REQ_MASK));
|
||||||
|
|
||||||
|
if (!drv) {
|
||||||
|
return -ENOMEDIUM;
|
||||||
|
}
|
||||||
|
|
||||||
if (drv->bdrv_co_preadv) {
|
if (drv->bdrv_co_preadv) {
|
||||||
return drv->bdrv_co_preadv(bs, offset, bytes, qiov, flags);
|
return drv->bdrv_co_preadv(bs, offset, bytes, qiov, flags);
|
||||||
}
|
}
|
||||||
@ -894,6 +898,10 @@ static int coroutine_fn bdrv_driver_pwritev(BlockDriverState *bs,
|
|||||||
|
|
||||||
assert(!(flags & ~BDRV_REQ_MASK));
|
assert(!(flags & ~BDRV_REQ_MASK));
|
||||||
|
|
||||||
|
if (!drv) {
|
||||||
|
return -ENOMEDIUM;
|
||||||
|
}
|
||||||
|
|
||||||
if (drv->bdrv_co_pwritev) {
|
if (drv->bdrv_co_pwritev) {
|
||||||
ret = drv->bdrv_co_pwritev(bs, offset, bytes, qiov,
|
ret = drv->bdrv_co_pwritev(bs, offset, bytes, qiov,
|
||||||
flags & bs->supported_write_flags);
|
flags & bs->supported_write_flags);
|
||||||
@ -945,6 +953,10 @@ bdrv_driver_pwritev_compressed(BlockDriverState *bs, uint64_t offset,
|
|||||||
{
|
{
|
||||||
BlockDriver *drv = bs->drv;
|
BlockDriver *drv = bs->drv;
|
||||||
|
|
||||||
|
if (!drv) {
|
||||||
|
return -ENOMEDIUM;
|
||||||
|
}
|
||||||
|
|
||||||
if (!drv->bdrv_co_pwritev_compressed) {
|
if (!drv->bdrv_co_pwritev_compressed) {
|
||||||
return -ENOTSUP;
|
return -ENOTSUP;
|
||||||
}
|
}
|
||||||
@ -975,6 +987,10 @@ static int coroutine_fn bdrv_co_do_copy_on_readv(BdrvChild *child,
|
|||||||
BDRV_REQUEST_MAX_BYTES);
|
BDRV_REQUEST_MAX_BYTES);
|
||||||
unsigned int progress = 0;
|
unsigned int progress = 0;
|
||||||
|
|
||||||
|
if (!drv) {
|
||||||
|
return -ENOMEDIUM;
|
||||||
|
}
|
||||||
|
|
||||||
/* FIXME We cannot require callers to have write permissions when all they
|
/* FIXME We cannot require callers to have write permissions when all they
|
||||||
* are doing is a read request. If we did things right, write permissions
|
* are doing is a read request. If we did things right, write permissions
|
||||||
* would be obtained anyway, but internally by the copy-on-read code. As
|
* would be obtained anyway, but internally by the copy-on-read code. As
|
||||||
@ -1291,6 +1307,10 @@ static int coroutine_fn bdrv_co_do_pwrite_zeroes(BlockDriverState *bs,
|
|||||||
bs->bl.request_alignment);
|
bs->bl.request_alignment);
|
||||||
int max_transfer = MIN_NON_ZERO(bs->bl.max_transfer, MAX_BOUNCE_BUFFER);
|
int max_transfer = MIN_NON_ZERO(bs->bl.max_transfer, MAX_BOUNCE_BUFFER);
|
||||||
|
|
||||||
|
if (!drv) {
|
||||||
|
return -ENOMEDIUM;
|
||||||
|
}
|
||||||
|
|
||||||
assert(alignment % bs->bl.request_alignment == 0);
|
assert(alignment % bs->bl.request_alignment == 0);
|
||||||
head = offset % alignment;
|
head = offset % alignment;
|
||||||
tail = (offset + bytes) % alignment;
|
tail = (offset + bytes) % alignment;
|
||||||
@ -1397,6 +1417,10 @@ static int coroutine_fn bdrv_aligned_pwritev(BdrvChild *child,
|
|||||||
uint64_t bytes_remaining = bytes;
|
uint64_t bytes_remaining = bytes;
|
||||||
int max_transfer;
|
int max_transfer;
|
||||||
|
|
||||||
|
if (!drv) {
|
||||||
|
return -ENOMEDIUM;
|
||||||
|
}
|
||||||
|
|
||||||
if (bdrv_has_readonly_bitmaps(bs)) {
|
if (bdrv_has_readonly_bitmaps(bs)) {
|
||||||
return -EPERM;
|
return -EPERM;
|
||||||
}
|
}
|
||||||
@ -1863,6 +1887,8 @@ static int coroutine_fn bdrv_co_block_status(BlockDriverState *bs,
|
|||||||
bytes = n;
|
bytes = n;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Must be non-NULL or bdrv_getlength() would have failed */
|
||||||
|
assert(bs->drv);
|
||||||
if (!bs->drv->bdrv_co_get_block_status) {
|
if (!bs->drv->bdrv_co_get_block_status) {
|
||||||
*pnum = bytes;
|
*pnum = bytes;
|
||||||
ret = BDRV_BLOCK_DATA | BDRV_BLOCK_ALLOCATED;
|
ret = BDRV_BLOCK_DATA | BDRV_BLOCK_ALLOCATED;
|
||||||
@ -2373,6 +2399,12 @@ int coroutine_fn bdrv_co_flush(BlockDriverState *bs)
|
|||||||
}
|
}
|
||||||
|
|
||||||
BLKDBG_EVENT(bs->file, BLKDBG_FLUSH_TO_DISK);
|
BLKDBG_EVENT(bs->file, BLKDBG_FLUSH_TO_DISK);
|
||||||
|
if (!bs->drv) {
|
||||||
|
/* bs->drv->bdrv_co_flush() might have ejected the BDS
|
||||||
|
* (even in case of apparent success) */
|
||||||
|
ret = -ENOMEDIUM;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
if (bs->drv->bdrv_co_flush_to_disk) {
|
if (bs->drv->bdrv_co_flush_to_disk) {
|
||||||
ret = bs->drv->bdrv_co_flush_to_disk(bs);
|
ret = bs->drv->bdrv_co_flush_to_disk(bs);
|
||||||
} else if (bs->drv->bdrv_aio_flush) {
|
} else if (bs->drv->bdrv_aio_flush) {
|
||||||
@ -2542,6 +2574,10 @@ int coroutine_fn bdrv_co_pdiscard(BlockDriverState *bs, int64_t offset,
|
|||||||
num = max_pdiscard;
|
num = max_pdiscard;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!bs->drv) {
|
||||||
|
ret = -ENOMEDIUM;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
if (bs->drv->bdrv_co_pdiscard) {
|
if (bs->drv->bdrv_co_pdiscard) {
|
||||||
ret = bs->drv->bdrv_co_pdiscard(bs, offset, num);
|
ret = bs->drv->bdrv_co_pdiscard(bs, offset, num);
|
||||||
} else {
|
} else {
|
||||||
|
@ -39,8 +39,14 @@ BlockDeviceInfo *bdrv_block_device_info(BlockBackend *blk,
|
|||||||
{
|
{
|
||||||
ImageInfo **p_image_info;
|
ImageInfo **p_image_info;
|
||||||
BlockDriverState *bs0;
|
BlockDriverState *bs0;
|
||||||
BlockDeviceInfo *info = g_malloc0(sizeof(*info));
|
BlockDeviceInfo *info;
|
||||||
|
|
||||||
|
if (!bs->drv) {
|
||||||
|
error_setg(errp, "Block device %s is ejected", bs->node_name);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
info = g_malloc0(sizeof(*info));
|
||||||
info->file = g_strdup(bs->filename);
|
info->file = g_strdup(bs->filename);
|
||||||
info->ro = bs->read_only;
|
info->ro = bs->read_only;
|
||||||
info->drv = g_strdup(bs->drv->format_name);
|
info->drv = g_strdup(bs->drv->format_name);
|
||||||
|
@ -342,12 +342,24 @@ static void secondary_do_checkpoint(BDRVReplicationState *s, Error **errp)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!s->active_disk->bs->drv) {
|
||||||
|
error_setg(errp, "Active disk %s is ejected",
|
||||||
|
s->active_disk->bs->node_name);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
ret = s->active_disk->bs->drv->bdrv_make_empty(s->active_disk->bs);
|
ret = s->active_disk->bs->drv->bdrv_make_empty(s->active_disk->bs);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
error_setg(errp, "Cannot make active disk empty");
|
error_setg(errp, "Cannot make active disk empty");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!s->hidden_disk->bs->drv) {
|
||||||
|
error_setg(errp, "Hidden disk %s is ejected",
|
||||||
|
s->hidden_disk->bs->node_name);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
ret = s->hidden_disk->bs->drv->bdrv_make_empty(s->hidden_disk->bs);
|
ret = s->hidden_disk->bs->drv->bdrv_make_empty(s->hidden_disk->bs);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
error_setg(errp, "Cannot make hidden disk empty");
|
error_setg(errp, "Cannot make hidden disk empty");
|
||||||
@ -511,6 +523,9 @@ static void replication_start(ReplicationState *rs, ReplicationMode mode,
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Must be true, or the bdrv_getlength() calls would have failed */
|
||||||
|
assert(s->active_disk->bs->drv && s->hidden_disk->bs->drv);
|
||||||
|
|
||||||
if (!s->active_disk->bs->drv->bdrv_make_empty ||
|
if (!s->active_disk->bs->drv->bdrv_make_empty ||
|
||||||
!s->hidden_disk->bs->drv->bdrv_make_empty) {
|
!s->hidden_disk->bs->drv->bdrv_make_empty) {
|
||||||
error_setg(errp,
|
error_setg(errp,
|
||||||
|
@ -2947,7 +2947,7 @@ static int do_commit(BDRVVVFATState* s)
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (s->qcow->bs->drv->bdrv_make_empty) {
|
if (s->qcow->bs->drv && s->qcow->bs->drv->bdrv_make_empty) {
|
||||||
s->qcow->bs->drv->bdrv_make_empty(s->qcow->bs);
|
s->qcow->bs->drv->bdrv_make_empty(s->qcow->bs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -337,6 +337,28 @@ $QEMU_IO -c "write 0 64k" "$TEST_IMG" | _filter_qemu_io
|
|||||||
|
|
||||||
# Can't repair this yet (TODO: We can just deallocate the cluster)
|
# Can't repair this yet (TODO: We can just deallocate the cluster)
|
||||||
|
|
||||||
|
echo
|
||||||
|
echo '=== Discarding with an unaligned refblock ==='
|
||||||
|
echo
|
||||||
|
|
||||||
|
_make_test_img 64M
|
||||||
|
|
||||||
|
$QEMU_IO -c "write 0 128k" "$TEST_IMG" | _filter_qemu_io
|
||||||
|
# Make our refblock unaligned
|
||||||
|
poke_file "$TEST_IMG" "$(($rt_offset))" "\x00\x00\x00\x00\x00\x00\x2a\x00"
|
||||||
|
# Now try to discard something that will be submitted as two requests
|
||||||
|
# (main part + tail)
|
||||||
|
$QEMU_IO -c "discard 0 65537" "$TEST_IMG"
|
||||||
|
|
||||||
|
echo '--- Repairing ---'
|
||||||
|
# Fails the first repair because the corruption prevents the check
|
||||||
|
# function from double-checking
|
||||||
|
# (Using -q for the first invocation, because otherwise the
|
||||||
|
# double-check error message appears above the summary for some
|
||||||
|
# reason -- so let's just hide the summary)
|
||||||
|
_check_test_img -q -r all
|
||||||
|
_check_test_img -r all
|
||||||
|
|
||||||
# success, all done
|
# success, all done
|
||||||
echo "*** done"
|
echo "*** done"
|
||||||
rm -f $seq.full
|
rm -f $seq.full
|
||||||
|
@ -317,4 +317,35 @@ discard 65536/65536 bytes at offset 0
|
|||||||
64 KiB, X ops; XX:XX:XX.X (XXX YYY/sec and XXX ops/sec)
|
64 KiB, X ops; XX:XX:XX.X (XXX YYY/sec and XXX ops/sec)
|
||||||
qcow2: Marking image as corrupt: Preallocated zero cluster offset 0x2a00 unaligned (guest offset: 0); further corruption events will be suppressed
|
qcow2: Marking image as corrupt: Preallocated zero cluster offset 0x2a00 unaligned (guest offset: 0); further corruption events will be suppressed
|
||||||
write failed: Input/output error
|
write failed: Input/output error
|
||||||
|
|
||||||
|
=== Discarding with an unaligned refblock ===
|
||||||
|
|
||||||
|
Formatting 'TEST_DIR/t.IMGFMT', fmt=IMGFMT size=67108864
|
||||||
|
wrote 131072/131072 bytes at offset 0
|
||||||
|
128 KiB, X ops; XX:XX:XX.X (XXX YYY/sec and XXX ops/sec)
|
||||||
|
qcow2: Marking image as corrupt: Refblock offset 0x2a00 unaligned (reftable index: 0); further corruption events will be suppressed
|
||||||
|
qcow2_free_clusters failed: Input/output error
|
||||||
|
discard failed: No medium found
|
||||||
|
--- Repairing ---
|
||||||
|
ERROR refcount block 0 is not cluster aligned; refcount table entry corrupted
|
||||||
|
qcow2: Marking image as corrupt: Refblock offset 0x2a00 unaligned (reftable index: 0); further corruption events will be suppressed
|
||||||
|
Can't get refcount for cluster 0: Input/output error
|
||||||
|
Can't get refcount for cluster 1: Input/output error
|
||||||
|
Can't get refcount for cluster 2: Input/output error
|
||||||
|
Can't get refcount for cluster 3: Input/output error
|
||||||
|
Can't get refcount for cluster 4: Input/output error
|
||||||
|
Can't get refcount for cluster 5: Input/output error
|
||||||
|
Can't get refcount for cluster 6: Input/output error
|
||||||
|
Rebuilding refcount structure
|
||||||
|
Repairing cluster 1 refcount=1 reference=0
|
||||||
|
qemu-img: Check failed: No medium found
|
||||||
|
Leaked cluster 1 refcount=1 reference=0
|
||||||
|
Repairing cluster 1 refcount=1 reference=0
|
||||||
|
The following inconsistencies were found and repaired:
|
||||||
|
|
||||||
|
1 leaked clusters
|
||||||
|
0 corruptions
|
||||||
|
|
||||||
|
Double checking the fixed image now...
|
||||||
|
No errors were found on the image.
|
||||||
*** done
|
*** done
|
||||||
|
Loading…
Reference in New Issue
Block a user