From: Kevin Wolf <kwolf@redhat.com>
To: Fam Zheng <famcool@gmail.com>
Cc: stefanha@gmail.com, qemu-devel@nongnu.org, hch@lst.de
Subject: Re: [Qemu-devel] [PATCH v3 2/8] VMDK: add twoGbMaxExtentSparse support
Date: Tue, 06 Sep 2011 12:38:08 +0200 [thread overview]
Message-ID: <4E65F810.9080701@redhat.com> (raw)
In-Reply-To: <1313162374-13147-3-git-send-email-famcool@gmail.com>
Am 12.08.2011 17:19, schrieb Fam Zheng:
> Add twoGbMaxExtentSparse support. Introduce vmdk_free_last_extent.
>
> Signed-off-by: Fam Zheng <famcool@gmail.com>
> ---
> block/vmdk.c | 133 ++++++++++++++++++++++++++++++++++++----------------------
> 1 files changed, 83 insertions(+), 50 deletions(-)
>
> diff --git a/block/vmdk.c b/block/vmdk.c
> index e22a893..ab15840 100644
> --- a/block/vmdk.c
> +++ b/block/vmdk.c
> @@ -174,6 +174,17 @@ static void vmdk_free_extents(BlockDriverState *bs)
> qemu_free(s->extents);
> }
>
> +static void vmdk_free_last_extent(BlockDriverState *bs)
> +{
> + BDRVVmdkState *s = bs->opaque;
> +
> + if (s->num_extents == 0) {
> + return;
> + }
> + s->num_extents--;
> + s->extents = qemu_realloc(s->extents, s->num_extents * sizeof(VmdkExtent));
> +}
> +
> static uint32_t vmdk_read_cid(BlockDriverState *bs, int parent)
> {
> char desc[DESC_SIZE];
> @@ -357,18 +368,18 @@ static int vmdk_init_tables(BlockDriverState *bs, VmdkExtent *extent)
> return ret;
> }
>
> -static int vmdk_open_vmdk3(BlockDriverState *bs, int flags)
> +static int vmdk_open_vmdk3(BlockDriverState *bs,
> + BlockDriverState *file,
> + int flags)
> {
> int ret;
> uint32_t magic;
> VMDK3Header header;
> - BDRVVmdkState *s = bs->opaque;
> VmdkExtent *extent;
>
> - s->desc_offset = 0x200;
> - ret = bdrv_pread(bs->file, sizeof(magic), &header, sizeof(header));
> + ret = bdrv_pread(file, sizeof(magic), &header, sizeof(header));
> if (ret < 0) {
> - goto fail;
> + return ret;
> }
> extent = vmdk_add_extent(bs,
> bs->file, false,
> @@ -378,58 +389,45 @@ static int vmdk_open_vmdk3(BlockDriverState *bs, int flags)
> le32_to_cpu(header.granularity));
> ret = vmdk_init_tables(bs, extent);
> if (ret) {
> - /* vmdk_init_tables cleans up on fail, so only free allocation of
> - * vmdk_add_extent here. */
> - goto fail;
> + /* free extent allocated by vmdk_add_extent */
> + vmdk_free_last_extent(bs);
> }
> - return 0;
> - fail:
> - vmdk_free_extents(bs);
> return ret;
> }
>
> -static int vmdk_open_vmdk4(BlockDriverState *bs, int flags)
> +static int vmdk_open_vmdk4(BlockDriverState *bs,
> + BlockDriverState *file,
> + int flags)
> {
> int ret;
> uint32_t magic;
> uint32_t l1_size, l1_entry_sectors;
> VMDK4Header header;
> - BDRVVmdkState *s = bs->opaque;
> VmdkExtent *extent;
>
> - s->desc_offset = 0x200;
> - ret = bdrv_pread(bs->file, sizeof(magic), &header, sizeof(header));
> + ret = bdrv_pread(file, sizeof(magic), &header, sizeof(header));
> if (ret < 0) {
> - goto fail;
> + return ret;
> }
> l1_entry_sectors = le32_to_cpu(header.num_gtes_per_gte)
> * le64_to_cpu(header.granularity);
> + if (l1_entry_sectors <= 0) {
> + return -EINVAL;
> + }
> l1_size = (le64_to_cpu(header.capacity) + l1_entry_sectors - 1)
> / l1_entry_sectors;
> - extent = vmdk_add_extent(bs, bs->file, false,
> + extent = vmdk_add_extent(bs, file, false,
> le64_to_cpu(header.capacity),
> le64_to_cpu(header.gd_offset) << 9,
> le64_to_cpu(header.rgd_offset) << 9,
> l1_size,
> le32_to_cpu(header.num_gtes_per_gte),
> le64_to_cpu(header.granularity));
> - if (extent->l1_entry_sectors <= 0) {
> - ret = -EINVAL;
> - goto fail;
> - }
> - /* try to open parent images, if exist */
> - ret = vmdk_parent_open(bs);
> - if (ret) {
> - goto fail;
> - }
> - s->parent_cid = vmdk_read_cid(bs, 1);
> ret = vmdk_init_tables(bs, extent);
> if (ret) {
> - goto fail;
> + /* free extent allocated by vmdk_add_extent */
> + vmdk_free_last_extent(bs);
> }
> - return 0;
> - fail:
> - vmdk_free_extents(bs);
> return ret;
> }
>
> @@ -460,6 +458,31 @@ static int vmdk_parse_description(const char *desc, const char *opt_name,
> return 0;
> }
>
> +/* Open an extent file and append to bs array */
> +static int vmdk_open_sparse(BlockDriverState *bs,
> + BlockDriverState *file,
> + int flags)
> +{
> + uint32_t magic;
> +
> + if (bdrv_pread(file, 0, &magic, sizeof(magic)) != sizeof(magic)) {
> + return -EIO;
> + }
> +
> + magic = be32_to_cpu(magic);
> + switch (magic) {
> + case VMDK3_MAGIC:
> + return vmdk_open_vmdk3(bs, file, flags);
> + break;
> + case VMDK4_MAGIC:
> + return vmdk_open_vmdk4(bs, file, flags);
> + break;
> + default:
> + return -EINVAL;
> + break;
> + }
> +}
> +
> static int vmdk_parse_extents(const char *desc, BlockDriverState *bs,
> const char *desc_file_path)
> {
> @@ -470,6 +493,8 @@ static int vmdk_parse_extents(const char *desc, BlockDriverState *bs,
> const char *p = desc;
> int64_t sectors = 0;
> int64_t flat_offset;
> + char extent_path[PATH_MAX];
> + BlockDriverState *extent_file;
>
> while (*p) {
> /* parse extent line:
> @@ -504,24 +529,29 @@ static int vmdk_parse_extents(const char *desc, BlockDriverState *bs,
> goto next_line;
> }
>
> + path_combine(extent_path, sizeof(extent_path),
> + desc_file_path, fname);
> + ret = bdrv_file_open(&extent_file, extent_path, bs->open_flags);
> + if (ret) {
> + return ret;
> + }
> +
> /* save to extents array */
> if (!strcmp(type, "FLAT")) {
> /* FLAT extent */
> - char extent_path[PATH_MAX];
> - BlockDriverState *extent_file;
> VmdkExtent *extent;
>
> - path_combine(extent_path, sizeof(extent_path),
> - desc_file_path, fname);
> - ret = bdrv_file_open(&extent_file, extent_path, bs->open_flags);
> - if (ret) {
> - return ret;
> - }
> extent = vmdk_add_extent(bs, extent_file, true, sectors,
> 0, 0, 0, 0, sectors);
> extent->flat_start_offset = flat_offset;
> + } else if (!strcmp(type, "SPARSE")) {
> + /* SPARSE extent */
> + ret = vmdk_open_sparse(bs, extent_file, bs->open_flags);
> + if (ret) {
> + bdrv_delete(extent_file);
> + return ret;
> + }
> } else {
> - /* SPARSE extent, not supported for now */
> fprintf(stderr,
> "VMDK: Not supported extent type \"%s\""".\n", type);
> return -ENOTSUP;
I think you'll leak extent_file in the error case. I'll merge the patch
anyway, please fix on top.
Kevin
next prev parent reply other threads:[~2011-09-06 10:35 UTC|newest]
Thread overview: 12+ messages / expand[flat|nested] mbox.gz Atom feed top
2011-08-12 15:19 [Qemu-devel] [PATCH v3 0/8] Add various VMDK subformats support Fam Zheng
2011-08-12 15:19 ` [Qemu-devel] [PATCH v3 1/8] VMDK: enable twoGbMaxExtentFlat Fam Zheng
2011-08-12 15:19 ` [Qemu-devel] [PATCH v3 2/8] VMDK: add twoGbMaxExtentSparse support Fam Zheng
2011-09-06 10:38 ` Kevin Wolf [this message]
2011-08-12 15:19 ` [Qemu-devel] [PATCH v3 3/8] VMDK: separate vmdk_read_extent/vmdk_write_extent Fam Zheng
2011-08-12 15:19 ` [Qemu-devel] [PATCH v3 4/8] VMDK: Opening compressed extent Fam Zheng
2011-08-12 15:19 ` [Qemu-devel] [PATCH v3 5/8] VMDK: read/write " Fam Zheng
2011-08-12 15:19 ` [Qemu-devel] [PATCH v3 6/8] VMDK: creating streamOptimized subformat Fam Zheng
2011-08-12 15:19 ` [Qemu-devel] [PATCH v3 7/8] VMDK: bugfix, open Haiku vmdk image Fam Zheng
2011-08-12 15:19 ` [Qemu-devel] [PATCH v3 8/8] VMDK: bugfix, opening vSphere 4 exported image Fam Zheng
2011-09-05 8:15 ` [Qemu-devel] [PATCH v3 0/8] Add various VMDK subformats support Stefan Hajnoczi
2011-09-06 10:40 ` Kevin Wolf
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=4E65F810.9080701@redhat.com \
--to=kwolf@redhat.com \
--cc=famcool@gmail.com \
--cc=hch@lst.de \
--cc=qemu-devel@nongnu.org \
--cc=stefanha@gmail.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.