Skip to content

Instantly share code, notes, and snippets.

@zakkak
Created March 24, 2020 23:25
Show Gist options
  • Save zakkak/ab08672ff9d137bbc0b2d0792a73b7d2 to your computer and use it in GitHub Desktop.
Save zakkak/ab08672ff9d137bbc0b2d0792a73b7d2 to your computer and use it in GitHub Desktop.
Resizing a filesystem using qemu-img and fdisk

Occasionally we will deploy a virtual instance into our KVM infrastructure and realize after the fact that we need more local disk space available. This is the process we use to expand the disk image. This process assumes the following:

  • You're using legacy disk partitions. The process for LVM is similar and I will describe that in another post.
  • The partition you need to resize is the last partition on the disk.

This process will work with either a qcow2 or raw disk image. For raw images you can also run fdisk on the host, potentially saving yourself a reboot, but that's less convenient for qcow2 format images.


We start with a 5.5G root filesystem with 4.4G free:

[root@localhost ~]# df -h /
Filesystem      Size  Used Avail Use% Mounted on
/dev/vda2       5.5G  864M  4.4G  17% /

We need to shut down the system to grow the underlying image:

[root@localhost ~]# poweroff

On the host, we use the qemu-img resize command to grow the image. First we need the path to the underlying disk image:

[lars@madhatter blog]$ virsh -c qemu:///system dumpxml lars-test-0 | grep file
    <disk type='file' device='disk'>
      <source file='/var/lib/libvirt/images/lars-test-0-1.img'/>

And now we increase the image size by 10G:

[lars@madhatter blog]$ sudo qemu-img resize /var/lib/libvirt/images/lars-test-0.img +10G
Image resized.

Now reboot the instance:

[lars@madhatter blog]$ virsh -c qemu:///system start lars-test-0

And login in on the console:

[lars@madhatter blog]$ virsh -c qemu:///system console lars-test-0
Connected to domain lars-test-0
Escape character is ^]

Fedora release 17 (Beefy Miracle)
Kernel 3.6.2-4.fc17.x86_64 on an x86_64 (ttyS0)

localhost login: root
Password:

We're going to use fdisk to modify the partition layout. Run fdisk on the system disk:

[root@localhost ~]# fdisk /dev/vda
Welcome to fdisk (util-linux 2.21.2).

Changes will remain in memory only, until you decide to write them.
Be careful before using the write command.

Print out the existing partition table and verify that you really are going to be modifying the final partition:

Command (m for help): p

Disk /dev/vda: 19.3 GB, 19327352832 bytes
16 heads, 63 sectors/track, 37449 cylinders, total 37748736 sectors
Units = sectors of 1 * 512 = 512 bytes
Sector size (logical/physical): 512 bytes / 512 bytes
I/O size (minimum/optimal): 512 bytes / 512 bytes
Disk identifier: 0x00007d9f

   Device Boot      Start         End      Blocks   Id  System
/dev/vda1   *        2048     1026047      512000   83  Linux
/dev/vda2         1026048     5154815     2064384   82  Linux swap / Solaris
/dev/vda3         5154816    16777215     5811200   83  Linux

Delete and recreate the final partition, in this case /dev/vda3...

Command (m for help): d
Partition number (1-4): 3
Partition 3 is deleted

...and create a new partition, accepting all the defaults. This will create a new partition starting int he same place and extending to the end of the disk:

Command (m for help): n
Partition type:
   p   primary (2 primary, 0 extended, 2 free)
   e   extended
Select (default p): p
Partition number (1-4, default 3): 3
First sector (5154816-37748735, default 5154816): 
Using default value 5154816
Last sector, +sectors or +size{K,M,G} (5154816-37748735, default 37748735): 
Using default value 37748735
Partition 3 of type Linux and of size 15.6 GiB is set

You can print out the new partition table to see that indeed /dev/vda3 is now larger:

Command (m for help): p

Disk /dev/vda: 19.3 GB, 19327352832 bytes
16 heads, 63 sectors/track, 37449 cylinders, total 37748736 sectors
Units = sectors of 1 * 512 = 512 bytes
Sector size (logical/physical): 512 bytes / 512 bytes
I/O size (minimum/optimal): 512 bytes / 512 bytes
Disk identifier: 0x00007d9f

   Device Boot      Start         End      Blocks   Id  System
/dev/vda1   *        2048     1026047      512000   83  Linux
/dev/vda2         1026048     5154815     2064384   82  Linux swap / Solaris
/dev/vda3         5154816    37748735    16296960   83  Linux

Write the changes to disk:

Command (m for help): w
The partition table has been altered!

Calling ioctl() to re-read partition table.

WARNING: Re-reading the partition table failed with error 16: Device or resource busy.
The kernel still uses the old table. The new table will be used at
the next reboot or after you run partprobe(8) or kpartx(8)
Syncing disks.

Note the warning! The kernel has cached a copy of the old partition table. We need to reboot the system before our changes are visible! So we reboot the system:

[root@localhost ~]# reboot

And log back in. Run df to see the current size of the root filesystem:

[root@localhost ~]# df -h /
Filesystem      Size  Used Avail Use% Mounted on
/dev/vda3       5.5G  864M  4.4G  17% /

Now run resize2fs to resize the root filesystem so that it expands to fill our extended /dev/vda3:

[root@localhost ~]# resize2fs /dev/vda3
resize2fs 1.42.3 (14-May-2012)
Filesystem at /dev/vda3 is mounted on /; on-line resizing required
old_desc_blocks = 1, new_desc_blocks = 1
The filesystem on /dev/vda3 is now 4074240 blocks long.

Run df again to see that we now have additional space available:

[root@localhost ~]# df -h /
Filesystem      Size  Used Avail Use% Mounted on
/dev/vda3        16G  867M   14G   6% /
[root@localhost ~]# 
@markst
Copy link

markst commented Mar 31, 2023

champion!

@ntzb
Copy link

ntzb commented Nov 22, 2023

what should I do if I want to resize the first partition?
I've already used qemu-img to resize with +5G.

linux@debian:~$ df -h .
Filesystem      Size  Used Avail Use% Mounted on
/dev/vda1       8.9G  5.1G  3.4G  61% /

and in root, from fdisk:

root@debian:~# fdisk /dev/vda
...
Command (m for help): p
Disk /dev/vda: 15 GiB, 16106127360 bytes, 31457280 sectors
Units: sectors of 1 * 512 = 512 bytes
Sector size (logical/physical): 512 bytes / 512 bytes
I/O size (minimum/optimal): 512 bytes / 512 bytes
Disklabel type: dos
Disk identifier: 0x1d220feb

Device     Boot    Start      End  Sectors  Size Id Type
/dev/vda1  *        2048 18970623 18968576    9G 83 Linux
/dev/vda2       18972670 20969471  1996802  975M  5 Extended
/dev/vda5       18972672 20969471  1996800  975M 82 Linux swap / Solaris

@zakkak
Copy link
Author

zakkak commented Nov 22, 2023

I am afraid I can't help. IIRC I forked this gist from another user for archival purposes.

@anzel601
Copy link

anzel601 commented Oct 2, 2024

what should I do if I want to resize the first partition? I've already used qemu-img to resize with +5G.

linux@debian:~$ df -h .
Filesystem      Size  Used Avail Use% Mounted on
/dev/vda1       8.9G  5.1G  3.4G  61% /

and in root, from fdisk:

root@debian:~# fdisk /dev/vda
...
Command (m for help): p
Disk /dev/vda: 15 GiB, 16106127360 bytes, 31457280 sectors
Units: sectors of 1 * 512 = 512 bytes
Sector size (logical/physical): 512 bytes / 512 bytes
I/O size (minimum/optimal): 512 bytes / 512 bytes
Disklabel type: dos
Disk identifier: 0x1d220feb

Device     Boot    Start      End  Sectors  Size Id Type
/dev/vda1  *        2048 18970623 18968576    9G 83 Linux
/dev/vda2       18972670 20969471  1996802  975M  5 Extended
/dev/vda5       18972672 20969471  1996800  975M 82 Linux swap / Solaris

In this case you have to remove Swap and extended partition, then as following remove partition /dev/vda1 and create to desired size(by default), if you want to have swap on same disk you can leave some space and create another partition for swap.
swap can be simply a file on root partiotion :D so default.

@Sudrien
Copy link

Sudrien commented Jan 17, 2025

what should I do if I want to resize the first partition? I've already used qemu-img to resize with +5G.

This is the same setup I've got with a Debian 12 VM on QEMU, by the way

If you have a non-root partition with data, not swap, you can move or resize it with parted / gparted before bringing fdisk in. parted moves data, fdisk does not. Otherwise, the is your non-LVM checklist:

  • fdisk /dev/vda
  • print all partitions with p
  • delete all partitons
  • if you forgot to p, bail out now and try again
  • re-create vda 1 primary with a size of -975M (or whatever the size was - DO NOT clean anything up if it prompts, this is your data)
  • re-create vda 2 extended
  • re-create vda 5
  • use t to change vda5 type to 82 (as well as any other p mismatches)
  • last chance to bail out
  • use w to write new partition table to disk
  • reboot (it will be slow, UUIDs need to be fixed)
  • resize2fs /dev/vda1 to use the new disk space - non-ext(2-4) filesystems will have a different utility
  • mkswap /dev/vda5 to get your swap in order
  • use blkid to get new partition UUIDs
  • edit /etc/fstab to use these new UUIDs (feel free to make backup copies, append the output of blkid with blkid >> /etc/fstab, stuff like that)
  • grub-install /dev/vda fixes initial startup
  • update-initramfs -u fixes kernel config
  • reboot

Back to full speed

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment