[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
Re: [Qemu-discuss] [Qemu-devel] Estimation of qcow2 image size converted
From: |
John Snow |
Subject: |
Re: [Qemu-discuss] [Qemu-devel] Estimation of qcow2 image size converted from raw image |
Date: |
Mon, 13 Feb 2017 12:03:35 -0500 |
User-agent: |
Mozilla/5.0 (X11; Linux x86_64; rv:45.0) Gecko/20100101 Thunderbird/45.7.0 |
CCing qemu-block;
On 02/13/2017 10:46 AM, Maor Lipchuk wrote:
> Hi all,
>
> I was wondering if that is possible to provide a new API that
> estimates the size of
> qcow2 image converted from a raw image. We could use this new API to
> allocate the
> size more precisely before the convert operation.
>
I'm not sure you'd need an API to do this, you could estimate it
yourself pretty effectively.
Naively, just loop 64KiB at a time and check if each 64KiB chunk is zero
or not. If it isn't, add +64KiB to the filesize estimate. If it is, skip
that chunk.
On filesystems that already support sparse allocation, you can use lseek
with SEEK_DATA or SEEK_HOLE to find where the data/zeroes are and do
something more clever to find out where the zeroes are and estimate that
way.
Then you'll add a certain number of metadata blocks to finish, and
you'll have a pretty solid estimate.
> What are we trying to do:
> - Convert raw sparse image from NFS or from block device to qcow2 image
> on thin provisioned block device
>
> - In ovirt thin provisioned block device is a regular lv, and we like
> allocate only the required size for the the qcow file.
>
> Our current (stupid) solution is to allocate the entire LV using the
> size of the raw image.
>
> Here is an example flow:
>
> $ truncate -s 10G test.raw
>
> We don't know what will be the size of the qcow on the block storage,
> so we allocate the entire LV:
>
> $ lvcreate --size 10G vg/lv
>
> Then we convert the file to the new LV:
>
> $ qemu-img convert -f raw -O qcow2 test.raw /dev/vg/lv
>
> After the copy we can check the actual size:
>
> $ qemu-img check /dev/vg/lv
>
> And reduce the LV:
>
> $ lvreduce -L128m vg/lv
>
> But we like to avoid the allocation, and allocate only the needed size
> before we convert the image.
>
> We found that if we create a file with one byte for each cluster (64K),
> qcow2 file will be bigger than the raw file:
>
Makes sense. You essentially allocate the entire file this way (No
unallocated clusters) and then you have to pay the metadata tax on top
if it.
> Creating worst case raw file:
>
> with open("worst.raw", "wb") as f:
> for offset in range(64 * 1024 - 1, 10 * 1024**3, 64 * 1024):
> f.seek(offset)
> f.write("x")
>
> $ ls -lh worst.raw
> -rw-rw-r--. 1 user user 10G Feb 13 16:43 worst.raw
>
> $ du -sh worst.raw
> 642M worst.raw
>
> $ ls -lh worst.qcow2
> -rw-r--r--. 1 user user 11G Feb 13 17:10 worst.qcow2
>
> Now compare that to the best case:
>
> with open("best.raw", "wb") as f:
> for i in range(10 * 1024**3 / (64*1024)):
> f.write("x" * 4096)
>
So, 640MiB of "x" contiguously from the beginning of the file?
You're only touching about ... 10,240 clusters that way. Makes sense
that the qcow2 is nearly the same size as the written data.
> $ ls -lh best.raw
> -rw-rw-r--. 1 user user 10G Feb 13 17:18 best.raw
>
> $ du -sh best.raw
> 641M best.raw
>
> $ qemu-img convert -p -f raw -O qcow2 best.raw best.qcow2
>
> $ ls -lh best.qcow2
> -rw-r--r--. 1 user user 641M Feb 13 17:21 best.qcow2
>
> $ du -sh best.qcow2
> 641M best.qcow2
>
> So it seems that to estimate the size of the qcow2 file, we need
> to check not only the number of blocks but the location of the blocks.
>
Well, you need to check the number of allocated clusters. "blocks" are
not a meaningful concept to qcow2 exactly. One byte written to every
single cluster will fully allocate the file.
> We can probably use qemu-img map to estimate:
>
> $ qemu-img map worst.raw --output json
> [{ "start": 0, "length": 61440, "depth": 0, "zero": true, "data":
> false, "offset": 0},
> { "start": 61440, "length": 4096, "depth": 0, "zero": false, "data":
> true, "offset": 61440},
> { "start": 65536, "length": 61440, "depth": 0, "zero": true, "data":
> false, "offset": 65536},
> { "start": 126976, "length": 4096, "depth": 0, "zero": false, "data":
> true, "offset": 126976},
> { "start": 131072, "length": 61440, "depth": 0, "zero": true, "data":
> false, "offset": 131072},
> ...
>
> $ qemu-img map best.raw --output json
> [{ "start": 0, "length": 671088640, "depth": 0, "zero": false, "data":
> true, "offset": 0},
> { "start": 671088640, "length": 10066325504, "depth": 0, "zero": true,
> "data": false, "offset": 671088640},
> { "start": 10737414144, "length": 4096, "depth": 0, "zero": false,
> "data": true, "offset": 10737414144}]
>
> But this means we have to include qcow2 allocation logic in our code, and
> the calculation will break when qcow2 changes the format.
>
Not likely to change considerably, but fair enough of a point.
Also keep in mind that changing the cluster size will give you different
answers, too -- but that different cluster sizes will effect the runtime
performance of the image as well.
> We think that the best way to solve this issue is to return this info
> from qemu-img, maybe as a flag to qemu-img convert that will
> calculate the size of the converted image without doing any writes.
>
Might not be too hard to add, but it wouldn't necessarily be any more
accurate than if you implemented the same logic, I think.
Still, it'd be up to us to keep it up to date, but I don't know what
guarantees we could provide about the accuracy of the estimate or
preventing it from bitrot if there are format changes..
>
> See also:
> https://bugzilla.redhat.com/1358717 - Export of vm with thin provision
> disk from NFS Data domain and Import to Block Data domain makes
> virtual and Actual size of disk same.
>
> https://bugzilla.redhat.com/1419240 - Creating a Clone vm from
> template with Format "QCOW2" and Target "block based storage" has a
> disk with same actual and virtual size.
>
>
> Regards,
> Maor
>
--js