Marc Marí | 311e666 | 2014-09-01 12:07:54 +0200 | [diff] [blame] | 1 | /* |
| 2 | * libqos virtio PCI driver |
| 3 | * |
| 4 | * Copyright (c) 2014 Marc Marí |
| 5 | * |
| 6 | * This work is licensed under the terms of the GNU GPL, version 2 or later. |
| 7 | * See the COPYING file in the top-level directory. |
| 8 | */ |
| 9 | |
Peter Maydell | 681c28a | 2016-02-08 18:08:51 +0000 | [diff] [blame] | 10 | #include "qemu/osdep.h" |
Marc-André Lureau | 907b510 | 2022-03-30 13:39:05 +0400 | [diff] [blame] | 11 | #include "../libqtest.h" |
Paolo Bonzini | a2ce7db | 2020-08-04 20:00:40 +0200 | [diff] [blame] | 12 | #include "virtio.h" |
| 13 | #include "virtio-pci.h" |
| 14 | #include "pci.h" |
| 15 | #include "pci-pc.h" |
Xuzhou Cheng | b243c73 | 2022-08-24 17:40:03 +0800 | [diff] [blame] | 16 | #include "libqos-malloc.h" |
Paolo Bonzini | a2ce7db | 2020-08-04 20:00:40 +0200 | [diff] [blame] | 17 | #include "malloc-pc.h" |
| 18 | #include "qgraph.h" |
Stefan Hajnoczi | ee3b850 | 2016-05-09 13:47:37 +0100 | [diff] [blame] | 19 | #include "standard-headers/linux/virtio_ring.h" |
Stefan Hajnoczi | c75f4c0 | 2016-05-09 13:47:41 +0100 | [diff] [blame] | 20 | #include "standard-headers/linux/virtio_pci.h" |
Marc Marí | 311e666 | 2014-09-01 12:07:54 +0200 | [diff] [blame] | 21 | |
Stefan Hajnoczi | 7ad1e70 | 2016-05-09 13:47:35 +0100 | [diff] [blame] | 22 | #include "hw/pci/pci.h" |
Marc Marí | 311e666 | 2014-09-01 12:07:54 +0200 | [diff] [blame] | 23 | #include "hw/pci/pci_regs.h" |
| 24 | |
Stefan Hajnoczi | d08f68b | 2019-10-23 11:04:25 +0100 | [diff] [blame] | 25 | #include "virtio-pci-modern.h" |
| 26 | |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 27 | /* virtio-pci is a superclass of all virtio-xxx-pci devices; |
| 28 | * the relation between virtio-pci and virtio-xxx-pci is implicit, |
| 29 | * and therefore virtio-pci does not produce virtio and is not |
| 30 | * reached by any edge, not even as a "contains" edge. |
| 31 | * In facts, every device is a QVirtioPCIDevice with |
| 32 | * additional fields, since every one has its own |
| 33 | * number of queues and various attributes. |
| 34 | * Virtio-pci provides default functions to start the |
| 35 | * hw and destroy the object, and nodes that want to |
| 36 | * override them should always remember to call the |
| 37 | * original qvirtio_pci_destructor and qvirtio_pci_start_hw. |
| 38 | */ |
| 39 | |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 40 | #define CONFIG_BASE(dev) (VIRTIO_PCI_CONFIG_OFF((dev)->pdev->msix_enabled)) |
David Gibson | 246fc0f | 2016-10-20 14:08:07 +1100 | [diff] [blame] | 41 | |
| 42 | static uint8_t qvirtio_pci_config_readb(QVirtioDevice *d, uint64_t off) |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 43 | { |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 44 | QVirtioPCIDevice *dev = container_of(d, QVirtioPCIDevice, vdev); |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 45 | return qpci_io_readb(dev->pdev, dev->bar, CONFIG_BASE(dev) + off); |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 46 | } |
| 47 | |
Laurent Vivier | 30ca440 | 2016-10-17 12:30:24 +0200 | [diff] [blame] | 48 | /* PCI is always read in little-endian order |
| 49 | * but virtio ( < 1.0) is in guest order |
| 50 | * so with a big-endian guest the order has been reversed, |
| 51 | * reverse it again |
Stefan Hajnoczi | 9598f9e | 2019-10-23 11:04:24 +0100 | [diff] [blame] | 52 | * virtio-1.0 is always little-endian, like PCI |
Laurent Vivier | 30ca440 | 2016-10-17 12:30:24 +0200 | [diff] [blame] | 53 | */ |
| 54 | |
David Gibson | 246fc0f | 2016-10-20 14:08:07 +1100 | [diff] [blame] | 55 | static uint16_t qvirtio_pci_config_readw(QVirtioDevice *d, uint64_t off) |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 56 | { |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 57 | QVirtioPCIDevice *dev = container_of(d, QVirtioPCIDevice, vdev); |
Laurent Vivier | 30ca440 | 2016-10-17 12:30:24 +0200 | [diff] [blame] | 58 | uint16_t value; |
| 59 | |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 60 | value = qpci_io_readw(dev->pdev, dev->bar, CONFIG_BASE(dev) + off); |
Laurent Vivier | 30ca440 | 2016-10-17 12:30:24 +0200 | [diff] [blame] | 61 | if (qvirtio_is_big_endian(d)) { |
| 62 | value = bswap16(value); |
| 63 | } |
| 64 | return value; |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 65 | } |
| 66 | |
David Gibson | 246fc0f | 2016-10-20 14:08:07 +1100 | [diff] [blame] | 67 | static uint32_t qvirtio_pci_config_readl(QVirtioDevice *d, uint64_t off) |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 68 | { |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 69 | QVirtioPCIDevice *dev = container_of(d, QVirtioPCIDevice, vdev); |
Laurent Vivier | 30ca440 | 2016-10-17 12:30:24 +0200 | [diff] [blame] | 70 | uint32_t value; |
| 71 | |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 72 | value = qpci_io_readl(dev->pdev, dev->bar, CONFIG_BASE(dev) + off); |
Laurent Vivier | 30ca440 | 2016-10-17 12:30:24 +0200 | [diff] [blame] | 73 | if (qvirtio_is_big_endian(d)) { |
| 74 | value = bswap32(value); |
| 75 | } |
| 76 | return value; |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 77 | } |
| 78 | |
David Gibson | 246fc0f | 2016-10-20 14:08:07 +1100 | [diff] [blame] | 79 | static uint64_t qvirtio_pci_config_readq(QVirtioDevice *d, uint64_t off) |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 80 | { |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 81 | QVirtioPCIDevice *dev = container_of(d, QVirtioPCIDevice, vdev); |
David Gibson | f775f45 | 2016-10-19 15:00:21 +1100 | [diff] [blame] | 82 | uint64_t val; |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 83 | |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 84 | val = qpci_io_readq(dev->pdev, dev->bar, CONFIG_BASE(dev) + off); |
Laurent Vivier | 8b4b80c | 2016-10-17 12:30:22 +0200 | [diff] [blame] | 85 | if (qvirtio_is_big_endian(d)) { |
David Gibson | f775f45 | 2016-10-19 15:00:21 +1100 | [diff] [blame] | 86 | val = bswap64(val); |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 87 | } |
| 88 | |
David Gibson | f775f45 | 2016-10-19 15:00:21 +1100 | [diff] [blame] | 89 | return val; |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 90 | } |
| 91 | |
Stefan Hajnoczi | a934035 | 2019-10-23 11:04:12 +0100 | [diff] [blame] | 92 | static uint64_t qvirtio_pci_get_features(QVirtioDevice *d) |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 93 | { |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 94 | QVirtioPCIDevice *dev = container_of(d, QVirtioPCIDevice, vdev); |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 95 | return qpci_io_readl(dev->pdev, dev->bar, VIRTIO_PCI_HOST_FEATURES); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 96 | } |
| 97 | |
Stefan Hajnoczi | a934035 | 2019-10-23 11:04:12 +0100 | [diff] [blame] | 98 | static void qvirtio_pci_set_features(QVirtioDevice *d, uint64_t features) |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 99 | { |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 100 | QVirtioPCIDevice *dev = container_of(d, QVirtioPCIDevice, vdev); |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 101 | qpci_io_writel(dev->pdev, dev->bar, VIRTIO_PCI_GUEST_FEATURES, features); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 102 | } |
| 103 | |
Stefan Hajnoczi | a934035 | 2019-10-23 11:04:12 +0100 | [diff] [blame] | 104 | static uint64_t qvirtio_pci_get_guest_features(QVirtioDevice *d) |
Marc Marí | f294b02 | 2014-09-01 12:07:57 +0200 | [diff] [blame] | 105 | { |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 106 | QVirtioPCIDevice *dev = container_of(d, QVirtioPCIDevice, vdev); |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 107 | return qpci_io_readl(dev->pdev, dev->bar, VIRTIO_PCI_GUEST_FEATURES); |
Marc Marí | f294b02 | 2014-09-01 12:07:57 +0200 | [diff] [blame] | 108 | } |
| 109 | |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 110 | static uint8_t qvirtio_pci_get_status(QVirtioDevice *d) |
| 111 | { |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 112 | QVirtioPCIDevice *dev = container_of(d, QVirtioPCIDevice, vdev); |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 113 | return qpci_io_readb(dev->pdev, dev->bar, VIRTIO_PCI_STATUS); |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 114 | } |
| 115 | |
| 116 | static void qvirtio_pci_set_status(QVirtioDevice *d, uint8_t status) |
| 117 | { |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 118 | QVirtioPCIDevice *dev = container_of(d, QVirtioPCIDevice, vdev); |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 119 | qpci_io_writeb(dev->pdev, dev->bar, VIRTIO_PCI_STATUS, status); |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 120 | } |
| 121 | |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 122 | static bool qvirtio_pci_get_queue_isr_status(QVirtioDevice *d, QVirtQueue *vq) |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 123 | { |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 124 | QVirtioPCIDevice *dev = container_of(d, QVirtioPCIDevice, vdev); |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 125 | QVirtQueuePCI *vqpci = (QVirtQueuePCI *)vq; |
| 126 | uint32_t data; |
| 127 | |
| 128 | if (dev->pdev->msix_enabled) { |
| 129 | g_assert_cmpint(vqpci->msix_entry, !=, -1); |
| 130 | if (qpci_msix_masked(dev->pdev, vqpci->msix_entry)) { |
| 131 | /* No ISR checking should be done if masked, but read anyway */ |
| 132 | return qpci_msix_pending(dev->pdev, vqpci->msix_entry); |
| 133 | } else { |
Thomas Huth | 2380d31 | 2019-07-18 17:08:51 +0200 | [diff] [blame] | 134 | data = qtest_readl(dev->pdev->bus->qts, vqpci->msix_addr); |
Marc Marí | 1e34cf9 | 2015-02-24 17:34:14 +0100 | [diff] [blame] | 135 | if (data == vqpci->msix_data) { |
Thomas Huth | 2380d31 | 2019-07-18 17:08:51 +0200 | [diff] [blame] | 136 | qtest_writel(dev->pdev->bus->qts, vqpci->msix_addr, 0); |
Marc Marí | 1e34cf9 | 2015-02-24 17:34:14 +0100 | [diff] [blame] | 137 | return true; |
| 138 | } else { |
| 139 | return false; |
| 140 | } |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 141 | } |
| 142 | } else { |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 143 | return qpci_io_readb(dev->pdev, dev->bar, VIRTIO_PCI_ISR) & 1; |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 144 | } |
| 145 | } |
| 146 | |
| 147 | static bool qvirtio_pci_get_config_isr_status(QVirtioDevice *d) |
| 148 | { |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 149 | QVirtioPCIDevice *dev = container_of(d, QVirtioPCIDevice, vdev); |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 150 | uint32_t data; |
| 151 | |
| 152 | if (dev->pdev->msix_enabled) { |
| 153 | g_assert_cmpint(dev->config_msix_entry, !=, -1); |
| 154 | if (qpci_msix_masked(dev->pdev, dev->config_msix_entry)) { |
| 155 | /* No ISR checking should be done if masked, but read anyway */ |
| 156 | return qpci_msix_pending(dev->pdev, dev->config_msix_entry); |
| 157 | } else { |
Thomas Huth | 2380d31 | 2019-07-18 17:08:51 +0200 | [diff] [blame] | 158 | data = qtest_readl(dev->pdev->bus->qts, dev->config_msix_addr); |
Marc Marí | 1e34cf9 | 2015-02-24 17:34:14 +0100 | [diff] [blame] | 159 | if (data == dev->config_msix_data) { |
Thomas Huth | 2380d31 | 2019-07-18 17:08:51 +0200 | [diff] [blame] | 160 | qtest_writel(dev->pdev->bus->qts, dev->config_msix_addr, 0); |
Marc Marí | 1e34cf9 | 2015-02-24 17:34:14 +0100 | [diff] [blame] | 161 | return true; |
| 162 | } else { |
| 163 | return false; |
| 164 | } |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 165 | } |
| 166 | } else { |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 167 | return qpci_io_readb(dev->pdev, dev->bar, VIRTIO_PCI_ISR) & 2; |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 168 | } |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 169 | } |
| 170 | |
Thomas Huth | b57ebd5 | 2019-09-03 08:18:46 +0200 | [diff] [blame] | 171 | static void qvirtio_pci_wait_config_isr_status(QVirtioDevice *d, |
| 172 | gint64 timeout_us) |
| 173 | { |
| 174 | QVirtioPCIDevice *dev = container_of(d, QVirtioPCIDevice, vdev); |
| 175 | gint64 start_time = g_get_monotonic_time(); |
| 176 | |
| 177 | do { |
| 178 | g_assert(g_get_monotonic_time() - start_time <= timeout_us); |
| 179 | qtest_clock_step(dev->pdev->bus->qts, 100); |
| 180 | } while (!qvirtio_pci_get_config_isr_status(d)); |
| 181 | } |
| 182 | |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 183 | static void qvirtio_pci_queue_select(QVirtioDevice *d, uint16_t index) |
| 184 | { |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 185 | QVirtioPCIDevice *dev = container_of(d, QVirtioPCIDevice, vdev); |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 186 | qpci_io_writeb(dev->pdev, dev->bar, VIRTIO_PCI_QUEUE_SEL, index); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 187 | } |
| 188 | |
| 189 | static uint16_t qvirtio_pci_get_queue_size(QVirtioDevice *d) |
| 190 | { |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 191 | QVirtioPCIDevice *dev = container_of(d, QVirtioPCIDevice, vdev); |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 192 | return qpci_io_readw(dev->pdev, dev->bar, VIRTIO_PCI_QUEUE_NUM); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 193 | } |
| 194 | |
Stefan Hajnoczi | 1e59a86 | 2019-10-23 11:04:20 +0100 | [diff] [blame] | 195 | static void qvirtio_pci_set_queue_address(QVirtioDevice *d, QVirtQueue *vq) |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 196 | { |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 197 | QVirtioPCIDevice *dev = container_of(d, QVirtioPCIDevice, vdev); |
Stefan Hajnoczi | 1e59a86 | 2019-10-23 11:04:20 +0100 | [diff] [blame] | 198 | uint64_t pfn = vq->desc / VIRTIO_PCI_VRING_ALIGN; |
| 199 | |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 200 | qpci_io_writel(dev->pdev, dev->bar, VIRTIO_PCI_QUEUE_PFN, pfn); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 201 | } |
| 202 | |
Stefan Hajnoczi | e56536b | 2019-10-23 11:04:22 +0100 | [diff] [blame] | 203 | QVirtQueue *qvirtio_pci_virtqueue_setup_common(QVirtioDevice *d, |
| 204 | QGuestAllocator *alloc, |
| 205 | uint16_t index) |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 206 | { |
Stefan Hajnoczi | a934035 | 2019-10-23 11:04:12 +0100 | [diff] [blame] | 207 | uint64_t feat; |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 208 | uint64_t addr; |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 209 | QVirtQueuePCI *vqpci; |
Thomas Huth | 8b898f5 | 2019-05-15 19:43:23 +0200 | [diff] [blame] | 210 | QVirtioPCIDevice *qvpcidev = container_of(d, QVirtioPCIDevice, vdev); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 211 | |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 212 | vqpci = g_malloc0(sizeof(*vqpci)); |
Stefan Hajnoczi | e56536b | 2019-10-23 11:04:22 +0100 | [diff] [blame] | 213 | feat = d->bus->get_guest_features(d); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 214 | |
Stefan Hajnoczi | e56536b | 2019-10-23 11:04:22 +0100 | [diff] [blame] | 215 | d->bus->queue_select(d, index); |
Stefan Hajnoczi | bccd82b | 2019-10-23 11:04:18 +0100 | [diff] [blame] | 216 | vqpci->vq.vdev = d; |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 217 | vqpci->vq.index = index; |
Stefan Hajnoczi | e56536b | 2019-10-23 11:04:22 +0100 | [diff] [blame] | 218 | vqpci->vq.size = d->bus->get_queue_size(d); |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 219 | vqpci->vq.free_head = 0; |
| 220 | vqpci->vq.num_free = vqpci->vq.size; |
Stefan Hajnoczi | c75f4c0 | 2016-05-09 13:47:41 +0100 | [diff] [blame] | 221 | vqpci->vq.align = VIRTIO_PCI_VRING_ALIGN; |
Stefan Hajnoczi | a934035 | 2019-10-23 11:04:12 +0100 | [diff] [blame] | 222 | vqpci->vq.indirect = feat & (1ull << VIRTIO_RING_F_INDIRECT_DESC); |
| 223 | vqpci->vq.event = feat & (1ull << VIRTIO_RING_F_EVENT_IDX); |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 224 | |
| 225 | vqpci->msix_entry = -1; |
| 226 | vqpci->msix_addr = 0; |
| 227 | vqpci->msix_data = 0x12345678; |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 228 | |
| 229 | /* Check different than 0 */ |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 230 | g_assert_cmpint(vqpci->vq.size, !=, 0); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 231 | |
| 232 | /* Check power of 2 */ |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 233 | g_assert_cmpint(vqpci->vq.size & (vqpci->vq.size - 1), ==, 0); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 234 | |
Stefan Hajnoczi | c75f4c0 | 2016-05-09 13:47:41 +0100 | [diff] [blame] | 235 | addr = guest_alloc(alloc, qvring_size(vqpci->vq.size, |
| 236 | VIRTIO_PCI_VRING_ALIGN)); |
Thomas Huth | 8b898f5 | 2019-05-15 19:43:23 +0200 | [diff] [blame] | 237 | qvring_init(qvpcidev->pdev->bus->qts, alloc, &vqpci->vq, addr); |
Stefan Hajnoczi | e56536b | 2019-10-23 11:04:22 +0100 | [diff] [blame] | 238 | d->bus->set_queue_address(d, &vqpci->vq); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 239 | |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 240 | return &vqpci->vq; |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 241 | } |
| 242 | |
Stefan Hajnoczi | e56536b | 2019-10-23 11:04:22 +0100 | [diff] [blame] | 243 | void qvirtio_pci_virtqueue_cleanup_common(QVirtQueue *vq, |
Stefan Hajnoczi | f1d3b99 | 2016-05-05 16:53:35 +0100 | [diff] [blame] | 244 | QGuestAllocator *alloc) |
| 245 | { |
| 246 | QVirtQueuePCI *vqpci = container_of(vq, QVirtQueuePCI, vq); |
| 247 | |
| 248 | guest_free(alloc, vq->desc); |
| 249 | g_free(vqpci); |
| 250 | } |
| 251 | |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 252 | static void qvirtio_pci_virtqueue_kick(QVirtioDevice *d, QVirtQueue *vq) |
| 253 | { |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 254 | QVirtioPCIDevice *dev = container_of(d, QVirtioPCIDevice, vdev); |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 255 | qpci_io_writew(dev->pdev, dev->bar, VIRTIO_PCI_QUEUE_NOTIFY, vq->index); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 256 | } |
| 257 | |
Stefan Hajnoczi | 9598f9e | 2019-10-23 11:04:24 +0100 | [diff] [blame] | 258 | static const QVirtioBus qvirtio_pci_legacy = { |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 259 | .config_readb = qvirtio_pci_config_readb, |
| 260 | .config_readw = qvirtio_pci_config_readw, |
| 261 | .config_readl = qvirtio_pci_config_readl, |
| 262 | .config_readq = qvirtio_pci_config_readq, |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 263 | .get_features = qvirtio_pci_get_features, |
| 264 | .set_features = qvirtio_pci_set_features, |
Marc Marí | f294b02 | 2014-09-01 12:07:57 +0200 | [diff] [blame] | 265 | .get_guest_features = qvirtio_pci_get_guest_features, |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 266 | .get_status = qvirtio_pci_get_status, |
| 267 | .set_status = qvirtio_pci_set_status, |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 268 | .get_queue_isr_status = qvirtio_pci_get_queue_isr_status, |
Thomas Huth | b57ebd5 | 2019-09-03 08:18:46 +0200 | [diff] [blame] | 269 | .wait_config_isr_status = qvirtio_pci_wait_config_isr_status, |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 270 | .queue_select = qvirtio_pci_queue_select, |
| 271 | .get_queue_size = qvirtio_pci_get_queue_size, |
| 272 | .set_queue_address = qvirtio_pci_set_queue_address, |
Stefan Hajnoczi | e56536b | 2019-10-23 11:04:22 +0100 | [diff] [blame] | 273 | .virtqueue_setup = qvirtio_pci_virtqueue_setup_common, |
| 274 | .virtqueue_cleanup = qvirtio_pci_virtqueue_cleanup_common, |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 275 | .virtqueue_kick = qvirtio_pci_virtqueue_kick, |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 276 | }; |
| 277 | |
Stefan Hajnoczi | 957d8d1 | 2019-10-23 11:04:21 +0100 | [diff] [blame] | 278 | static void qvirtio_pci_set_config_vector(QVirtioPCIDevice *d, uint16_t entry) |
| 279 | { |
| 280 | uint16_t vector; |
| 281 | |
| 282 | qpci_io_writew(d->pdev, d->bar, VIRTIO_MSI_CONFIG_VECTOR, entry); |
| 283 | vector = qpci_io_readw(d->pdev, d->bar, VIRTIO_MSI_CONFIG_VECTOR); |
| 284 | g_assert_cmphex(vector, !=, VIRTIO_MSI_NO_VECTOR); |
| 285 | } |
| 286 | |
| 287 | static void qvirtio_pci_set_queue_vector(QVirtioPCIDevice *d, uint16_t vq_idx, |
| 288 | uint16_t entry) |
| 289 | { |
| 290 | uint16_t vector; |
| 291 | |
| 292 | qvirtio_pci_queue_select(&d->vdev, vq_idx); |
| 293 | qpci_io_writew(d->pdev, d->bar, VIRTIO_MSI_QUEUE_VECTOR, entry); |
| 294 | vector = qpci_io_readw(d->pdev, d->bar, VIRTIO_MSI_QUEUE_VECTOR); |
| 295 | g_assert_cmphex(vector, !=, VIRTIO_MSI_NO_VECTOR); |
| 296 | } |
| 297 | |
| 298 | static const QVirtioPCIMSIXOps qvirtio_pci_msix_ops_legacy = { |
| 299 | .set_config_vector = qvirtio_pci_set_config_vector, |
| 300 | .set_queue_vector = qvirtio_pci_set_queue_vector, |
| 301 | }; |
| 302 | |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 303 | void qvirtio_pci_device_enable(QVirtioPCIDevice *d) |
| 304 | { |
| 305 | qpci_device_enable(d->pdev); |
Stefan Hajnoczi | f17429e | 2019-10-23 11:04:23 +0100 | [diff] [blame] | 306 | d->bar = qpci_iomap(d->pdev, d->bar_idx, NULL); |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 307 | } |
| 308 | |
| 309 | void qvirtio_pci_device_disable(QVirtioPCIDevice *d) |
| 310 | { |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 311 | qpci_iounmap(d->pdev, d->bar); |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 312 | } |
| 313 | |
| 314 | void qvirtqueue_pci_msix_setup(QVirtioPCIDevice *d, QVirtQueuePCI *vqpci, |
| 315 | QGuestAllocator *alloc, uint16_t entry) |
| 316 | { |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 317 | uint32_t control; |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 318 | uint64_t off; |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 319 | |
| 320 | g_assert(d->pdev->msix_enabled); |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 321 | off = d->pdev->msix_table_off + (entry * 16); |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 322 | |
| 323 | g_assert_cmpint(entry, >=, 0); |
| 324 | g_assert_cmpint(entry, <, qpci_msix_table_size(d->pdev)); |
| 325 | vqpci->msix_entry = entry; |
| 326 | |
| 327 | vqpci->msix_addr = guest_alloc(alloc, 4); |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 328 | qpci_io_writel(d->pdev, d->pdev->msix_table_bar, |
| 329 | off + PCI_MSIX_ENTRY_LOWER_ADDR, vqpci->msix_addr & ~0UL); |
| 330 | qpci_io_writel(d->pdev, d->pdev->msix_table_bar, |
| 331 | off + PCI_MSIX_ENTRY_UPPER_ADDR, |
| 332 | (vqpci->msix_addr >> 32) & ~0UL); |
| 333 | qpci_io_writel(d->pdev, d->pdev->msix_table_bar, |
| 334 | off + PCI_MSIX_ENTRY_DATA, vqpci->msix_data); |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 335 | |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 336 | control = qpci_io_readl(d->pdev, d->pdev->msix_table_bar, |
| 337 | off + PCI_MSIX_ENTRY_VECTOR_CTRL); |
| 338 | qpci_io_writel(d->pdev, d->pdev->msix_table_bar, |
| 339 | off + PCI_MSIX_ENTRY_VECTOR_CTRL, |
| 340 | control & ~PCI_MSIX_ENTRY_CTRL_MASKBIT); |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 341 | |
Stefan Hajnoczi | 957d8d1 | 2019-10-23 11:04:21 +0100 | [diff] [blame] | 342 | d->msix_ops->set_queue_vector(d, vqpci->vq.index, entry); |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 343 | } |
| 344 | |
| 345 | void qvirtio_pci_set_msix_configuration_vector(QVirtioPCIDevice *d, |
| 346 | QGuestAllocator *alloc, uint16_t entry) |
| 347 | { |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 348 | uint32_t control; |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 349 | uint64_t off; |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 350 | |
| 351 | g_assert(d->pdev->msix_enabled); |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 352 | off = d->pdev->msix_table_off + (entry * 16); |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 353 | |
| 354 | g_assert_cmpint(entry, >=, 0); |
| 355 | g_assert_cmpint(entry, <, qpci_msix_table_size(d->pdev)); |
| 356 | d->config_msix_entry = entry; |
| 357 | |
| 358 | d->config_msix_data = 0x12345678; |
| 359 | d->config_msix_addr = guest_alloc(alloc, 4); |
| 360 | |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 361 | qpci_io_writel(d->pdev, d->pdev->msix_table_bar, |
| 362 | off + PCI_MSIX_ENTRY_LOWER_ADDR, d->config_msix_addr & ~0UL); |
| 363 | qpci_io_writel(d->pdev, d->pdev->msix_table_bar, |
| 364 | off + PCI_MSIX_ENTRY_UPPER_ADDR, |
| 365 | (d->config_msix_addr >> 32) & ~0UL); |
| 366 | qpci_io_writel(d->pdev, d->pdev->msix_table_bar, |
| 367 | off + PCI_MSIX_ENTRY_DATA, d->config_msix_data); |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 368 | |
David Gibson | b4ba67d | 2016-10-24 15:52:06 +1100 | [diff] [blame] | 369 | control = qpci_io_readl(d->pdev, d->pdev->msix_table_bar, |
| 370 | off + PCI_MSIX_ENTRY_VECTOR_CTRL); |
| 371 | qpci_io_writel(d->pdev, d->pdev->msix_table_bar, |
| 372 | off + PCI_MSIX_ENTRY_VECTOR_CTRL, |
| 373 | control & ~PCI_MSIX_ENTRY_CTRL_MASKBIT); |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 374 | |
Stefan Hajnoczi | 957d8d1 | 2019-10-23 11:04:21 +0100 | [diff] [blame] | 375 | d->msix_ops->set_config_vector(d, entry); |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 376 | } |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 377 | |
| 378 | void qvirtio_pci_destructor(QOSGraphObject *obj) |
| 379 | { |
| 380 | QVirtioPCIDevice *dev = (QVirtioPCIDevice *)obj; |
| 381 | qvirtio_pci_device_disable(dev); |
Paolo Bonzini | 6e68204 | 2018-11-15 12:28:38 +0100 | [diff] [blame] | 382 | g_free(dev->pdev); |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 383 | } |
| 384 | |
| 385 | void qvirtio_pci_start_hw(QOSGraphObject *obj) |
| 386 | { |
| 387 | QVirtioPCIDevice *dev = (QVirtioPCIDevice *)obj; |
| 388 | qvirtio_pci_device_enable(dev); |
| 389 | qvirtio_start_device(&dev->vdev); |
| 390 | } |
| 391 | |
Stefan Hajnoczi | 9598f9e | 2019-10-23 11:04:24 +0100 | [diff] [blame] | 392 | static void qvirtio_pci_init_legacy(QVirtioPCIDevice *dev) |
| 393 | { |
| 394 | dev->vdev.device_type = qpci_config_readw(dev->pdev, PCI_SUBSYSTEM_ID); |
| 395 | dev->bar_idx = 0; |
| 396 | dev->vdev.bus = &qvirtio_pci_legacy; |
| 397 | dev->msix_ops = &qvirtio_pci_msix_ops_legacy; |
| 398 | dev->vdev.big_endian = qtest_big_endian(dev->pdev->bus->qts); |
| 399 | } |
| 400 | |
Paolo Bonzini | 6e68204 | 2018-11-15 12:28:38 +0100 | [diff] [blame] | 401 | static void qvirtio_pci_init_from_pcidev(QVirtioPCIDevice *dev, QPCIDevice *pci_dev) |
| 402 | { |
| 403 | dev->pdev = pci_dev; |
Paolo Bonzini | 6e68204 | 2018-11-15 12:28:38 +0100 | [diff] [blame] | 404 | dev->config_msix_entry = -1; |
| 405 | |
Stefan Hajnoczi | d08f68b | 2019-10-23 11:04:25 +0100 | [diff] [blame] | 406 | if (!qvirtio_pci_init_virtio_1(dev)) { |
| 407 | qvirtio_pci_init_legacy(dev); |
| 408 | } |
Paolo Bonzini | 6e68204 | 2018-11-15 12:28:38 +0100 | [diff] [blame] | 409 | |
| 410 | /* each virtio-xxx-pci device should override at least this function */ |
| 411 | dev->obj.get_driver = NULL; |
| 412 | dev->obj.start_hw = qvirtio_pci_start_hw; |
| 413 | dev->obj.destructor = qvirtio_pci_destructor; |
| 414 | } |
| 415 | |
Emanuele Giuseppe Esposito | 1ce66ec | 2018-07-18 16:02:41 +0200 | [diff] [blame] | 416 | void virtio_pci_init(QVirtioPCIDevice *dev, QPCIBus *bus, QPCIAddress * addr) |
| 417 | { |
| 418 | QPCIDevice *pci_dev = qpci_device_find(bus, addr->devfn); |
| 419 | g_assert_nonnull(pci_dev); |
| 420 | qvirtio_pci_init_from_pcidev(dev, pci_dev); |
| 421 | } |
| 422 | |
| 423 | QVirtioPCIDevice *virtio_pci_new(QPCIBus *bus, QPCIAddress * addr) |
| 424 | { |
| 425 | QVirtioPCIDevice *dev; |
| 426 | QPCIDevice *pci_dev = qpci_device_find(bus, addr->devfn); |
| 427 | if (!pci_dev) { |
| 428 | return NULL; |
| 429 | } |
| 430 | |
| 431 | dev = g_new0(QVirtioPCIDevice, 1); |
| 432 | qvirtio_pci_init_from_pcidev(dev, pci_dev); |
| 433 | dev->obj.free = g_free; |
| 434 | return dev; |
| 435 | } |