Eduardo Habkost | f6e501a | 2018-12-05 17:57:04 -0200 | [diff] [blame] | 1 | """ |
| 2 | Check compatibility of virtio device types |
| 3 | """ |
| 4 | # Copyright (c) 2018 Red Hat, Inc. |
| 5 | # |
| 6 | # Author: |
| 7 | # Eduardo Habkost <ehabkost@redhat.com> |
| 8 | # |
| 9 | # This work is licensed under the terms of the GNU GPL, version 2 or |
| 10 | # later. See the COPYING file in the top-level directory. |
| 11 | import sys |
| 12 | import os |
| 13 | |
John Snow | abf0bf9 | 2019-06-27 17:28:14 -0400 | [diff] [blame] | 14 | from qemu.machine import QEMUMachine |
Philippe Mathieu-Daudé | 2283b62 | 2021-09-27 18:14:33 +0200 | [diff] [blame] | 15 | from avocado_qemu import QemuSystemTest |
Eduardo Habkost | f6e501a | 2018-12-05 17:57:04 -0200 | [diff] [blame] | 16 | |
| 17 | # Virtio Device IDs: |
| 18 | VIRTIO_NET = 1 |
| 19 | VIRTIO_BLOCK = 2 |
| 20 | VIRTIO_CONSOLE = 3 |
| 21 | VIRTIO_RNG = 4 |
| 22 | VIRTIO_BALLOON = 5 |
| 23 | VIRTIO_RPMSG = 7 |
| 24 | VIRTIO_SCSI = 8 |
| 25 | VIRTIO_9P = 9 |
| 26 | VIRTIO_RPROC_SERIAL = 11 |
| 27 | VIRTIO_CAIF = 12 |
| 28 | VIRTIO_GPU = 16 |
| 29 | VIRTIO_INPUT = 18 |
| 30 | VIRTIO_VSOCK = 19 |
| 31 | VIRTIO_CRYPTO = 20 |
| 32 | |
| 33 | PCI_VENDOR_ID_REDHAT_QUMRANET = 0x1af4 |
| 34 | |
| 35 | # Device IDs for legacy/transitional devices: |
| 36 | PCI_LEGACY_DEVICE_IDS = { |
| 37 | VIRTIO_NET: 0x1000, |
| 38 | VIRTIO_BLOCK: 0x1001, |
| 39 | VIRTIO_BALLOON: 0x1002, |
| 40 | VIRTIO_CONSOLE: 0x1003, |
| 41 | VIRTIO_SCSI: 0x1004, |
| 42 | VIRTIO_RNG: 0x1005, |
| 43 | VIRTIO_9P: 0x1009, |
| 44 | VIRTIO_VSOCK: 0x1012, |
| 45 | } |
| 46 | |
| 47 | def pci_modern_device_id(virtio_devid): |
| 48 | return virtio_devid + 0x1040 |
| 49 | |
| 50 | def devtype_implements(vm, devtype, implements): |
| 51 | return devtype in [d['name'] for d in vm.command('qom-list-types', implements=implements)] |
| 52 | |
| 53 | def get_pci_interfaces(vm, devtype): |
| 54 | interfaces = ('pci-express-device', 'conventional-pci-device') |
| 55 | return [i for i in interfaces if devtype_implements(vm, devtype, i)] |
| 56 | |
Philippe Mathieu-Daudé | 2283b62 | 2021-09-27 18:14:33 +0200 | [diff] [blame] | 57 | class VirtioVersionCheck(QemuSystemTest): |
Eduardo Habkost | f6e501a | 2018-12-05 17:57:04 -0200 | [diff] [blame] | 58 | """ |
| 59 | Check if virtio-version-specific device types result in the |
| 60 | same device tree created by `disable-modern` and |
| 61 | `disable-legacy`. |
| 62 | |
Cleber Rosa | b194713 | 2019-03-12 13:18:09 -0400 | [diff] [blame] | 63 | :avocado: tags=arch:x86_64 |
Eduardo Habkost | f6e501a | 2018-12-05 17:57:04 -0200 | [diff] [blame] | 64 | """ |
| 65 | |
| 66 | # just in case there are failures, show larger diff: |
| 67 | maxDiff = 4096 |
| 68 | |
| 69 | def run_device(self, devtype, opts=None, machine='pc'): |
| 70 | """ |
| 71 | Run QEMU with `-device DEVTYPE`, return device info from `query-pci` |
| 72 | """ |
| 73 | with QEMUMachine(self.qemu_bin) as vm: |
| 74 | vm.set_machine(machine) |
| 75 | if opts: |
| 76 | devtype += ',' + opts |
| 77 | vm.add_args('-device', '%s,id=devfortest' % (devtype)) |
| 78 | vm.add_args('-S') |
| 79 | vm.launch() |
| 80 | |
| 81 | pcibuses = vm.command('query-pci') |
| 82 | alldevs = [dev for bus in pcibuses for dev in bus['devices']] |
| 83 | devfortest = [dev for dev in alldevs |
| 84 | if dev['qdev_id'] == 'devfortest'] |
| 85 | return devfortest[0], get_pci_interfaces(vm, devtype) |
| 86 | |
| 87 | |
| 88 | def assert_devids(self, dev, devid, non_transitional=False): |
| 89 | self.assertEqual(dev['id']['vendor'], PCI_VENDOR_ID_REDHAT_QUMRANET) |
| 90 | self.assertEqual(dev['id']['device'], devid) |
| 91 | if non_transitional: |
| 92 | self.assertTrue(0x1040 <= dev['id']['device'] <= 0x107f) |
| 93 | self.assertGreaterEqual(dev['id']['subsystem'], 0x40) |
| 94 | |
| 95 | def check_all_variants(self, qemu_devtype, virtio_devid): |
| 96 | """Check if a virtio device type and its variants behave as expected""" |
| 97 | # Force modern mode: |
| 98 | dev_modern, _ = self.run_device(qemu_devtype, |
| 99 | 'disable-modern=off,disable-legacy=on') |
| 100 | self.assert_devids(dev_modern, pci_modern_device_id(virtio_devid), |
| 101 | non_transitional=True) |
| 102 | |
| 103 | # <prefix>-non-transitional device types should be 100% equivalent to |
| 104 | # <prefix>,disable-modern=off,disable-legacy=on |
| 105 | dev_1_0, nt_ifaces = self.run_device('%s-non-transitional' % (qemu_devtype)) |
| 106 | self.assertEqual(dev_modern, dev_1_0) |
| 107 | |
| 108 | # Force transitional mode: |
| 109 | dev_trans, _ = self.run_device(qemu_devtype, |
| 110 | 'disable-modern=off,disable-legacy=off') |
| 111 | self.assert_devids(dev_trans, PCI_LEGACY_DEVICE_IDS[virtio_devid]) |
| 112 | |
| 113 | # Force legacy mode: |
| 114 | dev_legacy, _ = self.run_device(qemu_devtype, |
| 115 | 'disable-modern=on,disable-legacy=off') |
| 116 | self.assert_devids(dev_legacy, PCI_LEGACY_DEVICE_IDS[virtio_devid]) |
| 117 | |
| 118 | # No options: default to transitional on PC machine-type: |
| 119 | no_opts_pc, generic_ifaces = self.run_device(qemu_devtype) |
| 120 | self.assertEqual(dev_trans, no_opts_pc) |
| 121 | |
| 122 | #TODO: check if plugging on a PCI Express bus will make the |
| 123 | # device non-transitional |
| 124 | #no_opts_q35 = self.run_device(qemu_devtype, machine='q35') |
| 125 | #self.assertEqual(dev_modern, no_opts_q35) |
| 126 | |
| 127 | # <prefix>-transitional device types should be 100% equivalent to |
| 128 | # <prefix>,disable-modern=off,disable-legacy=off |
| 129 | dev_trans, trans_ifaces = self.run_device('%s-transitional' % (qemu_devtype)) |
| 130 | self.assertEqual(dev_trans, dev_trans) |
| 131 | |
| 132 | # ensure the interface information is correct: |
| 133 | self.assertIn('conventional-pci-device', generic_ifaces) |
| 134 | self.assertIn('pci-express-device', generic_ifaces) |
| 135 | |
| 136 | self.assertIn('conventional-pci-device', nt_ifaces) |
| 137 | self.assertIn('pci-express-device', nt_ifaces) |
| 138 | |
| 139 | self.assertIn('conventional-pci-device', trans_ifaces) |
| 140 | self.assertNotIn('pci-express-device', trans_ifaces) |
| 141 | |
| 142 | |
| 143 | def test_conventional_devs(self): |
| 144 | self.check_all_variants('virtio-net-pci', VIRTIO_NET) |
| 145 | # virtio-blk requires 'driver' parameter |
| 146 | #self.check_all_variants('virtio-blk-pci', VIRTIO_BLOCK) |
| 147 | self.check_all_variants('virtio-serial-pci', VIRTIO_CONSOLE) |
| 148 | self.check_all_variants('virtio-rng-pci', VIRTIO_RNG) |
| 149 | self.check_all_variants('virtio-balloon-pci', VIRTIO_BALLOON) |
| 150 | self.check_all_variants('virtio-scsi-pci', VIRTIO_SCSI) |
| 151 | # virtio-9p requires 'fsdev' parameter |
| 152 | #self.check_all_variants('virtio-9p-pci', VIRTIO_9P) |
| 153 | |
| 154 | def check_modern_only(self, qemu_devtype, virtio_devid): |
| 155 | """Check if a modern-only virtio device type behaves as expected""" |
| 156 | # Force modern mode: |
| 157 | dev_modern, _ = self.run_device(qemu_devtype, |
| 158 | 'disable-modern=off,disable-legacy=on') |
| 159 | self.assert_devids(dev_modern, pci_modern_device_id(virtio_devid), |
| 160 | non_transitional=True) |
| 161 | |
| 162 | # No options: should be modern anyway |
| 163 | dev_no_opts, ifaces = self.run_device(qemu_devtype) |
| 164 | self.assertEqual(dev_modern, dev_no_opts) |
| 165 | |
| 166 | self.assertIn('conventional-pci-device', ifaces) |
| 167 | self.assertIn('pci-express-device', ifaces) |
| 168 | |
| 169 | def test_modern_only_devs(self): |
| 170 | self.check_modern_only('virtio-vga', VIRTIO_GPU) |
| 171 | self.check_modern_only('virtio-gpu-pci', VIRTIO_GPU) |
| 172 | self.check_modern_only('virtio-mouse-pci', VIRTIO_INPUT) |
| 173 | self.check_modern_only('virtio-tablet-pci', VIRTIO_INPUT) |
| 174 | self.check_modern_only('virtio-keyboard-pci', VIRTIO_INPUT) |