How to use KVM from the command line on Debian or Ubuntu

There are different ways to manage virtual machines (VMs) running on KVM hypervisor. For example, virt-manager is a popular GUI-based front-end for VM management. However, if you would like to use KVM on a headless server, GUI-based solutions will not be ideal. In fact, you can create and manage KVM VMs purely from the command line using kvm command-line wrapper script. Alternatively, you can use virsh which is an easier-to-use command-line user interface for managing guest VMs. Underneath virsh, it communicates wtih libvirtd service which can control several different hypervisors including KVM, Xen, QEMU, LXC and OpenVZ.

A command-line management interface such as virsh is also useful when you would like to "automate" the provisioning and management of VMs. Also, the fact that virsh supports multiple hypervisors means you can manage different hypervisors via the same virsh interface.

In this tutorial, I will demonstrate how to run KVM from the command line by using virsh on Debian or Ubuntu.

Step One: Verify Hardware Virtualization Support

As a first step, verify that the host CPU is equipped with hardware virtualization extensions (e.g., Intel VT or AMD-V), which are required for KVM. The following command will do.

$ egrep '(vmx|svm)' --color /proc/cpuinfo

If the output does not contain vmx or svm flag, it means the host CPU does not have hardware virtualization support. Thus you cannot use KVM on the host. After verifying that the host CPU comes with vmx or svm, proceed to install KVM next.

For KVM, it is not required to run a 64-bit kernel on the KVM host, but generally it is recommended.

Step Two: Install KVM

Using apt-get, install KVM and related user-space tools.

$ sudo apt-get install qemu-kvm libvirt-bin

During installation, libvirtd group (or libvirt-qemu on Debian) will be created, and your userID will be automatically added to the group. This will allows you to manage VMs as a non-root regular user. You can verify that by using id command, which will show your group IDs:

$ id <your-userID>

If for some reason, libvirtd (or libvirt-qemu) is not found in your groupID list, you can manually add yourself to the group as follows.

On Ubuntu:

$ sudo adduser [youruserID] libvirtd

On Debian:

$ sudo adduser [youruserID] libvirt-qemu

Reload updated group membership info as follows. Upon asked for a password, enter your login password.

$ exec su -l $USER

At this point, you should be able to run virsh as a regular user. As a test, try the command below, which will show a list of available VMs (currently none). If you do not encounter a permission error, it means everything is okay so far.

$ virsh list
 Id    Name                           State

Step Three: Configure Bridged Networking

One way to enable KVM VMs to access external networks is via a Linux bridge created on a KVM host. The bridge interconnects the virtual interfaces of VMs with the physical interface of the host, so that the VMs can send or receive traffic via the physical interface. This is called bridged networking.

Here is how to create and configure a Linux bridge br0 for bridged networking with KVM.

First, install a necessary package, and create a Linux bridge from the command line.

$ sudo apt-get install bridge-utils
$ sudo brctl addbr br0

The next step is to configure Linux bridge in /etc/network/interfaces, so that the bridge is configured automatically upon boot. To use /etc/network/interfaces, you need to disable Network Manager on your system (if you are using it). Follow the this instruction to disable Network Manager.

After disabling Network Manager, go ahead and configure Linux bridge br0 in /etc/network/interfaces as follows.

#auto eth0
#iface eth0 inet dhcp

auto br0
iface br0 inet dhcp
        bridge_ports eth0
        bridge_stp off
        bridge_fd 0
        bridge_maxwait 0

In the above I assume that eth0 is the primary network interface that is connected to external networks. Also, I assume that eth0 is getting its IP address via DHCP. Note that there is no configuration for eth0 in /etc/network/interface. The Linux bridge br0 takes up the configuration of eth0 as eth0 is enslaved to the bridge br0.

Restart network service, and verify that Linux bridge is configured successfully. If successful, br0 should be assigned the eth0's DHCP IP address, and eth0 should not have any IP address assigned.

$ sudo /etc/init.d/networking restart
$ ifconfig

If for any reason eth0 still retains the IP address which is assigned to br0, you may have to explicitly remove the IP address from eth0.

Step Four: Create a VM from the Command Line

With KVM, the configuration of a VM is stored in a domain XML file. Thus, the first step to create a VM is to prepare its domain XML file.

The following is a sample domain XML file of a VM. You can customize it as needed.

<domain type='kvm'>
    <boot dev='cdrom'/>
  <clock offset='utc'/>
    <disk type="file" device="disk">
      <driver name="qemu" type="raw"/>
      <source file="/home/dev/images/alice.img"/>
      <target dev="vda" bus="virtio"/>
      <address type="pci" domain="0x0000" bus="0x00" slot="0x04" function="0x0"/>
    <disk type="file" device="cdrom">
      <driver name="qemu" type="raw"/>
      <source file="/home/dev/iso/CentOS-6.5-x86_64-minimal.iso"/>
      <target dev="hdc" bus="ide"/>
      <address type="drive" controller="0" bus="1" target="0" unit="0"/>
    <interface type='bridge'>
      <source bridge='br0'/>
      <mac address="00:00:A3:B0:56:10"/>
    <controller type="ide" index="0">
      <address type="pci" domain="0x0000" bus="0x00" slot="0x01" function="0x1"/>
    <input type='mouse' bus='ps2'/>
    <graphics type='vnc' port='-1' autoport="yes" listen=''/>
    <console type='pty'>
      <target port='0'/>

The above domain XML file defines the following VM.

  • 1GB memory, one vCPU and one hard drive.
  • Disk image: /home/dev/images/alice.img.
  • Boot from CD-ROM (/home/dev/iso/CentOS-6.5-x86_64-minimal.iso).
  • Networking: one network interface bridged to br0
  • Remote access via VNC.

The UUID string inside <uuid></uuid> can be randomly generated. To get a random UUID, you can use uuid command-line tool.

$ sudo apt-get install uuid
$ uuid

Another way to create a domain XML file is to dump the domain information of an existing VM as follows.

$ virsh dumpxml alice > bob.xml

Step Five: Start VM from the Command Line

Before starting a VM, you need to create its initial disk image. For that, you can use qemu-img command, which comes with qemu-kvm package you installed. The following command creates 10GB size empty disk image of qcow2 type:

$ qemu-img create -f qcow2 /home/dev/images/alice.img 10G

The advantage of using "qcow2" (as opposed to "raw") as a disk image format is that a "qcow2"-type disk image is not created as a full size (10GB) initially, but grows as the disk gets populated. So it is more space-efficient.

Now you are ready to start a VM using the domain XML file you created earlier. The following command will create a VM, and automatically start it.

$ virsh create alice.xml
Domain alice created from alice.xml

NOTE: If you run the above command with an already created VM, it will wipe out the VM without warning. If you already created a VM, you can instead use the following command to just start the VM.

$ virsh start alice.xml

Verify that a new domain has been created and started successfully with:

$ virsh list
 Id    Name                           State
 3     alice                          running

Also, verify that the virtual interface for the VM (e.g., vnet0) is successfully added to the Linux bridge br0 that you created earlier.

$ sudo brctl show

Step Six: Remote Access a VM

To access the console of a running VM remotely, you can use any VNC client.

First, find out the VNC port number for the VM as follows.

$ sudo netstat -nap | egrep '(kvm|qemu)'

In this example, the VNC port number for alice VM is 5900.

Then launch a VNC client, and connect to a VNC server running at <KVM-host-IP>:5900. In our example, the VM is supposed to boot into CentOS CD-ROM.

Manage VMs with virsh

The following lists common usages of virsh command.

To create a new guest domain and start a VM:

$ virsh create alice.xml

To stop a VM and destroy a guest domain:

$ virsh destroy alice

To shutdown a VM (without destroying a domain):

$ virsh shutdown alice

To suspend a VM:

$ virsh suspend alice

To resume a suspended VM:

$ virsh resume alice

To access login console of a running VM:

$ virsh console alice

To autostart a VM upon host booting:

$ virsh autostart alice

To get domain information of a VM:

$ virsh dominfo alice

To edit domain XML of a VM:

$ virsh edit alice

The above will invoke the domain XML file of the VM with a default text editor. Any change in the XML will automatically be validated by libvirt for correctness.

You can also manage VMs from within a virsh session. To create and enter a new virsh session, simply run:

$ virsh

At the virsh prompt, you can use any virsh commands.


1. I am getting the error while trying to create a VM:

error: internal error: no supported architecture for os type 'hvm'

You can get this error if your hardware does not have hardware virtualization support (e.g., Intel VT or AMD-V), which is required to run KVM. If you are getting this error even when your CPU comes with Intel VT or AMD-V, here are possible solutions:

First, check if kvm kernel module is missing.

$ lsmod | grep kvm

If kvm kernel module is not loaded, you must load it as follows.

$ sudo modprobe kvm_intel (for Intel processor)
$ sudo modprobe kvm_amd (for AMD processor)

The second solution is adding "--connect qemu:///system" argument to virsh command as follows. This argument may be needed when you are using more than one hypervisor (e.g., VMware, VirtualBox) on the server hardware.

$ virsh --connect qemu:///system create alice.xml

2. I am getting the error while trying to access login console of my VM:

$ virsh console alice
error: internal error: cannot find character device <null>

This error occurs because you did not define a console device in the VM's XML file. Add the following inside the "device" section of the XML file.

    <console type='pty'>
      <target port='0'/>

Subscribe to Xmodulo

Do you want to receive Linux FAQs, detailed tutorials and tips published at Xmodulo? Enter your email address below, and we will deliver our Linux posts straight to your email box, for free. Delivery powered by Google Feedburner.

Support Xmodulo

Did you find this tutorial helpful? Then please be generous and support Xmodulo!

The following two tabs change content below.
Dan Nanni is the founder and also a regular contributor of He is a Linux/FOSS enthusiast who loves to get his hands dirty with his Linux box. He likes to procrastinate when he is supposed to be busy and productive. When he is otherwise free, he likes to watch movies and shop for the coolest gadgets.

23 thoughts on “How to use KVM from the command line on Debian or Ubuntu

  1. Just a note: you don't need to actually edit the XML manually in most cases. I like using virt-install, and tweaking the result if needed.

    • What I like doing is creating one virtual machine, get it the way I want it, dump the XML, and use that XML as a template for generating more virtual machines.
      It's a bit of a pain, but it allows you to generate a lot of systems quickly once you get it going.

    • If you use virsh edit you'll be dropped into vi (or your default text editor) and you can edit it the xml file from there, virsh will prevent saving bad xml if you make a mistake, or have typing errors.

  2. Or do it really from the command line, without the need of an XML configuration file:
    kvm \
    -m 512m \
    -device e1000,romfile=,vlan=0 \
    -net tap \
    -usbdevice tablet \
    -hda hda
    "hda" is a file that you previously created and made bootable. If you don't have hda bootable, then you can point to an external kernel, just add a " \" after "-hda hda" and then this:
    -append "root=/dev/sda1 ro quiet init=/bin/systemd" \
    -kernel linux-3.12/arch/x86/boot/bzImage
    (I didn't specify an initrd, because I dislike initrds on well-defined systems. And for my, VMs are well-defined).

  3. Hi,

    great tutorial. Can you tell me, why VNC running in localhost? I cannot connect to host-PC:5900 :(

    # netstat -nap | grep kvm
    tcp 0 0* LISTEN 11771/kvm
    unix 2 [ ACC ] STREAM LISTENING 1180344 11771/kvm /var/lib/libvirt/qemu/unifi.monitor
    unix 3 [ ] STREAM CONNECTED 1172338 11771/kvm /var/lib/libvirt/qemu/unifi.monitor


    • Simply replace:
      <graphics type='vnc' port='-1' autoport="yes" listen=''/>
      <graphics type='vnc' port='-1' autoport="yes" listen=''/>

      But be aware that it's not a good idea to expose VNC port to the world for security reasons.

  4. how can you actually login to a VM with virsh in the command line?

    I used to manage my VM remotely with ssh, but somehow the network settings or the ssh-server of the VM seem to have a problem and I can't access it anymore with ssh (connection is refused). So I need to access it locally from the command line to fix this problem. Host and guest both are debian/ubuntu servers without graphical interface

  5. Hello, I have followed your tutorial and am very satisfied with it, but I am just having an issue getting the network settings to work on the guest VM. Upon running 'brctl show' there isn't the venet interface assigned to it, so I am unsure as to what to do to fix this.

  6. You made a typo...And this unfortunately wiped out one of my existing VMs...I guess I should have used better judgement than just follow your post.

    >>Now you are ready to start a VM using the domain XML file you created earlier. The >>following command will start a VM.

    >>$ virsh create alice.xml

    This should be

    $ virsh start alice.xml

    Creating a VM with the same name as an existing one wipes it out without warning, replacing it with a blank template.

    Please edit your article.


    • Ouch. Sorry to hear that.
      Actually, it is not a typo as I'm creating alice VM from scratch. "virsh create" command creates and also starts a VM (unless you specify "--paused" option). But you are right. If you already created alice VM before, you should simply run "virsh start". I added some clarification. Thanks.

  7. Thank you for this amazing tutorial!
    I am totally new to VMs but I need to create VMs to run some experiment.
    I have a question when I reached step 4 of the tutorial. Where should I create the file named alice.xml? Should it be in a specific directory? And regarding the "boot from CD-ROM" can I add any os I want into the directory "/home/dev/iso/"? Sorry if my question seems so naive :) Thank you.

    • You can put *.xml file anywhere, e.g., in the same directory where you run virsh. If you placed .xml file in an arbitrary directory, just spell out the full path when running virsh:

      $ virsh create /path/to/alice.xml

      As for CD-ROM, you can use any bootable ISO.

  8. Hi Dan, I have another question:)
    I am not able to access the VM I created using the console, when I type "virsh console vm1" the system hag.
    I have a remote access to a physical node that running Ubuntu 12.04. In this node I created a VM using your .xml file, I only changed the iso and I added the following lines into the device section of the xml file:

    and I also have the following lines inside my device section:

    Any idea how I can connect to the VM? I googled it but nothing actually worked form me. Can I ssh to the VM from the host machine? The vnet0 doesn't have an IP address so I am not sure how to ssh.

  9. Hi Dan,

    I edited my .xml file to include your edit, but I still get this error -

    root@ubuntu:~# virsh console vcp-vmx1
    Connected to domain vcp-vmx1
    Escape character is ^]
    error: internal error: character device is not using a PTY

    • You are probably missing the device definition in your xml file.
      You can edit the xml file directly, which I don't recommend, or run the following.

      virsh shutdown <domain>
      virsh edit <domain>

      You should see vi or your default text editor starting with the xml file.
      Add the following after </interface> and before <mouse> sections in the resulting xml

          <serial type='pty'>
            <target port='0'/>
          <console type='pty'>
            <target type='serial' port='0'/>

      Verify against any typing errors, save and quit your text editor.
      virsh will check the syntax and validate the xml file, and if there are errors, you'd see message about changes not saved.

      Now you have to make the changes recognized by kvm.

      virsh define /path/to/domainxmlfile.xml

      virsh start <domain>

      Check if you can now use the console:

      virsh console <domain>

      Give it couple of seconds, then hit Enter.
      You should be at the log in prompt of your VM.

  10. Some related commands and methods I recommend.

    Forget about the messy install process that requires a graphical (VNC) setup. Use virt-builder which provides pre-made minimal installs for fairly wide range of popular Linux server distributions. virt-builder just creates a disk image you can use.

    Once you have a VM they wan you want it, and you want to create more of the same, use the virt-clone command. It will quickly duplicate the source VM disk image and create a new config file.

    And lastly, check out all of the virt-{whatever} commands provided by the libguestfs tools. Easily manipulate the contents of any disk image while it isn't in use by a VM.

    • Good comments. virt-builder is indeed a useful tool to create bootable images for well known distros quickly. I was planning to cover VM cloning in another article. Thanks for mentioning virt-clone.

  11. I can't get the virtual installation to go completely full screen. for some reason when I go full screen there are two big bars on each side of the screen that could be filled with content from the guest OS instead they are just back bars. Please help me to get that issue resolved I will appreciate it. thank you

Leave a comment

Your email address will not be published. Required fields are marked *