Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 1 | /* |
| 2 | * MSI-X device support |
| 3 | * |
| 4 | * This module includes support for MSI-X in pci devices. |
| 5 | * |
| 6 | * Author: Michael S. Tsirkin <mst@redhat.com> |
| 7 | * |
| 8 | * Copyright (c) 2009, Red Hat Inc, Michael S. Tsirkin (mst@redhat.com) |
| 9 | * |
| 10 | * This work is licensed under the terms of the GNU GPL, version 2. See |
| 11 | * the COPYING file in the top-level directory. |
Paolo Bonzini | 6b620ca | 2012-01-13 17:44:23 +0100 | [diff] [blame] | 12 | * |
| 13 | * Contributions after 2012-01-13 are licensed under the terms of the |
| 14 | * GNU GPL, version 2 or (at your option) any later version. |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 15 | */ |
| 16 | |
Peter Maydell | 97d5408 | 2016-01-26 18:17:15 +0000 | [diff] [blame] | 17 | #include "qemu/osdep.h" |
Michael S. Tsirkin | c759b24 | 2012-12-12 23:05:42 +0200 | [diff] [blame] | 18 | #include "hw/hw.h" |
| 19 | #include "hw/pci/msi.h" |
| 20 | #include "hw/pci/msix.h" |
| 21 | #include "hw/pci/pci.h" |
Stefano Stabellini | 428c3ec | 2016-01-13 14:59:09 +0000 | [diff] [blame] | 22 | #include "hw/xen/xen.h" |
Paolo Bonzini | 1de7afc | 2012-12-17 18:20:00 +0100 | [diff] [blame] | 23 | #include "qemu/range.h" |
Cao jin | ee640c6 | 2017-01-17 14:18:48 +0800 | [diff] [blame] | 24 | #include "qapi/error.h" |
Peter Xu | 993b1f4 | 2017-05-09 14:00:43 +0800 | [diff] [blame] | 25 | #include "trace.h" |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 26 | |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 27 | #define MSIX_CAP_LENGTH 12 |
| 28 | |
Michael S. Tsirkin | 2760952 | 2009-11-25 12:18:00 +0200 | [diff] [blame] | 29 | /* MSI enable bit and maskall bit are in byte 1 in FLAGS register */ |
| 30 | #define MSIX_CONTROL_OFFSET (PCI_MSIX_FLAGS + 1) |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 31 | #define MSIX_ENABLE_MASK (PCI_MSIX_FLAGS_ENABLE >> 8) |
Michael S. Tsirkin | 5b5cb08 | 2009-11-25 12:19:32 +0200 | [diff] [blame] | 32 | #define MSIX_MASKALL_MASK (PCI_MSIX_FLAGS_MASKALL >> 8) |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 33 | |
Michael S. Tsirkin | 4c93bfa | 2012-12-21 00:27:02 +0200 | [diff] [blame] | 34 | MSIMessage msix_get_message(PCIDevice *dev, unsigned vector) |
Jan Kiszka | bc4caf4 | 2012-05-17 10:32:29 -0300 | [diff] [blame] | 35 | { |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 36 | uint8_t *table_entry = dev->msix_table + vector * PCI_MSIX_ENTRY_SIZE; |
Jan Kiszka | bc4caf4 | 2012-05-17 10:32:29 -0300 | [diff] [blame] | 37 | MSIMessage msg; |
| 38 | |
| 39 | msg.address = pci_get_quad(table_entry + PCI_MSIX_ENTRY_LOWER_ADDR); |
| 40 | msg.data = pci_get_long(table_entry + PCI_MSIX_ENTRY_DATA); |
| 41 | return msg; |
| 42 | } |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 43 | |
Alexey Kardashevskiy | 932d4a4 | 2012-07-19 10:35:07 +1000 | [diff] [blame] | 44 | /* |
| 45 | * Special API for POWER to configure the vectors through |
| 46 | * a side channel. Should never be used by devices. |
| 47 | */ |
| 48 | void msix_set_message(PCIDevice *dev, int vector, struct MSIMessage msg) |
| 49 | { |
| 50 | uint8_t *table_entry = dev->msix_table + vector * PCI_MSIX_ENTRY_SIZE; |
| 51 | |
| 52 | pci_set_quad(table_entry + PCI_MSIX_ENTRY_LOWER_ADDR, msg.address); |
| 53 | pci_set_long(table_entry + PCI_MSIX_ENTRY_DATA, msg.data); |
| 54 | table_entry[PCI_MSIX_ENTRY_VECTOR_CTRL] &= ~PCI_MSIX_ENTRY_CTRL_MASKBIT; |
| 55 | } |
| 56 | |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 57 | static uint8_t msix_pending_mask(int vector) |
| 58 | { |
| 59 | return 1 << (vector % 8); |
| 60 | } |
| 61 | |
| 62 | static uint8_t *msix_pending_byte(PCIDevice *dev, int vector) |
| 63 | { |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 64 | return dev->msix_pba + vector / 8; |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 65 | } |
| 66 | |
| 67 | static int msix_is_pending(PCIDevice *dev, int vector) |
| 68 | { |
| 69 | return *msix_pending_byte(dev, vector) & msix_pending_mask(vector); |
| 70 | } |
| 71 | |
Michael S. Tsirkin | 70f8ee3 | 2012-12-18 13:54:32 +0200 | [diff] [blame] | 72 | void msix_set_pending(PCIDevice *dev, unsigned int vector) |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 73 | { |
| 74 | *msix_pending_byte(dev, vector) |= msix_pending_mask(vector); |
| 75 | } |
| 76 | |
Dmitry Fleytman | 3bdfaab | 2016-06-01 11:23:31 +0300 | [diff] [blame] | 77 | void msix_clr_pending(PCIDevice *dev, int vector) |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 78 | { |
| 79 | *msix_pending_byte(dev, vector) &= ~msix_pending_mask(vector); |
| 80 | } |
| 81 | |
Michael S. Tsirkin | 70f8ee3 | 2012-12-18 13:54:32 +0200 | [diff] [blame] | 82 | static bool msix_vector_masked(PCIDevice *dev, unsigned int vector, bool fmask) |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 83 | { |
Stefano Stabellini | 428c3ec | 2016-01-13 14:59:09 +0000 | [diff] [blame] | 84 | unsigned offset = vector * PCI_MSIX_ENTRY_SIZE; |
Michael S. Tsirkin | e1e4bf2 | 2016-02-13 20:50:50 +0200 | [diff] [blame] | 85 | uint8_t *data = &dev->msix_table[offset + PCI_MSIX_ENTRY_DATA]; |
Stefano Stabellini | 428c3ec | 2016-01-13 14:59:09 +0000 | [diff] [blame] | 86 | /* MSIs on Xen can be remapped into pirqs. In those cases, masking |
| 87 | * and unmasking go through the PV evtchn path. */ |
Michael S. Tsirkin | e1e4bf2 | 2016-02-13 20:50:50 +0200 | [diff] [blame] | 88 | if (xen_enabled() && xen_is_pirq_msi(pci_get_long(data))) { |
Stefano Stabellini | 428c3ec | 2016-01-13 14:59:09 +0000 | [diff] [blame] | 89 | return false; |
| 90 | } |
| 91 | return fmask || dev->msix_table[offset + PCI_MSIX_ENTRY_VECTOR_CTRL] & |
| 92 | PCI_MSIX_ENTRY_CTRL_MASKBIT; |
Michael S. Tsirkin | 5b5cb08 | 2009-11-25 12:19:32 +0200 | [diff] [blame] | 93 | } |
| 94 | |
Michael S. Tsirkin | 70f8ee3 | 2012-12-18 13:54:32 +0200 | [diff] [blame] | 95 | bool msix_is_masked(PCIDevice *dev, unsigned int vector) |
Michael S. Tsirkin | 5b5cb08 | 2009-11-25 12:19:32 +0200 | [diff] [blame] | 96 | { |
Michael S. Tsirkin | ae392c4 | 2011-11-21 18:57:50 +0200 | [diff] [blame] | 97 | return msix_vector_masked(dev, vector, dev->msix_function_masked); |
| 98 | } |
| 99 | |
Jan Kiszka | 2cdfe53 | 2012-05-17 10:32:31 -0300 | [diff] [blame] | 100 | static void msix_fire_vector_notifier(PCIDevice *dev, |
| 101 | unsigned int vector, bool is_masked) |
| 102 | { |
| 103 | MSIMessage msg; |
| 104 | int ret; |
| 105 | |
| 106 | if (!dev->msix_vector_use_notifier) { |
| 107 | return; |
| 108 | } |
| 109 | if (is_masked) { |
| 110 | dev->msix_vector_release_notifier(dev, vector); |
| 111 | } else { |
| 112 | msg = msix_get_message(dev, vector); |
| 113 | ret = dev->msix_vector_use_notifier(dev, vector, msg); |
| 114 | assert(ret >= 0); |
| 115 | } |
| 116 | } |
| 117 | |
Michael S. Tsirkin | ae392c4 | 2011-11-21 18:57:50 +0200 | [diff] [blame] | 118 | static void msix_handle_mask_update(PCIDevice *dev, int vector, bool was_masked) |
| 119 | { |
| 120 | bool is_masked = msix_is_masked(dev, vector); |
Jan Kiszka | 2cdfe53 | 2012-05-17 10:32:31 -0300 | [diff] [blame] | 121 | |
Michael S. Tsirkin | ae392c4 | 2011-11-21 18:57:50 +0200 | [diff] [blame] | 122 | if (is_masked == was_masked) { |
| 123 | return; |
| 124 | } |
| 125 | |
Jan Kiszka | 2cdfe53 | 2012-05-17 10:32:31 -0300 | [diff] [blame] | 126 | msix_fire_vector_notifier(dev, vector, is_masked); |
| 127 | |
Michael S. Tsirkin | ae392c4 | 2011-11-21 18:57:50 +0200 | [diff] [blame] | 128 | if (!is_masked && msix_is_pending(dev, vector)) { |
Michael S. Tsirkin | 5b5cb08 | 2009-11-25 12:19:32 +0200 | [diff] [blame] | 129 | msix_clr_pending(dev, vector); |
| 130 | msix_notify(dev, vector); |
| 131 | } |
| 132 | } |
| 133 | |
Peter Xu | 993b1f4 | 2017-05-09 14:00:43 +0800 | [diff] [blame] | 134 | static bool msix_masked(PCIDevice *dev) |
| 135 | { |
| 136 | return dev->config[dev->msix_cap + MSIX_CONTROL_OFFSET] & MSIX_MASKALL_MASK; |
| 137 | } |
| 138 | |
Michael S. Tsirkin | 5032224 | 2011-11-21 18:57:21 +0200 | [diff] [blame] | 139 | static void msix_update_function_masked(PCIDevice *dev) |
| 140 | { |
Peter Xu | 993b1f4 | 2017-05-09 14:00:43 +0800 | [diff] [blame] | 141 | dev->msix_function_masked = !msix_enabled(dev) || msix_masked(dev); |
Michael S. Tsirkin | 5032224 | 2011-11-21 18:57:21 +0200 | [diff] [blame] | 142 | } |
| 143 | |
Michael S. Tsirkin | 5b5cb08 | 2009-11-25 12:19:32 +0200 | [diff] [blame] | 144 | /* Handle MSI-X capability config write. */ |
| 145 | void msix_write_config(PCIDevice *dev, uint32_t addr, |
| 146 | uint32_t val, int len) |
| 147 | { |
| 148 | unsigned enable_pos = dev->msix_cap + MSIX_CONTROL_OFFSET; |
| 149 | int vector; |
Michael S. Tsirkin | 5032224 | 2011-11-21 18:57:21 +0200 | [diff] [blame] | 150 | bool was_masked; |
Michael S. Tsirkin | 5b5cb08 | 2009-11-25 12:19:32 +0200 | [diff] [blame] | 151 | |
Jan Kiszka | 7c9958b | 2012-05-11 11:42:39 -0300 | [diff] [blame] | 152 | if (!msix_present(dev) || !range_covers_byte(addr, len, enable_pos)) { |
Michael S. Tsirkin | 5b5cb08 | 2009-11-25 12:19:32 +0200 | [diff] [blame] | 153 | return; |
| 154 | } |
| 155 | |
Peter Xu | 993b1f4 | 2017-05-09 14:00:43 +0800 | [diff] [blame] | 156 | trace_msix_write_config(dev->name, msix_enabled(dev), msix_masked(dev)); |
| 157 | |
Michael S. Tsirkin | 5032224 | 2011-11-21 18:57:21 +0200 | [diff] [blame] | 158 | was_masked = dev->msix_function_masked; |
| 159 | msix_update_function_masked(dev); |
| 160 | |
Michael S. Tsirkin | 5b5cb08 | 2009-11-25 12:19:32 +0200 | [diff] [blame] | 161 | if (!msix_enabled(dev)) { |
| 162 | return; |
| 163 | } |
| 164 | |
Isaku Yamahata | e407bf1 | 2011-01-20 16:21:40 +0900 | [diff] [blame] | 165 | pci_device_deassert_intx(dev); |
Michael S. Tsirkin | 5b5cb08 | 2009-11-25 12:19:32 +0200 | [diff] [blame] | 166 | |
Michael S. Tsirkin | 5032224 | 2011-11-21 18:57:21 +0200 | [diff] [blame] | 167 | if (dev->msix_function_masked == was_masked) { |
Michael S. Tsirkin | 5b5cb08 | 2009-11-25 12:19:32 +0200 | [diff] [blame] | 168 | return; |
| 169 | } |
| 170 | |
| 171 | for (vector = 0; vector < dev->msix_entries_nr; ++vector) { |
Michael S. Tsirkin | ae392c4 | 2011-11-21 18:57:50 +0200 | [diff] [blame] | 172 | msix_handle_mask_update(dev, vector, |
| 173 | msix_vector_masked(dev, vector, was_masked)); |
Michael S. Tsirkin | 5b5cb08 | 2009-11-25 12:19:32 +0200 | [diff] [blame] | 174 | } |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 175 | } |
| 176 | |
Avi Kivity | a8170e5 | 2012-10-23 12:30:10 +0200 | [diff] [blame] | 177 | static uint64_t msix_table_mmio_read(void *opaque, hwaddr addr, |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 178 | unsigned size) |
Alex Williamson | eebcb0a | 2012-06-14 12:16:19 -0600 | [diff] [blame] | 179 | { |
| 180 | PCIDevice *dev = opaque; |
Alex Williamson | eebcb0a | 2012-06-14 12:16:19 -0600 | [diff] [blame] | 181 | |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 182 | return pci_get_long(dev->msix_table + addr); |
Alex Williamson | eebcb0a | 2012-06-14 12:16:19 -0600 | [diff] [blame] | 183 | } |
| 184 | |
Avi Kivity | a8170e5 | 2012-10-23 12:30:10 +0200 | [diff] [blame] | 185 | static void msix_table_mmio_write(void *opaque, hwaddr addr, |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 186 | uint64_t val, unsigned size) |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 187 | { |
| 188 | PCIDevice *dev = opaque; |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 189 | int vector = addr / PCI_MSIX_ENTRY_SIZE; |
Michael S. Tsirkin | ae392c4 | 2011-11-21 18:57:50 +0200 | [diff] [blame] | 190 | bool was_masked; |
Michael S. Tsirkin | 9a93b61 | 2011-11-21 18:57:31 +0200 | [diff] [blame] | 191 | |
Michael S. Tsirkin | ae392c4 | 2011-11-21 18:57:50 +0200 | [diff] [blame] | 192 | was_masked = msix_is_masked(dev, vector); |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 193 | pci_set_long(dev->msix_table + addr, val); |
Michael S. Tsirkin | ae392c4 | 2011-11-21 18:57:50 +0200 | [diff] [blame] | 194 | msix_handle_mask_update(dev, vector, was_masked); |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 195 | } |
| 196 | |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 197 | static const MemoryRegionOps msix_table_mmio_ops = { |
| 198 | .read = msix_table_mmio_read, |
| 199 | .write = msix_table_mmio_write, |
Alexander Graf | 68d1e1f | 2012-12-06 04:11:33 +0100 | [diff] [blame] | 200 | .endianness = DEVICE_LITTLE_ENDIAN, |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 201 | .valid = { |
| 202 | .min_access_size = 4, |
| 203 | .max_access_size = 4, |
| 204 | }, |
| 205 | }; |
| 206 | |
Avi Kivity | a8170e5 | 2012-10-23 12:30:10 +0200 | [diff] [blame] | 207 | static uint64_t msix_pba_mmio_read(void *opaque, hwaddr addr, |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 208 | unsigned size) |
| 209 | { |
| 210 | PCIDevice *dev = opaque; |
Michael S. Tsirkin | bbef882 | 2012-12-12 16:10:02 +0200 | [diff] [blame] | 211 | if (dev->msix_vector_poll_notifier) { |
| 212 | unsigned vector_start = addr * 8; |
| 213 | unsigned vector_end = MIN(addr + size * 8, dev->msix_entries_nr); |
| 214 | dev->msix_vector_poll_notifier(dev, vector_start, vector_end); |
| 215 | } |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 216 | |
| 217 | return pci_get_long(dev->msix_pba + addr); |
| 218 | } |
| 219 | |
Marc-André Lureau | 43b11a9 | 2015-06-26 14:25:29 +0200 | [diff] [blame] | 220 | static void msix_pba_mmio_write(void *opaque, hwaddr addr, |
| 221 | uint64_t val, unsigned size) |
| 222 | { |
| 223 | } |
| 224 | |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 225 | static const MemoryRegionOps msix_pba_mmio_ops = { |
| 226 | .read = msix_pba_mmio_read, |
Marc-André Lureau | 43b11a9 | 2015-06-26 14:25:29 +0200 | [diff] [blame] | 227 | .write = msix_pba_mmio_write, |
Alexander Graf | 68d1e1f | 2012-12-06 04:11:33 +0100 | [diff] [blame] | 228 | .endianness = DEVICE_LITTLE_ENDIAN, |
Avi Kivity | 95524ae | 2011-08-08 16:09:26 +0300 | [diff] [blame] | 229 | .valid = { |
| 230 | .min_access_size = 4, |
| 231 | .max_access_size = 4, |
| 232 | }, |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 233 | }; |
| 234 | |
Michael S. Tsirkin | ae1be0b | 2009-11-25 11:41:48 +0200 | [diff] [blame] | 235 | static void msix_mask_all(struct PCIDevice *dev, unsigned nentries) |
| 236 | { |
| 237 | int vector; |
Jan Kiszka | 5b5f133 | 2012-05-17 10:32:30 -0300 | [diff] [blame] | 238 | |
Michael S. Tsirkin | ae1be0b | 2009-11-25 11:41:48 +0200 | [diff] [blame] | 239 | for (vector = 0; vector < nentries; ++vector) { |
Jan Kiszka | 01731cf | 2011-06-09 09:39:56 +0200 | [diff] [blame] | 240 | unsigned offset = |
| 241 | vector * PCI_MSIX_ENTRY_SIZE + PCI_MSIX_ENTRY_VECTOR_CTRL; |
Jan Kiszka | 5b5f133 | 2012-05-17 10:32:30 -0300 | [diff] [blame] | 242 | bool was_masked = msix_is_masked(dev, vector); |
| 243 | |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 244 | dev->msix_table[offset] |= PCI_MSIX_ENTRY_CTRL_MASKBIT; |
Jan Kiszka | 5b5f133 | 2012-05-17 10:32:30 -0300 | [diff] [blame] | 245 | msix_handle_mask_update(dev, vector, was_masked); |
Michael S. Tsirkin | ae1be0b | 2009-11-25 11:41:48 +0200 | [diff] [blame] | 246 | } |
| 247 | } |
| 248 | |
Cao jin | ee640c6 | 2017-01-17 14:18:48 +0800 | [diff] [blame] | 249 | /* |
| 250 | * Make PCI device @dev MSI-X capable |
| 251 | * @nentries is the max number of MSI-X vectors that the device support. |
| 252 | * @table_bar is the MemoryRegion that MSI-X table structure resides. |
| 253 | * @table_bar_nr is number of base address register corresponding to @table_bar. |
| 254 | * @table_offset indicates the offset that the MSI-X table structure starts with |
| 255 | * in @table_bar. |
| 256 | * @pba_bar is the MemoryRegion that the Pending Bit Array structure resides. |
| 257 | * @pba_bar_nr is number of base address register corresponding to @pba_bar. |
| 258 | * @pba_offset indicates the offset that the Pending Bit Array structure |
| 259 | * starts with in @pba_bar. |
| 260 | * Non-zero @cap_pos puts capability MSI-X at that offset in PCI config space. |
| 261 | * @errp is for returning errors. |
| 262 | * |
| 263 | * Return 0 on success; set @errp and return -errno on error: |
| 264 | * -ENOTSUP means lacking msi support for a msi-capable platform. |
| 265 | * -EINVAL means capability overlap, happens when @cap_pos is non-zero, |
| 266 | * also means a programming error, except device assignment, which can check |
| 267 | * if a real HW is broken. |
| 268 | */ |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 269 | int msix_init(struct PCIDevice *dev, unsigned short nentries, |
Alex Williamson | 5a2c202 | 2012-06-14 12:16:47 -0600 | [diff] [blame] | 270 | MemoryRegion *table_bar, uint8_t table_bar_nr, |
| 271 | unsigned table_offset, MemoryRegion *pba_bar, |
Cao jin | ee640c6 | 2017-01-17 14:18:48 +0800 | [diff] [blame] | 272 | uint8_t pba_bar_nr, unsigned pba_offset, uint8_t cap_pos, |
| 273 | Error **errp) |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 274 | { |
Alex Williamson | 5a2c202 | 2012-06-14 12:16:47 -0600 | [diff] [blame] | 275 | int cap; |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 276 | unsigned table_size, pba_size; |
Alex Williamson | 5a2c202 | 2012-06-14 12:16:47 -0600 | [diff] [blame] | 277 | uint8_t *config; |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 278 | |
Jan Kiszka | 60ba3cc | 2011-10-15 14:33:17 +0200 | [diff] [blame] | 279 | /* Nothing to do if MSI is not supported by interrupt controller */ |
Michael S. Tsirkin | 226419d | 2016-03-04 11:24:28 +0200 | [diff] [blame] | 280 | if (!msi_nonbroken) { |
Cao jin | ee640c6 | 2017-01-17 14:18:48 +0800 | [diff] [blame] | 281 | error_setg(errp, "MSI-X is not supported by interrupt controller"); |
Jan Kiszka | 60ba3cc | 2011-10-15 14:33:17 +0200 | [diff] [blame] | 282 | return -ENOTSUP; |
| 283 | } |
Alex Williamson | 5a2c202 | 2012-06-14 12:16:47 -0600 | [diff] [blame] | 284 | |
| 285 | if (nentries < 1 || nentries > PCI_MSIX_FLAGS_QSIZE + 1) { |
Cao jin | ee640c6 | 2017-01-17 14:18:48 +0800 | [diff] [blame] | 286 | error_setg(errp, "The number of MSI-X vectors is invalid"); |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 287 | return -EINVAL; |
Alex Williamson | 5a2c202 | 2012-06-14 12:16:47 -0600 | [diff] [blame] | 288 | } |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 289 | |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 290 | table_size = nentries * PCI_MSIX_ENTRY_SIZE; |
| 291 | pba_size = QEMU_ALIGN_UP(nentries, 64) / 8; |
| 292 | |
Alex Williamson | 5a2c202 | 2012-06-14 12:16:47 -0600 | [diff] [blame] | 293 | /* Sanity test: table & pba don't overlap, fit within BARs, min aligned */ |
| 294 | if ((table_bar_nr == pba_bar_nr && |
| 295 | ranges_overlap(table_offset, table_size, pba_offset, pba_size)) || |
| 296 | table_offset + table_size > memory_region_size(table_bar) || |
| 297 | pba_offset + pba_size > memory_region_size(pba_bar) || |
| 298 | (table_offset | pba_offset) & PCI_MSIX_FLAGS_BIRMASK) { |
Cao jin | ee640c6 | 2017-01-17 14:18:48 +0800 | [diff] [blame] | 299 | error_setg(errp, "table & pba overlap, or they don't fit in BARs," |
| 300 | " or don't align"); |
Alex Williamson | 5a2c202 | 2012-06-14 12:16:47 -0600 | [diff] [blame] | 301 | return -EINVAL; |
| 302 | } |
| 303 | |
Mao Zhongyi | 2784127 | 2017-06-27 14:16:51 +0800 | [diff] [blame] | 304 | cap = pci_add_capability(dev, PCI_CAP_ID_MSIX, |
Cao jin | ee640c6 | 2017-01-17 14:18:48 +0800 | [diff] [blame] | 305 | cap_pos, MSIX_CAP_LENGTH, errp); |
Alex Williamson | 5a2c202 | 2012-06-14 12:16:47 -0600 | [diff] [blame] | 306 | if (cap < 0) { |
| 307 | return cap; |
| 308 | } |
| 309 | |
| 310 | dev->msix_cap = cap; |
| 311 | dev->cap_present |= QEMU_PCI_CAP_MSIX; |
| 312 | config = dev->config + cap; |
| 313 | |
| 314 | pci_set_word(config + PCI_MSIX_FLAGS, nentries - 1); |
| 315 | dev->msix_entries_nr = nentries; |
| 316 | dev->msix_function_masked = true; |
| 317 | |
| 318 | pci_set_long(config + PCI_MSIX_TABLE, table_offset | table_bar_nr); |
| 319 | pci_set_long(config + PCI_MSIX_PBA, pba_offset | pba_bar_nr); |
| 320 | |
| 321 | /* Make flags bit writable. */ |
| 322 | dev->wmask[cap + MSIX_CONTROL_OFFSET] |= MSIX_ENABLE_MASK | |
| 323 | MSIX_MASKALL_MASK; |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 324 | |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 325 | dev->msix_table = g_malloc0(table_size); |
| 326 | dev->msix_pba = g_malloc0(pba_size); |
Alex Williamson | 5a2c202 | 2012-06-14 12:16:47 -0600 | [diff] [blame] | 327 | dev->msix_entry_used = g_malloc0(nentries * sizeof *dev->msix_entry_used); |
| 328 | |
Michael S. Tsirkin | ae1be0b | 2009-11-25 11:41:48 +0200 | [diff] [blame] | 329 | msix_mask_all(dev, nentries); |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 330 | |
Paolo Bonzini | 40c5dce | 2013-06-06 21:25:08 -0400 | [diff] [blame] | 331 | memory_region_init_io(&dev->msix_table_mmio, OBJECT(dev), &msix_table_mmio_ops, dev, |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 332 | "msix-table", table_size); |
Alex Williamson | 5a2c202 | 2012-06-14 12:16:47 -0600 | [diff] [blame] | 333 | memory_region_add_subregion(table_bar, table_offset, &dev->msix_table_mmio); |
Paolo Bonzini | 40c5dce | 2013-06-06 21:25:08 -0400 | [diff] [blame] | 334 | memory_region_init_io(&dev->msix_pba_mmio, OBJECT(dev), &msix_pba_mmio_ops, dev, |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 335 | "msix-pba", pba_size); |
Alex Williamson | 5a2c202 | 2012-06-14 12:16:47 -0600 | [diff] [blame] | 336 | memory_region_add_subregion(pba_bar, pba_offset, &dev->msix_pba_mmio); |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 337 | |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 338 | return 0; |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 339 | } |
| 340 | |
Alex Williamson | 53f9492 | 2012-06-14 12:15:51 -0600 | [diff] [blame] | 341 | int msix_init_exclusive_bar(PCIDevice *dev, unsigned short nentries, |
Cao jin | ee640c6 | 2017-01-17 14:18:48 +0800 | [diff] [blame] | 342 | uint8_t bar_nr, Error **errp) |
Alex Williamson | 53f9492 | 2012-06-14 12:15:51 -0600 | [diff] [blame] | 343 | { |
| 344 | int ret; |
| 345 | char *name; |
Jason Wang | a0ccd21 | 2015-04-23 14:21:49 +0800 | [diff] [blame] | 346 | uint32_t bar_size = 4096; |
| 347 | uint32_t bar_pba_offset = bar_size / 2; |
| 348 | uint32_t bar_pba_size = (nentries / 8 + 1) * 8; |
Alex Williamson | 53f9492 | 2012-06-14 12:15:51 -0600 | [diff] [blame] | 349 | |
| 350 | /* |
| 351 | * Migration compatibility dictates that this remains a 4k |
| 352 | * BAR with the vector table in the lower half and PBA in |
Jason Wang | a0ccd21 | 2015-04-23 14:21:49 +0800 | [diff] [blame] | 353 | * the upper half for nentries which is lower or equal to 128. |
| 354 | * No need to care about using more than 65 entries for legacy |
| 355 | * machine types who has at most 64 queues. |
Alex Williamson | 53f9492 | 2012-06-14 12:15:51 -0600 | [diff] [blame] | 356 | */ |
Jason Wang | a0ccd21 | 2015-04-23 14:21:49 +0800 | [diff] [blame] | 357 | if (nentries * PCI_MSIX_ENTRY_SIZE > bar_pba_offset) { |
| 358 | bar_pba_offset = nentries * PCI_MSIX_ENTRY_SIZE; |
| 359 | } |
Alex Williamson | 53f9492 | 2012-06-14 12:15:51 -0600 | [diff] [blame] | 360 | |
Jason Wang | a0ccd21 | 2015-04-23 14:21:49 +0800 | [diff] [blame] | 361 | if (bar_pba_offset + bar_pba_size > 4096) { |
| 362 | bar_size = bar_pba_offset + bar_pba_size; |
| 363 | } |
| 364 | |
Peter Maydell | 9bff5d8 | 2015-07-24 13:33:07 +0100 | [diff] [blame] | 365 | bar_size = pow2ceil(bar_size); |
Alex Williamson | 53f9492 | 2012-06-14 12:15:51 -0600 | [diff] [blame] | 366 | |
Gerd Hoffmann | 5f893b4 | 2012-08-13 13:05:43 +0200 | [diff] [blame] | 367 | name = g_strdup_printf("%s-msix", dev->name); |
Jason Wang | a0ccd21 | 2015-04-23 14:21:49 +0800 | [diff] [blame] | 368 | memory_region_init(&dev->msix_exclusive_bar, OBJECT(dev), name, bar_size); |
Gerd Hoffmann | 5f893b4 | 2012-08-13 13:05:43 +0200 | [diff] [blame] | 369 | g_free(name); |
Alex Williamson | 53f9492 | 2012-06-14 12:15:51 -0600 | [diff] [blame] | 370 | |
| 371 | ret = msix_init(dev, nentries, &dev->msix_exclusive_bar, bar_nr, |
Jason Wang | a0ccd21 | 2015-04-23 14:21:49 +0800 | [diff] [blame] | 372 | 0, &dev->msix_exclusive_bar, |
| 373 | bar_nr, bar_pba_offset, |
Cao jin | ee640c6 | 2017-01-17 14:18:48 +0800 | [diff] [blame] | 374 | 0, errp); |
Alex Williamson | 53f9492 | 2012-06-14 12:15:51 -0600 | [diff] [blame] | 375 | if (ret) { |
Alex Williamson | 53f9492 | 2012-06-14 12:15:51 -0600 | [diff] [blame] | 376 | return ret; |
| 377 | } |
| 378 | |
| 379 | pci_register_bar(dev, bar_nr, PCI_BASE_ADDRESS_SPACE_MEMORY, |
| 380 | &dev->msix_exclusive_bar); |
| 381 | |
| 382 | return 0; |
| 383 | } |
| 384 | |
Michael S. Tsirkin | 98304c8 | 2009-11-25 12:24:14 +0200 | [diff] [blame] | 385 | static void msix_free_irq_entries(PCIDevice *dev) |
| 386 | { |
| 387 | int vector; |
| 388 | |
| 389 | for (vector = 0; vector < dev->msix_entries_nr; ++vector) { |
| 390 | dev->msix_entry_used[vector] = 0; |
| 391 | msix_clr_pending(dev, vector); |
| 392 | } |
| 393 | } |
| 394 | |
Michael S. Tsirkin | 3cac001 | 2012-08-29 19:40:56 +0300 | [diff] [blame] | 395 | static void msix_clear_all_vectors(PCIDevice *dev) |
| 396 | { |
| 397 | int vector; |
| 398 | |
| 399 | for (vector = 0; vector < dev->msix_entries_nr; ++vector) { |
| 400 | msix_clr_pending(dev, vector); |
| 401 | } |
| 402 | } |
| 403 | |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 404 | /* Clean up resources for the device. */ |
Alex Williamson | 572992e | 2012-06-14 12:16:57 -0600 | [diff] [blame] | 405 | void msix_uninit(PCIDevice *dev, MemoryRegion *table_bar, MemoryRegion *pba_bar) |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 406 | { |
Jan Kiszka | 44701ab | 2012-06-04 16:53:48 +0200 | [diff] [blame] | 407 | if (!msix_present(dev)) { |
Alex Williamson | 572992e | 2012-06-14 12:16:57 -0600 | [diff] [blame] | 408 | return; |
Jan Kiszka | 44701ab | 2012-06-04 16:53:48 +0200 | [diff] [blame] | 409 | } |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 410 | pci_del_capability(dev, PCI_CAP_ID_MSIX, MSIX_CAP_LENGTH); |
| 411 | dev->msix_cap = 0; |
| 412 | msix_free_irq_entries(dev); |
| 413 | dev->msix_entries_nr = 0; |
Alex Williamson | 5a2c202 | 2012-06-14 12:16:47 -0600 | [diff] [blame] | 414 | memory_region_del_subregion(pba_bar, &dev->msix_pba_mmio); |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 415 | g_free(dev->msix_pba); |
| 416 | dev->msix_pba = NULL; |
Alex Williamson | 5a2c202 | 2012-06-14 12:16:47 -0600 | [diff] [blame] | 417 | memory_region_del_subregion(table_bar, &dev->msix_table_mmio); |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 418 | g_free(dev->msix_table); |
| 419 | dev->msix_table = NULL; |
Anthony Liguori | 7267c09 | 2011-08-20 22:09:37 -0500 | [diff] [blame] | 420 | g_free(dev->msix_entry_used); |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 421 | dev->msix_entry_used = NULL; |
| 422 | dev->cap_present &= ~QEMU_PCI_CAP_MSIX; |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 423 | } |
| 424 | |
Alex Williamson | 53f9492 | 2012-06-14 12:15:51 -0600 | [diff] [blame] | 425 | void msix_uninit_exclusive_bar(PCIDevice *dev) |
| 426 | { |
| 427 | if (msix_present(dev)) { |
Alex Williamson | 5a2c202 | 2012-06-14 12:16:47 -0600 | [diff] [blame] | 428 | msix_uninit(dev, &dev->msix_exclusive_bar, &dev->msix_exclusive_bar); |
Alex Williamson | 53f9492 | 2012-06-14 12:15:51 -0600 | [diff] [blame] | 429 | } |
| 430 | } |
| 431 | |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 432 | void msix_save(PCIDevice *dev, QEMUFile *f) |
| 433 | { |
Michael S. Tsirkin | 9a3e12c | 2009-07-01 16:28:00 +0300 | [diff] [blame] | 434 | unsigned n = dev->msix_entries_nr; |
| 435 | |
Jan Kiszka | 44701ab | 2012-06-04 16:53:48 +0200 | [diff] [blame] | 436 | if (!msix_present(dev)) { |
Michael S. Tsirkin | 9a3e12c | 2009-07-01 16:28:00 +0300 | [diff] [blame] | 437 | return; |
Michael S. Tsirkin | 72755a7 | 2009-07-05 15:58:52 +0300 | [diff] [blame] | 438 | } |
Michael S. Tsirkin | 9a3e12c | 2009-07-01 16:28:00 +0300 | [diff] [blame] | 439 | |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 440 | qemu_put_buffer(f, dev->msix_table, n * PCI_MSIX_ENTRY_SIZE); |
Marc-André Lureau | 0ef1efc | 2017-06-22 13:04:16 +0200 | [diff] [blame] | 441 | qemu_put_buffer(f, dev->msix_pba, DIV_ROUND_UP(n, 8)); |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 442 | } |
| 443 | |
| 444 | /* Should be called after restoring the config space. */ |
| 445 | void msix_load(PCIDevice *dev, QEMUFile *f) |
| 446 | { |
| 447 | unsigned n = dev->msix_entries_nr; |
Jan Kiszka | 2cdfe53 | 2012-05-17 10:32:31 -0300 | [diff] [blame] | 448 | unsigned int vector; |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 449 | |
Jan Kiszka | 44701ab | 2012-06-04 16:53:48 +0200 | [diff] [blame] | 450 | if (!msix_present(dev)) { |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 451 | return; |
Blue Swirl | 98846d7 | 2009-07-05 08:11:39 +0000 | [diff] [blame] | 452 | } |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 453 | |
Michael S. Tsirkin | 3cac001 | 2012-08-29 19:40:56 +0300 | [diff] [blame] | 454 | msix_clear_all_vectors(dev); |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 455 | qemu_get_buffer(f, dev->msix_table, n * PCI_MSIX_ENTRY_SIZE); |
Marc-André Lureau | 0ef1efc | 2017-06-22 13:04:16 +0200 | [diff] [blame] | 456 | qemu_get_buffer(f, dev->msix_pba, DIV_ROUND_UP(n, 8)); |
Michael S. Tsirkin | 5032224 | 2011-11-21 18:57:21 +0200 | [diff] [blame] | 457 | msix_update_function_masked(dev); |
Jan Kiszka | 2cdfe53 | 2012-05-17 10:32:31 -0300 | [diff] [blame] | 458 | |
| 459 | for (vector = 0; vector < n; vector++) { |
| 460 | msix_handle_mask_update(dev, vector, true); |
| 461 | } |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 462 | } |
| 463 | |
| 464 | /* Does device support MSI-X? */ |
| 465 | int msix_present(PCIDevice *dev) |
| 466 | { |
| 467 | return dev->cap_present & QEMU_PCI_CAP_MSIX; |
| 468 | } |
| 469 | |
| 470 | /* Is MSI-X enabled? */ |
| 471 | int msix_enabled(PCIDevice *dev) |
| 472 | { |
| 473 | return (dev->cap_present & QEMU_PCI_CAP_MSIX) && |
Michael S. Tsirkin | 2760952 | 2009-11-25 12:18:00 +0200 | [diff] [blame] | 474 | (dev->config[dev->msix_cap + MSIX_CONTROL_OFFSET] & |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 475 | MSIX_ENABLE_MASK); |
| 476 | } |
| 477 | |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 478 | /* Send an MSI-X message */ |
| 479 | void msix_notify(PCIDevice *dev, unsigned vector) |
| 480 | { |
Jan Kiszka | bc4caf4 | 2012-05-17 10:32:29 -0300 | [diff] [blame] | 481 | MSIMessage msg; |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 482 | |
Cao jin | 9348243 | 2017-01-17 14:18:46 +0800 | [diff] [blame] | 483 | if (vector >= dev->msix_entries_nr || !dev->msix_entry_used[vector]) { |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 484 | return; |
Cao jin | 9348243 | 2017-01-17 14:18:46 +0800 | [diff] [blame] | 485 | } |
| 486 | |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 487 | if (msix_is_masked(dev, vector)) { |
| 488 | msix_set_pending(dev, vector); |
| 489 | return; |
| 490 | } |
| 491 | |
Jan Kiszka | bc4caf4 | 2012-05-17 10:32:29 -0300 | [diff] [blame] | 492 | msg = msix_get_message(dev, vector); |
| 493 | |
Pavel Fedin | 38d40ff | 2015-05-27 15:59:59 +0300 | [diff] [blame] | 494 | msi_send_message(dev, msg); |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 495 | } |
| 496 | |
| 497 | void msix_reset(PCIDevice *dev) |
| 498 | { |
Jan Kiszka | 44701ab | 2012-06-04 16:53:48 +0200 | [diff] [blame] | 499 | if (!msix_present(dev)) { |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 500 | return; |
Jan Kiszka | 44701ab | 2012-06-04 16:53:48 +0200 | [diff] [blame] | 501 | } |
Michael S. Tsirkin | 3cac001 | 2012-08-29 19:40:56 +0300 | [diff] [blame] | 502 | msix_clear_all_vectors(dev); |
Michael S. Tsirkin | 2760952 | 2009-11-25 12:18:00 +0200 | [diff] [blame] | 503 | dev->config[dev->msix_cap + MSIX_CONTROL_OFFSET] &= |
Paolo Bonzini | 7d37435 | 2018-12-13 23:37:37 +0100 | [diff] [blame] | 504 | ~dev->wmask[dev->msix_cap + MSIX_CONTROL_OFFSET]; |
Alex Williamson | d35e428 | 2012-06-14 12:16:37 -0600 | [diff] [blame] | 505 | memset(dev->msix_table, 0, dev->msix_entries_nr * PCI_MSIX_ENTRY_SIZE); |
| 506 | memset(dev->msix_pba, 0, QEMU_ALIGN_UP(dev->msix_entries_nr, 64) / 8); |
Michael S. Tsirkin | ae1be0b | 2009-11-25 11:41:48 +0200 | [diff] [blame] | 507 | msix_mask_all(dev, dev->msix_entries_nr); |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 508 | } |
| 509 | |
| 510 | /* PCI spec suggests that devices make it possible for software to configure |
| 511 | * less vectors than supported by the device, but does not specify a standard |
| 512 | * mechanism for devices to do so. |
| 513 | * |
| 514 | * We support this by asking devices to declare vectors software is going to |
| 515 | * actually use, and checking this on the notification path. Devices that |
| 516 | * don't want to follow the spec suggestion can declare all vectors as used. */ |
| 517 | |
| 518 | /* Mark vector as used. */ |
| 519 | int msix_vector_use(PCIDevice *dev, unsigned vector) |
| 520 | { |
Cao jin | 9348243 | 2017-01-17 14:18:46 +0800 | [diff] [blame] | 521 | if (vector >= dev->msix_entries_nr) { |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 522 | return -EINVAL; |
Cao jin | 9348243 | 2017-01-17 14:18:46 +0800 | [diff] [blame] | 523 | } |
| 524 | |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 525 | dev->msix_entry_used[vector]++; |
| 526 | return 0; |
| 527 | } |
| 528 | |
| 529 | /* Mark vector as unused. */ |
| 530 | void msix_vector_unuse(PCIDevice *dev, unsigned vector) |
| 531 | { |
Michael S. Tsirkin | 98304c8 | 2009-11-25 12:24:14 +0200 | [diff] [blame] | 532 | if (vector >= dev->msix_entries_nr || !dev->msix_entry_used[vector]) { |
| 533 | return; |
| 534 | } |
| 535 | if (--dev->msix_entry_used[vector]) { |
| 536 | return; |
| 537 | } |
| 538 | msix_clr_pending(dev, vector); |
Michael S. Tsirkin | 02eb84d | 2009-06-21 19:49:54 +0300 | [diff] [blame] | 539 | } |
Michael S. Tsirkin | b5f28bc | 2009-11-24 16:44:15 +0200 | [diff] [blame] | 540 | |
| 541 | void msix_unuse_all_vectors(PCIDevice *dev) |
| 542 | { |
Jan Kiszka | 44701ab | 2012-06-04 16:53:48 +0200 | [diff] [blame] | 543 | if (!msix_present(dev)) { |
Michael S. Tsirkin | b5f28bc | 2009-11-24 16:44:15 +0200 | [diff] [blame] | 544 | return; |
Jan Kiszka | 44701ab | 2012-06-04 16:53:48 +0200 | [diff] [blame] | 545 | } |
Michael S. Tsirkin | b5f28bc | 2009-11-24 16:44:15 +0200 | [diff] [blame] | 546 | msix_free_irq_entries(dev); |
| 547 | } |
Jan Kiszka | 2cdfe53 | 2012-05-17 10:32:31 -0300 | [diff] [blame] | 548 | |
Jan Kiszka | cb697aa | 2012-05-17 10:32:38 -0300 | [diff] [blame] | 549 | unsigned int msix_nr_vectors_allocated(const PCIDevice *dev) |
| 550 | { |
| 551 | return dev->msix_entries_nr; |
| 552 | } |
| 553 | |
Jan Kiszka | 2cdfe53 | 2012-05-17 10:32:31 -0300 | [diff] [blame] | 554 | static int msix_set_notifier_for_vector(PCIDevice *dev, unsigned int vector) |
| 555 | { |
| 556 | MSIMessage msg; |
| 557 | |
| 558 | if (msix_is_masked(dev, vector)) { |
| 559 | return 0; |
| 560 | } |
| 561 | msg = msix_get_message(dev, vector); |
| 562 | return dev->msix_vector_use_notifier(dev, vector, msg); |
| 563 | } |
| 564 | |
| 565 | static void msix_unset_notifier_for_vector(PCIDevice *dev, unsigned int vector) |
| 566 | { |
| 567 | if (msix_is_masked(dev, vector)) { |
| 568 | return; |
| 569 | } |
| 570 | dev->msix_vector_release_notifier(dev, vector); |
| 571 | } |
| 572 | |
| 573 | int msix_set_vector_notifiers(PCIDevice *dev, |
| 574 | MSIVectorUseNotifier use_notifier, |
Michael S. Tsirkin | bbef882 | 2012-12-12 16:10:02 +0200 | [diff] [blame] | 575 | MSIVectorReleaseNotifier release_notifier, |
| 576 | MSIVectorPollNotifier poll_notifier) |
Jan Kiszka | 2cdfe53 | 2012-05-17 10:32:31 -0300 | [diff] [blame] | 577 | { |
| 578 | int vector, ret; |
| 579 | |
| 580 | assert(use_notifier && release_notifier); |
| 581 | |
| 582 | dev->msix_vector_use_notifier = use_notifier; |
| 583 | dev->msix_vector_release_notifier = release_notifier; |
Michael S. Tsirkin | bbef882 | 2012-12-12 16:10:02 +0200 | [diff] [blame] | 584 | dev->msix_vector_poll_notifier = poll_notifier; |
Jan Kiszka | 2cdfe53 | 2012-05-17 10:32:31 -0300 | [diff] [blame] | 585 | |
| 586 | if ((dev->config[dev->msix_cap + MSIX_CONTROL_OFFSET] & |
| 587 | (MSIX_ENABLE_MASK | MSIX_MASKALL_MASK)) == MSIX_ENABLE_MASK) { |
| 588 | for (vector = 0; vector < dev->msix_entries_nr; vector++) { |
| 589 | ret = msix_set_notifier_for_vector(dev, vector); |
| 590 | if (ret < 0) { |
| 591 | goto undo; |
| 592 | } |
| 593 | } |
| 594 | } |
Michael S. Tsirkin | bbef882 | 2012-12-12 16:10:02 +0200 | [diff] [blame] | 595 | if (dev->msix_vector_poll_notifier) { |
| 596 | dev->msix_vector_poll_notifier(dev, 0, dev->msix_entries_nr); |
| 597 | } |
Jan Kiszka | 2cdfe53 | 2012-05-17 10:32:31 -0300 | [diff] [blame] | 598 | return 0; |
| 599 | |
| 600 | undo: |
| 601 | while (--vector >= 0) { |
| 602 | msix_unset_notifier_for_vector(dev, vector); |
| 603 | } |
| 604 | dev->msix_vector_use_notifier = NULL; |
| 605 | dev->msix_vector_release_notifier = NULL; |
| 606 | return ret; |
| 607 | } |
| 608 | |
| 609 | void msix_unset_vector_notifiers(PCIDevice *dev) |
| 610 | { |
| 611 | int vector; |
| 612 | |
| 613 | assert(dev->msix_vector_use_notifier && |
| 614 | dev->msix_vector_release_notifier); |
| 615 | |
| 616 | if ((dev->config[dev->msix_cap + MSIX_CONTROL_OFFSET] & |
| 617 | (MSIX_ENABLE_MASK | MSIX_MASKALL_MASK)) == MSIX_ENABLE_MASK) { |
| 618 | for (vector = 0; vector < dev->msix_entries_nr; vector++) { |
| 619 | msix_unset_notifier_for_vector(dev, vector); |
| 620 | } |
| 621 | } |
| 622 | dev->msix_vector_use_notifier = NULL; |
| 623 | dev->msix_vector_release_notifier = NULL; |
Michael S. Tsirkin | bbef882 | 2012-12-12 16:10:02 +0200 | [diff] [blame] | 624 | dev->msix_vector_poll_notifier = NULL; |
Jan Kiszka | 2cdfe53 | 2012-05-17 10:32:31 -0300 | [diff] [blame] | 625 | } |
Gerd Hoffmann | 340b50c | 2013-05-07 15:16:58 +0200 | [diff] [blame] | 626 | |
Jianjun Duan | 2c21ee7 | 2017-01-19 11:00:50 -0800 | [diff] [blame] | 627 | static int put_msix_state(QEMUFile *f, void *pv, size_t size, |
Marc-André Lureau | 03fee66 | 2018-11-14 17:29:30 +0400 | [diff] [blame] | 628 | const VMStateField *field, QJSON *vmdesc) |
Gerd Hoffmann | 340b50c | 2013-05-07 15:16:58 +0200 | [diff] [blame] | 629 | { |
| 630 | msix_save(pv, f); |
Jianjun Duan | 2c21ee7 | 2017-01-19 11:00:50 -0800 | [diff] [blame] | 631 | |
| 632 | return 0; |
Gerd Hoffmann | 340b50c | 2013-05-07 15:16:58 +0200 | [diff] [blame] | 633 | } |
| 634 | |
Jianjun Duan | 2c21ee7 | 2017-01-19 11:00:50 -0800 | [diff] [blame] | 635 | static int get_msix_state(QEMUFile *f, void *pv, size_t size, |
Marc-André Lureau | 03fee66 | 2018-11-14 17:29:30 +0400 | [diff] [blame] | 636 | const VMStateField *field) |
Gerd Hoffmann | 340b50c | 2013-05-07 15:16:58 +0200 | [diff] [blame] | 637 | { |
| 638 | msix_load(pv, f); |
| 639 | return 0; |
| 640 | } |
| 641 | |
| 642 | static VMStateInfo vmstate_info_msix = { |
| 643 | .name = "msix state", |
| 644 | .get = get_msix_state, |
| 645 | .put = put_msix_state, |
| 646 | }; |
| 647 | |
| 648 | const VMStateDescription vmstate_msix = { |
| 649 | .name = "msix", |
| 650 | .fields = (VMStateField[]) { |
| 651 | { |
| 652 | .name = "msix", |
| 653 | .version_id = 0, |
| 654 | .field_exists = NULL, |
| 655 | .size = 0, /* ouch */ |
| 656 | .info = &vmstate_info_msix, |
| 657 | .flags = VMS_SINGLE, |
| 658 | .offset = 0, |
| 659 | }, |
| 660 | VMSTATE_END_OF_LIST() |
| 661 | } |
| 662 | }; |