Назад | Перейти на главную страницу

подключение USB-ключа к виртуальной машине KVM

Я изо всех сил пытаюсь подключить USB-устройство, которое правильно определяется хостом, к kvm vm.

У меня недавно установлен Ubuntu Server 14.10 в качестве хоста KVM / QEMU. Я установил Ubuntu vm с помощью этой команды:

virt-install --connect qemu:///system \
            -n test01 \
            -r 1024 \
            --vcpus=2 \
            --disk path=/vmstorage/01/test01.img,size=5 \
            --vnc \
            --noautoconsole \
            --os-variant=ubuntuutopic \
            --hvm \
            --cdrom /path/to/ubuntu-14.10-server-i386.iso

После успешной установки virsh dumpxml test01 возвращается

<domain type='kvm' id='16'>
  <name>test01</name>
  <uuid>f58ca825-c999-4168-9f5a-616057d9955d</uuid>
  <memory unit='KiB'>1048576</memory>
  <currentMemory unit='KiB'>1048576</currentMemory>
  <vcpu placement='static'>2</vcpu>
  <resource>
    <partition>/machine</partition>
  </resource>
  <os>
    <type arch='x86_64' machine='pc-i440fx-utopic'>hvm</type>
    <boot dev='hd'/>
  </os>
  <features>
    <acpi/>
    <apic/>
    <pae/>
  </features>
  <cpu mode='custom' match='exact'>
    <model fallback='allow'>SandyBridge</model>
  </cpu>
  <clock offset='utc'>
    <timer name='rtc' tickpolicy='catchup'/>
    <timer name='pit' tickpolicy='delay'/>
    <timer name='hpet' present='no'/>
  </clock>
  <on_poweroff>destroy</on_poweroff>
  <on_reboot>restart</on_reboot>
  <on_crash>restart</on_crash>
  <devices>
    <emulator>/usr/bin/kvm-spice</emulator>
    <disk type='file' device='disk'>
      <driver name='qemu' type='raw'/>
      <source file='/vmstorage/01/test01.img'/>
      <backingStore/>
      <target dev='vda' bus='virtio'/>
      <alias name='virtio-disk0'/>
      <address type='pci' domain='0x0000' bus='0x00' slot='0x05' function='0x0'/>
    </disk>
    <disk type='block' device='cdrom'>
      <driver name='qemu' type='raw'/>
      <backingStore/>
      <target dev='hda' bus='ide'/>
      <readonly/>
      <alias name='ide0-0-0'/>
      <address type='drive' controller='0' bus='0' target='0' unit='0'/>
    </disk>
    <controller type='usb' index='0' model='ich9-ehci1'>
      <alias name='usb0'/>
      <address type='pci' domain='0x0000' bus='0x00' slot='0x04' function='0x7'/>
    </controller>
    <controller type='usb' index='0' model='ich9-uhci1'>
      <alias name='usb0'/>
      <master startport='0'/>
      <address type='pci' domain='0x0000' bus='0x00' slot='0x04' function='0x0' multifunction='on'/>
    </controller>
    <controller type='usb' index='0' model='ich9-uhci2'>
      <alias name='usb0'/>
      <master startport='2'/>
      <address type='pci' domain='0x0000' bus='0x00' slot='0x04' function='0x1'/>
    </controller>
    <controller type='usb' index='0' model='ich9-uhci3'>
      <alias name='usb0'/>
      <master startport='4'/>
      <address type='pci' domain='0x0000' bus='0x00' slot='0x04' function='0x2'/>
    </controller>
    <controller type='pci' index='0' model='pci-root'>
      <alias name='pci.0'/>
    </controller>
    <controller type='ide' index='0'>
      <alias name='ide0'/>
      <address type='pci' domain='0x0000' bus='0x00' slot='0x01' function='0x1'/>
    </controller>
    <interface type='bridge'>
      <mac address='52:54:00:11:b2:c1'/>
      <source bridge='br0'/>
      <target dev='vnet0'/>
      <model type='virtio'/>
      <alias name='net0'/>
      <address type='pci' domain='0x0000' bus='0x00' slot='0x03' function='0x0'/>
    </interface>
    <serial type='pty'>
      <source path='/dev/pts/0'/>
      <target port='0'/>
      <alias name='serial0'/>
    </serial>
    <console type='pty' tty='/dev/pts/0'>
      <source path='/dev/pts/0'/>
      <target type='serial' port='0'/>
      <alias name='serial0'/>
    </console>
    <input type='mouse' bus='ps2'/>
    <input type='keyboard' bus='ps2'/>
    <graphics type='vnc' port='5900' autoport='yes' listen='0.0.0.0'>
      <listen type='address' address='0.0.0.0'/>
    </graphics>
    <video>
      <model type='cirrus' vram='9216' heads='1'/>
      <alias name='video0'/>
      <address type='pci' domain='0x0000' bus='0x00' slot='0x02' function='0x0'/>
    </video>
    <memballoon model='virtio'>
      <alias name='balloon0'/>
      <address type='pci' domain='0x0000' bus='0x00' slot='0x06' function='0x0'/>
    </memballoon>
  </devices>
  <seclabel type='dynamic' model='apparmor' relabel='yes'>
    <label>libvirt-f58ca825-c999-4168-9f5a-616057d9955d</label>
    <imagelabel>libvirt-f58ca825-c999-4168-9f5a-616057d9955d</imagelabel>
  </seclabel>
</domain>

Теперь я хочу подключить USB-накопитель. После некоторого поиска в Google я нашел некоторые источники все это в основном предполагает следующий подход:

  1. прикрепить палку к хосту
  2. получить идентификатор поставщика и продукта

    root@host01:~# lsusb
    Bus 002 Device 004: ID 13fe:5100 Kingston Technology Company Inc. Flash Drive
    Bus 002 Device 002: ID 8087:0024 Intel Corp. Integrated Rate Matching Hub
    Bus 002 Device 001: ID 1d6b:0002 Linux Foundation 2.0 root hub
    Bus 001 Device 004: ID 0624:0249 Avocent Corp.
    Bus 001 Device 003: ID 0624:0248 Avocent Corp.
    Bus 001 Device 002: ID 8087:0024 Intel Corp. Integrated Rate Matching Hub
    Bus 001 Device 001: ID 1d6b:0002 Linux Foundation 2.0 root hub
    
  3. добавить новый фрагмент конфигурации в виртуальную машину через virsh edit test01

    <devices>
      <!- ... ->
      <hostdev mode='subsystem' type='usb' managed='yes'>
        <source>
          <vendor id='0x13fe'/>
          <product id='0x5100'/>
        </source>
      </hostdev>
    </devices/>
    
  4. перезапустить vm

После этого vm должен увидеть флешку через lsusb, но ничего не меняется. Если я попытаюсь подключить устройство ко второму виртуальному компьютеру, virsh start <othervm> терпит неудачу с error: Requested operation is not valid: USB device 002:003 is in use by driver QEMU, domain test01 НО Я все еще могу установить и получить доступ к флешке на хосте. Я пробовал разные флешки и разные vm-операционные системы (ubuntu и windows), безуспешно.

В некоторых инструкциях предлагается отключить apparmor, но /etc/init.d/apparmor stop ничего не менял.

Это сводит меня с ума, так как я не получаю никаких сообщений об ошибках или сообщениях журнала, и я не знаю, как узнать, что не так. Есть идеи о том, как подключить usb к виртуальной машине или, по крайней мере, как его проанализировать дальше?

Чтобы предоставить постоянный доступ к необработанным узлам USB-устройства пользователю, от имени которого работает гипервизор, вам необходимо создать правило udev; ответ на основе chown будет работать только до следующей перезагрузки.

В /lib/udev/rules.d, создайте файл вроде 51-usb_passthrough.rules :

SUBSYSTEM=="usb", ENV{DEVTYPE}=="usb_device", ATTRS{busnum}=="2" ATTRS{devpath}=="1" GROUP="kvm"

Здесь я использовал номера физической шины и портов для нацеливания на устройство (так как я предпочитаю проходить через физический порт независимо от того, какое устройство подключено, а не перенастраивать мою виртуальную машину каждый раз, когда я подключаю новое устройство), но, конечно, вы можете использовать любые атрибуты вы хотите; в GROUP аргумент определяет группу, которая будет владеть узлом устройства, это должен быть любой пользователь, запускающий виртуальные машины.

Бегать udevadm control --reload-rules чтобы новое правило вступило в силу немедленно (вам все равно нужно отключить / повторно подключить USB-устройства) или просто перезагрузите хост.

Наверное проблема с правами доступа. Вашему демону QEMU запрещен доступ к USB-устройствам. Пытаться:

chown libvirt-qemu /dev/bus/usb/ -R

или каким бы пользователем ни был ваш KVM. Это должно помочь.

Простое решение: подключитесь к

nc -U socket-file

на ваш монитор QEMU, где файл-сокет - это путь для мониторинга. Как найти: ps -FA | grep qemu. Вы найдете там в командной строке.

И запускаем на мониторе команду:

device_add usb-host,id=<ANY string>,hostbus=<BUS>,hostport=<PORT>

это сквозная передача по номеру порта. Вы можете пройти через любое устройство с любым VID / PID. Вы также можете передавать определенные устройства, используя

device_add usb-host,id=<ANY string>,vendorid=0x0461,productid= 0x0010

и их можно подключить к любому порту вашего хоста.

Также полезные команды:

info usbhost
info usb