From: "Benoît Canet" <benoit.canet@nodalink.com>
To: Max Reitz <mreitz@redhat.com>
Cc: "Kevin Wolf" <kwolf@redhat.com>,
qemu-devel@nongnu.org, "Stefan Hajnoczi" <stefanha@redhat.com>,
"Benoît Canet" <benoit.canet@nodalink.com>
Subject: Re: [Qemu-devel] [PATCH v3 02/10] qcow2: Split qcow2_check_refcounts()
Date: Fri, 22 Aug 2014 18:03:41 +0000 [thread overview]
Message-ID: <20140822180341.GC9526@nodalink.com> (raw)
In-Reply-To: <1408725104-17176-3-git-send-email-mreitz@redhat.com>
On Fri, Aug 22, 2014 at 06:31:36PM +0200, Max Reitz wrote:
> Put the code for calculating the reference counts and comparing them
> during qemu-img check into own functions.
>
> Signed-off-by: Max Reitz <mreitz@redhat.com>
> ---
> block/qcow2-refcount.c | 153 ++++++++++++++++++++++++++++++++-----------------
> 1 file changed, 102 insertions(+), 51 deletions(-)
>
> diff --git a/block/qcow2-refcount.c b/block/qcow2-refcount.c
> index 43665b8..5f0920b 100644
> --- a/block/qcow2-refcount.c
> +++ b/block/qcow2-refcount.c
> @@ -1496,71 +1496,70 @@ done:
> return new_offset;
> }
>
> +static int check_refblocks(BlockDriverState *bs, BdrvCheckResult *res,
> + BdrvCheckMode fix, uint16_t **refcount_table,
> + int64_t *nb_clusters);
> +
> /*
> - * Checks an image for refcount consistency.
> - *
> - * Returns 0 if no errors are found, the number of errors in case the image is
> - * detected as corrupted, and -errno when an internal error occurred.
> + * Calculates an in-memory refcount table.
> */
> -int qcow2_check_refcounts(BlockDriverState *bs, BdrvCheckResult *res,
> - BdrvCheckMode fix)
> +static int calculate_refcounts(BlockDriverState *bs, BdrvCheckResult *res,
> + BdrvCheckMode fix, uint16_t **refcount_table,
> + int64_t *nb_clusters)
> {
> BDRVQcowState *s = bs->opaque;
> - int64_t size, i, highest_cluster, nb_clusters;
> - int refcount1, refcount2;
> + int64_t i;
> QCowSnapshot *sn;
> - uint16_t *refcount_table;
> int ret;
>
> - size = bdrv_getlength(bs->file);
> - if (size < 0) {
> - res->check_errors++;
> - return size;
> - }
> -
> - nb_clusters = size_to_clusters(s, size);
> - if (nb_clusters > INT_MAX) {
> - res->check_errors++;
> - return -EFBIG;
> - }
> -
> - refcount_table = g_try_new0(uint16_t, nb_clusters);
> - if (nb_clusters && refcount_table == NULL) {
> + *refcount_table = g_try_new0(uint16_t, *nb_clusters);
> + if (*nb_clusters && *refcount_table == NULL) {
> res->check_errors++;
> return -ENOMEM;
> }
>
> - res->bfi.total_clusters =
> - size_to_clusters(s, bs->total_sectors * BDRV_SECTOR_SIZE);
> -
> /* header */
> - inc_refcounts(bs, res, refcount_table, nb_clusters,
> + inc_refcounts(bs, res, *refcount_table, *nb_clusters,
> 0, s->cluster_size);
>
> /* current L1 table */
> - ret = check_refcounts_l1(bs, res, refcount_table, nb_clusters,
> + ret = check_refcounts_l1(bs, res, *refcount_table, *nb_clusters,
> s->l1_table_offset, s->l1_size, CHECK_FRAG_INFO);
> if (ret < 0) {
> - goto fail;
> + return ret;
> }
>
> /* snapshots */
> - for(i = 0; i < s->nb_snapshots; i++) {
> + for (i = 0; i < s->nb_snapshots; i++) {
> sn = s->snapshots + i;
> - ret = check_refcounts_l1(bs, res, refcount_table, nb_clusters,
> + ret = check_refcounts_l1(bs, res, *refcount_table, *nb_clusters,
> sn->l1_table_offset, sn->l1_size, 0);
> if (ret < 0) {
> - goto fail;
> + return ret;
> }
> }
> - inc_refcounts(bs, res, refcount_table, nb_clusters,
> + inc_refcounts(bs, res, *refcount_table, *nb_clusters,
> s->snapshots_offset, s->snapshots_size);
>
> /* refcount data */
> - inc_refcounts(bs, res, refcount_table, nb_clusters,
> + inc_refcounts(bs, res, *refcount_table, *nb_clusters,
> s->refcount_table_offset,
> s->refcount_table_size * sizeof(uint64_t));
>
> + return check_refblocks(bs, res, fix, refcount_table, nb_clusters);
> +}
> +
> +/*
> + * Checks consistency of refblocks and accounts for each refblock in
> + * *refcount_table.
> + */
> +static int check_refblocks(BlockDriverState *bs, BdrvCheckResult *res,
> + BdrvCheckMode fix, uint16_t **refcount_table,
> + int64_t *nb_clusters)
> +{
> + BDRVQcowState *s = bs->opaque;
> + int64_t i;
> +
> for(i = 0; i < s->refcount_table_size; i++) {
> uint64_t offset, cluster;
> offset = s->refcount_table[i];
> @@ -1574,7 +1573,7 @@ int qcow2_check_refcounts(BlockDriverState *bs, BdrvCheckResult *res,
> continue;
> }
>
> - if (cluster >= nb_clusters) {
> + if (cluster >= *nb_clusters) {
> fprintf(stderr, "ERROR refcount block %" PRId64
> " is outside image\n", i);
> res->corruptions++;
> @@ -1582,14 +1581,14 @@ int qcow2_check_refcounts(BlockDriverState *bs, BdrvCheckResult *res,
> }
>
> if (offset != 0) {
> - inc_refcounts(bs, res, refcount_table, nb_clusters,
> + inc_refcounts(bs, res, *refcount_table, *nb_clusters,
> offset, s->cluster_size);
> - if (refcount_table[cluster] != 1) {
> + if ((*refcount_table)[cluster] != 1) {
> fprintf(stderr, "%s refcount block %" PRId64
> " refcount=%d\n",
> fix & BDRV_FIX_ERRORS ? "Repairing" :
> "ERROR",
> - i, refcount_table[cluster]);
> + i, (*refcount_table)[cluster]);
>
> if (fix & BDRV_FIX_ERRORS) {
> int64_t new_offset;
> @@ -1601,17 +1600,18 @@ int qcow2_check_refcounts(BlockDriverState *bs, BdrvCheckResult *res,
> }
>
> /* update refcounts */
> - if ((new_offset >> s->cluster_bits) >= nb_clusters) {
> + if ((new_offset >> s->cluster_bits) >= *nb_clusters) {
> /* increase refcount_table size if necessary */
> - int old_nb_clusters = nb_clusters;
> - nb_clusters = (new_offset >> s->cluster_bits) + 1;
> - refcount_table = g_renew(uint16_t, refcount_table,
> - nb_clusters);
> - memset(&refcount_table[old_nb_clusters], 0, (nb_clusters
> - - old_nb_clusters) * sizeof(uint16_t));
> + int old_nb_clusters = *nb_clusters;
> + *nb_clusters = (new_offset >> s->cluster_bits) + 1;
> + *refcount_table = g_renew(uint16_t, *refcount_table,
> + *nb_clusters);
> + memset(&(*refcount_table)[old_nb_clusters], 0,
> + (*nb_clusters - old_nb_clusters) *
> + sizeof(uint16_t));
> }
> - refcount_table[cluster]--;
> - inc_refcounts(bs, res, refcount_table, nb_clusters,
> + (*refcount_table)[cluster]--;
> + inc_refcounts(bs, res, *refcount_table, *nb_clusters,
> new_offset, s->cluster_size);
>
> res->corruptions_fixed++;
> @@ -1622,8 +1622,22 @@ int qcow2_check_refcounts(BlockDriverState *bs, BdrvCheckResult *res,
> }
> }
>
> - /* compare ref counts */
> - for (i = 0, highest_cluster = 0; i < nb_clusters; i++) {
> + return 0;
> +}
> +
> +/*
> + * Compares the actual reference count for each cluster in the image against the
> + * refcount as reported by the refcount structures on-disk.
> + */
> +static void compare_refcounts(BlockDriverState *bs, BdrvCheckResult *res,
> + BdrvCheckMode fix, int64_t *highest_cluster,
> + uint16_t *refcount_table, int64_t nb_clusters)
> +{
> + BDRVQcowState *s = bs->opaque;
> + int64_t i;
> + int refcount1, refcount2, ret;
> +
> + for (i = 0, *highest_cluster = 0; i < nb_clusters; i++) {
> refcount1 = get_refcount(bs, i);
> if (refcount1 < 0) {
> fprintf(stderr, "Can't get refcount for cluster %" PRId64 ": %s\n",
> @@ -1635,11 +1649,10 @@ int qcow2_check_refcounts(BlockDriverState *bs, BdrvCheckResult *res,
> refcount2 = refcount_table[i];
>
> if (refcount1 > 0 || refcount2 > 0) {
> - highest_cluster = i;
> + *highest_cluster = i;
> }
>
> if (refcount1 != refcount2) {
> -
> /* Check if we're allowed to fix the mismatch */
> int *num_fixed = NULL;
> if (refcount1 > refcount2 && (fix & BDRV_FIX_LEAKS)) {
> @@ -1672,6 +1685,44 @@ int qcow2_check_refcounts(BlockDriverState *bs, BdrvCheckResult *res,
> }
> }
> }
> +}
> +
> +/*
> + * Checks an image for refcount consistency.
> + *
> + * Returns 0 if no errors are found, the number of errors in case the image is
> + * detected as corrupted, and -errno when an internal error occurred.
> + */
> +int qcow2_check_refcounts(BlockDriverState *bs, BdrvCheckResult *res,
> + BdrvCheckMode fix)
> +{
> + BDRVQcowState *s = bs->opaque;
> + int64_t size, highest_cluster, nb_clusters;
> + uint16_t *refcount_table;
> + int ret;
> +
> + size = bdrv_getlength(bs->file);
> + if (size < 0) {
> + res->check_errors++;
> + return size;
> + }
> +
> + nb_clusters = size_to_clusters(s, size);
> + if (nb_clusters > INT_MAX) {
> + res->check_errors++;
> + return -EFBIG;
> + }
> +
> + res->bfi.total_clusters =
> + size_to_clusters(s, bs->total_sectors * BDRV_SECTOR_SIZE);
> +
> + ret = calculate_refcounts(bs, res, fix, &refcount_table, &nb_clusters);
> + if (ret < 0) {
> + goto fail;
> + }
> +
> + compare_refcounts(bs, res, fix, &highest_cluster, refcount_table,
> + nb_clusters);
>
> /* check OFLAG_COPIED */
> ret = check_oflag_copied(bs, res, fix);
> --
> 2.0.4
>
Thanks it's nicer to read.
My opinion changed from "It must be good Max took care of it" to "I did not find any bug" :)
Reviewed-by: Benoît Canet <benoit.canet@nodalink.com>
next prev parent reply other threads:[~2014-08-22 18:03 UTC|newest]
Thread overview: 28+ messages / expand[flat|nested] mbox.gz Atom feed top
2014-08-22 16:31 [Qemu-devel] [PATCH v3 00/10] qcow2: Fix image repairing Max Reitz
2014-08-22 16:31 ` [Qemu-devel] [PATCH v3 01/10] qcow2: Fix leaks in dirty images Max Reitz
2014-08-22 18:44 ` Benoît Canet
2014-08-22 16:31 ` [Qemu-devel] [PATCH v3 02/10] qcow2: Split qcow2_check_refcounts() Max Reitz
2014-08-22 18:03 ` Benoît Canet [this message]
2014-08-22 16:31 ` [Qemu-devel] [PATCH v3 03/10] qcow2: Pull check_refblocks() up Max Reitz
2014-08-22 18:04 ` Benoît Canet
2014-08-22 16:31 ` [Qemu-devel] [PATCH v3 04/10] qcow2: Reuse refcount table in calculate_refcounts() Max Reitz
2014-08-22 18:07 ` Benoît Canet
2014-08-22 16:31 ` [Qemu-devel] [PATCH v3 05/10] qcow2: Fix refcount blocks beyond image end Max Reitz
2014-08-22 18:20 ` Benoît Canet
2014-08-26 13:07 ` Eric Blake
2014-08-26 18:06 ` Max Reitz
2014-08-22 16:31 ` [Qemu-devel] [PATCH v3 06/10] qcow2: Do not perform potentially damaging repairs Max Reitz
2014-08-22 18:25 ` Benoît Canet
2014-08-22 16:31 ` [Qemu-devel] [PATCH v3 07/10] qcow2: Rebuild refcount structure during check Max Reitz
2014-08-25 17:40 ` Benoît Canet
2014-08-27 18:37 ` Max Reitz
2014-08-22 16:31 ` [Qemu-devel] [PATCH v3 08/10] qcow2: Clean up after refcount rebuild Max Reitz
2014-08-22 18:43 ` Benoît Canet
2014-08-22 16:31 ` [Qemu-devel] [PATCH v3 09/10] iotests: Fix test outputs Max Reitz
2014-08-26 13:09 ` Eric Blake
2014-08-26 18:03 ` Max Reitz
2014-08-22 16:31 ` [Qemu-devel] [PATCH v3 10/10] iotests: Add test for potentially damaging repairs Max Reitz
2014-08-22 18:50 ` Benoît Canet
2014-08-22 19:55 ` Eric Blake
2014-08-22 20:55 ` Benoît Canet
2014-08-26 17:53 ` Max Reitz
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20140822180341.GC9526@nodalink.com \
--to=benoit.canet@nodalink.com \
--cc=kwolf@redhat.com \
--cc=mreitz@redhat.com \
--cc=qemu-devel@nongnu.org \
--cc=stefanha@redhat.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.