we are using kvm/qemu with qcow2-images for our virtual machines.
qcow2 has this nice feature where the image file only allocates the actually needed space by the virtual-machine. but how do i shrink back the image file, if the virtual machine's allocated space gets smaller?
example:
1.) i create a new image with qcow2 format, size 100GB
2.) i use this image to install ubuntu. installation needs about 10 gb, the image-file grows up to about 10GB. nothing unexpected so far.
3.) i fill up the image with about 40 GB of additional data. the image-file grows up to 50GB. i am ok with that :-)
4.) this is where it gets strange: i delete all of the 40GB data on the image, but the image-size still eats up 50GB.
question: how do i free up that 40GB of data and shrink the image to the only needed 10 GB?
thanks in advance, berni
The image will not shrink automatically, since when you delete files, you don't actually delete data (this is why undelete works). Qemu has a facility to shrink qcow2 images back, but what the utility does is really deduplicate the zeroes from the disk, leaving all other information intact. So the idea would be to:
dd if=/dev/zero of=/some/file
until you run out of space)qemu-img convert -O qcow2 original_image.qcow2 deduplicated_image.qcow2
This, afaik, will only work with qcow2 images, I haven't tested other formats.
virt-sparsify can do all this with less hassle on your part: http://libguestfs.org/virt-sparsify.1.html
You can use virt-sparsify for this. It automatically zeros and sparsifies free space on most filesystems it finds within the source disk image (supports ext2/3/4, btrfs, NTFS, etc) .
Step-by-step instructions for in-place conversion
Shut down the VM
Keep a backup
Convert:
virt-sparsify --in-place disk.img
Step-by-step instructions for conversion to another file (this is safer but requires more free space):
Shut down the VM
Convert:
virt-sparsify disk.img new-file.img
Swap old with new image file:
mv disk.img disk.img.bak && mv new-file.img disk.img
If VM starts normally you can remove the backup:
rm disk.img.bak
P.S.: You can also convert between formats when you use virt-sparsify:
In addition to the accepted answer (which describe the more common, general method to shrink a qcow2 file), modern version (ie: RHEL7+) of the libvirt/qemu/qcow2 stack supports the
virtio-scsi
driver which supports thediscard='unmap'
option.If the virtual machine uses
virtio-scsi
and the libvirt definition includediscard='unmap'
, a simplefstrim
on the guest filesystem will release assigned-but-unused free space on the host. This can be checked withdu -hs <guestdisk>
on host side (note:ls -al <guestdisk>
will not show the real allocated size, just the logical one).For even newer libvirt/qemu instances (ie: RHEL8+), guest file size can be reduced even if the guest OS does not support trim: by enabling both
discard='unmap'
anddiscard_zeroes='unmap'
(and using thevirtio-scsi
driver), writing a sequential stream of zero on guest side (ie: viadd if=/dev/zero of=/zero.img bs=1M count=1024
) will trigger host-size trimming of the affected LBA ranges. However, please note that this can be CPU-intensive for the host (which had to "parse" any guest write searching for repeating zeroes), so it should be only enabled in specific cases.Both methods will reduce the physical allocated size (what
du -hs
shows). For reducing the logical size (what a simplels -al
gives) you need to usevirt-spasify
orqemu-img
(as described in the other answers).virt-sparsify --compress fileA fileB
will reduce the size of the image, however it creates the qcow2 image as another file, so you will have to rename and move things around. You cannot use--compress
and--in-place
at the same time. See the following link to learn more. http://libguestfs.org/virt-sparsify.1.htmlIn addition to other answers about
virt-sparsify
:Temporary Folder
virt-sparsify uses your /tmp folder to create/manipulate the new disk image before place it in destination folder. So, your / partition should have free space even if your volume file is in other partition.
If you don't have enouth space, you can set a new temp folder with
--tmp my_tmp_folder
option. Example:--in-place
;Verbose
There is an verbose option to help you see what's going on. Example:
Check before delete
As said before, always boot your vm and check if everything is working before delete the original disk image.