Create nested VMs

Nested virtualization is allowed by default, so unless somebody modifies the constraint for nested virtualization, you do not need to make any changes before you create nested VMs in an organization, folder, or project. If your project doesn't belong to an organization, nested virtualization is allowed by default and you can't change the constraint. For information about how to modify the constraint that determines whether you can create nested VMs, see Manage the nested virtualization constraint.

This document describes how to create various types of level 2 (L2) virtual machine (VM) instances. Before creating a nested VM, you must create an L1 VM that has nested virtualization enabled. For a description of L1 and L2 VMs, see the Nested virtualization overview.

After creating an L1 VM that has nested virtualization enabled, you can do any of the following:

Before you begin

Creating an L2 VM with external network access

Create an L2 VM with external network access by using the following procedure. This procedure uses qemu-system-x86_64 to start the L2 VM. If you are using another procedure to create an L2 VM and are experiencing trouble, reproduce the issue using this procedure before contacting Support.

  1. Create an L1 VM that has nested virtualization enabled.

  2. Use the gcloud compute ssh command to connect to the VM:

    gcloud compute ssh VM_NAME
    

    Replace VM_NAME with the name of the VM to connect to.

  3. Install the latest qemu-kvm package:

    sudo apt update && sudo apt install qemu-kvm -y
    
  4. Download a QEMU-compatible OS image to use for the L2 VM.

  5. Use the following command to start the L2 VM. When prompted, log in with user: root, password: root.

    sudo qemu-system-x86_64 -enable-kvm -hda IMAGE_NAME -m 512 -curses
    

    Replace IMAGE_NAME with the name of the QEMU-compatible OS image to use for the L2 VM.

  6. Test that your L2 VM has external access:

    user@nested-vm:~$ host google.com
    

Creating an L2 VM with a private network bridge to the L1 VM

Create an L2 VM with a private network bridge to the previously created L1 VM by using the following procedure. For information about changing the default maximum transmission unit (MTU) for your VPC network, see the maximum transmission unit overview.

  1. Create an L1 VM that has nested virtualization enabled.

  2. Use the gcloud compute ssh command to connect to the VM:

    gcloud compute ssh VM_NAME
    

    Replace VM_NAME with the name of the VM to connect to.

  3. Install the packages necessary to create the private bridge:

    sudo apt update && sudo apt install uml-utilities qemu-kvm bridge-utils virtinst libvirt-daemon-system libvirt-clients -y
    
  4. Start the default network that comes with the libvirt package:

    sudo virsh net-start default
    
  5. Run the following command to check that you have the virbr0 bridge:

    ip addr
    
  6. The output is similar to the following:

    1: lo: <LOOPBACK,UP,LOWER_UP> mtu 65536 qdisc noqueue state UNKNOWN group default qlen 1
        link/loopback 00:00:00:00:00:00 brd 00:00:00:00:00:00
        inet 127.0.0.1/8 scope host lo
           valid_lft forever preferred_lft forever
        inet6 ::1/128 scope host
           valid_lft forever preferred_lft forever
    2: eth0: <BROADCAST,MULTICAST,UP,LOWER_UP> mtu 1460 qdisc pfifo_fast state UP group default qlen 1000
        link/ether 42:01:0a:80:00:15 brd ff:ff:ff:ff:ff:ff
        inet 10.128.0.21/32 brd 10.128.0.21 scope global eth0
           valid_lft forever preferred_lft forever
        inet6 fe80::4001:aff:fe80:15/64 scope link
           valid_lft forever preferred_lft forever
    3: virbr0: <NO-CARRIER,BROADCAST,MULTICAST,UP> mtu 1500 qdisc noqueue state DOWN group default qlen 1000
        link/ether 52:54:00:8c:a6:a1 brd ff:ff:ff:ff:ff:ff
        inet 192.168.122.1/24 brd 192.168.122.255 scope global virbr0
           valid_lft forever preferred_lft forever
    4: virbr0-nic: <BROADCAST,MULTICAST> mtu 1500 qdisc pfifo_fast master virbr0 state DOWN group default qlen 1000
        link/ether 52:54:00:8c:a6:a1 brd ff:ff:ff:ff:ff:ff
    
  7. Create a tap interface to go from the L1 VM to the L2 VM:

    sudo tunctl -t tap0
    sudo ifconfig tap0 up
    
  8. Bond the tap interface to the private bridge:

    sudo brctl addif virbr0 tap0
    
  9. Run the following command to verify the setup of the bridge network:

    sudo brctl show
    
  10. The output is similar to the following:

    bridge name     bridge id               STP enabled     interfaces
    virbr0          8000.5254008ca6a1       yes             tap0
                                                            virbr0-nic
    
  11. Download a QEMU-compatible OS image to use for the L2 VM.

  12. Run screen, and press Enter at the welcome prompt:

    screen
    
  13. Use the following command to start the L2 VM. When prompted, login in with user: root, password: root.

    sudo qemu-system-x86_64 -enable-kvm -hda IMAGE_NAME -m 512 -net nic -net tap,ifname=tap0,script=no -curses
    

    Replace IMAGE_NAME with the name of the QEMU-compatible OS image to use for the L2 VM.

  14. On the L2 VM, run ip addr show to confirm that the VM has an address in the virbr0 space—for example, 192.168.122.89:

    user@nested-vm:~$ ip addr
    
  15. Start a placeholder web server on port 8000:

    user@nested-vm:~$ python -m http.server
    
  16. Detach from the screen session with Ctrl+A, Ctrl+D.

  17. Test that your L1 VM can ping the L2 VM, replacing the following IP address with the IP address of the L2 VM:

    curl 192.168.122.89:8000
    
  18. The output is similar to the following:

    <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 3.2 Final//EN"><html>
    <title>Directory listing for /</title>
    <body>
    <h2>Directory listing for /</h2>
    <hr>
    <ol>
    <li><a href=".aptitude/">.aptitude/</a>
    <li><a href=".bashrc">.bashrc</a>
    <li><a href=".profile">.profile</a>
    </ol>
    <hr>
    </body>
    </html>
    

Creating an L2 VM with network access from outside the L1 VM