Marc Marí | 311e666 | 2014-09-01 12:07:54 +0200 | [diff] [blame] | 1 | /* |
| 2 | * libqos virtio definitions |
| 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 | |
| 10 | #ifndef LIBQOS_VIRTIO_H |
| 11 | #define LIBQOS_VIRTIO_H |
| 12 | |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 13 | #include "libqos/malloc.h" |
| 14 | |
Marc Marí | 311e666 | 2014-09-01 12:07:54 +0200 | [diff] [blame] | 15 | #define QVIRTIO_VENDOR_ID 0x1AF4 |
| 16 | |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 17 | #define QVIRTIO_RESET 0x0 |
| 18 | #define QVIRTIO_ACKNOWLEDGE 0x1 |
| 19 | #define QVIRTIO_DRIVER 0x2 |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 20 | #define QVIRTIO_DRIVER_OK 0x4 |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 21 | |
Marc Marí | 311e666 | 2014-09-01 12:07:54 +0200 | [diff] [blame] | 22 | #define QVIRTIO_NET_DEVICE_ID 0x1 |
| 23 | #define QVIRTIO_BLK_DEVICE_ID 0x2 |
| 24 | |
Marc Marí | f294b02 | 2014-09-01 12:07:57 +0200 | [diff] [blame] | 25 | #define QVIRTIO_F_NOTIFY_ON_EMPTY 0x01000000 |
| 26 | #define QVIRTIO_F_ANY_LAYOUT 0x08000000 |
| 27 | #define QVIRTIO_F_RING_INDIRECT_DESC 0x10000000 |
| 28 | #define QVIRTIO_F_RING_EVENT_IDX 0x20000000 |
Marc Marí | 1053587 | 2014-09-01 12:08:00 +0200 | [diff] [blame] | 29 | #define QVIRTIO_F_BAD_FEATURE 0x40000000 |
Marc Marí | f294b02 | 2014-09-01 12:07:57 +0200 | [diff] [blame] | 30 | |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 31 | #define QVRING_DESC_F_NEXT 0x1 |
| 32 | #define QVRING_DESC_F_WRITE 0x2 |
| 33 | #define QVRING_DESC_F_INDIRECT 0x4 |
| 34 | |
| 35 | #define QVIRTIO_F_NOTIFY_ON_EMPTY 0x01000000 |
| 36 | #define QVIRTIO_F_ANY_LAYOUT 0x08000000 |
| 37 | #define QVIRTIO_F_RING_INDIRECT_DESC 0x10000000 |
| 38 | #define QVIRTIO_F_RING_EVENT_IDX 0x20000000 |
| 39 | #define QVIRTIO_F_BAD_FEATURE 0x40000000 |
| 40 | |
| 41 | #define QVRING_AVAIL_F_NO_INTERRUPT 1 |
| 42 | |
| 43 | #define QVRING_USED_F_NO_NOTIFY 1 |
| 44 | |
Marc Marí | 311e666 | 2014-09-01 12:07:54 +0200 | [diff] [blame] | 45 | typedef struct QVirtioDevice { |
| 46 | /* Device type */ |
| 47 | uint16_t device_type; |
| 48 | } QVirtioDevice; |
| 49 | |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 50 | typedef struct QVRingDesc { |
| 51 | uint64_t addr; |
| 52 | uint32_t len; |
| 53 | uint16_t flags; |
| 54 | uint16_t next; |
| 55 | } QVRingDesc; |
| 56 | |
| 57 | typedef struct QVRingAvail { |
| 58 | uint16_t flags; |
| 59 | uint16_t idx; |
| 60 | uint16_t ring[0]; /* This is an array of uint16_t */ |
Marc Marí | 1053587 | 2014-09-01 12:08:00 +0200 | [diff] [blame] | 61 | uint16_t used_event; |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 62 | } QVRingAvail; |
| 63 | |
| 64 | typedef struct QVRingUsedElem { |
| 65 | uint32_t id; |
| 66 | uint32_t len; |
| 67 | } QVRingUsedElem; |
| 68 | |
| 69 | typedef struct QVRingUsed { |
| 70 | uint16_t flags; |
| 71 | uint16_t idx; |
| 72 | QVRingUsedElem ring[0]; /* This is an array of QVRingUsedElem structs */ |
Marc Marí | 1053587 | 2014-09-01 12:08:00 +0200 | [diff] [blame] | 73 | uint16_t avail_event; |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 74 | } QVRingUsed; |
| 75 | |
| 76 | typedef struct QVirtQueue { |
| 77 | uint64_t desc; /* This points to an array of QVRingDesc */ |
| 78 | uint64_t avail; /* This points to a QVRingAvail */ |
| 79 | uint64_t used; /* This points to a QVRingDesc */ |
| 80 | uint16_t index; |
| 81 | uint32_t size; |
| 82 | uint32_t free_head; |
| 83 | uint32_t num_free; |
| 84 | uint32_t align; |
Marc Marí | f294b02 | 2014-09-01 12:07:57 +0200 | [diff] [blame] | 85 | bool indirect; |
Marc Marí | 1053587 | 2014-09-01 12:08:00 +0200 | [diff] [blame] | 86 | bool event; |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 87 | } QVirtQueue; |
| 88 | |
Marc Marí | f294b02 | 2014-09-01 12:07:57 +0200 | [diff] [blame] | 89 | typedef struct QVRingIndirectDesc { |
| 90 | uint64_t desc; /* This points to an array fo QVRingDesc */ |
| 91 | uint16_t index; |
| 92 | uint16_t elem; |
| 93 | } QVRingIndirectDesc; |
| 94 | |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 95 | typedef struct QVirtioBus { |
Marc Marí | 728312b | 2015-02-24 22:21:51 +0100 | [diff] [blame^] | 96 | uint8_t (*config_readb)(QVirtioDevice *d, uint64_t addr); |
| 97 | uint16_t (*config_readw)(QVirtioDevice *d, uint64_t addr); |
| 98 | uint32_t (*config_readl)(QVirtioDevice *d, uint64_t addr); |
| 99 | uint64_t (*config_readq)(QVirtioDevice *d, uint64_t addr); |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 100 | |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 101 | /* Get features of the device */ |
| 102 | uint32_t (*get_features)(QVirtioDevice *d); |
| 103 | |
Marc Marí | f294b02 | 2014-09-01 12:07:57 +0200 | [diff] [blame] | 104 | /* Set features of the device */ |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 105 | void (*set_features)(QVirtioDevice *d, uint32_t features); |
| 106 | |
Marc Marí | f294b02 | 2014-09-01 12:07:57 +0200 | [diff] [blame] | 107 | /* Get features of the guest */ |
| 108 | uint32_t (*get_guest_features)(QVirtioDevice *d); |
| 109 | |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 110 | /* Get status of the device */ |
| 111 | uint8_t (*get_status)(QVirtioDevice *d); |
| 112 | |
| 113 | /* Set status of the device */ |
| 114 | void (*set_status)(QVirtioDevice *d, uint8_t status); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 115 | |
Marc Marí | 5836811 | 2014-09-01 12:07:59 +0200 | [diff] [blame] | 116 | /* Get the queue ISR status of the device */ |
| 117 | bool (*get_queue_isr_status)(QVirtioDevice *d, QVirtQueue *vq); |
| 118 | |
| 119 | /* Get the configuration ISR status of the device */ |
| 120 | bool (*get_config_isr_status)(QVirtioDevice *d); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 121 | |
| 122 | /* Select a queue to work on */ |
| 123 | void (*queue_select)(QVirtioDevice *d, uint16_t index); |
| 124 | |
| 125 | /* Get the size of the selected queue */ |
| 126 | uint16_t (*get_queue_size)(QVirtioDevice *d); |
| 127 | |
| 128 | /* Set the address of the selected queue */ |
| 129 | void (*set_queue_address)(QVirtioDevice *d, uint32_t pfn); |
| 130 | |
| 131 | /* Setup the virtqueue specified by index */ |
| 132 | QVirtQueue *(*virtqueue_setup)(QVirtioDevice *d, QGuestAllocator *alloc, |
| 133 | uint16_t index); |
| 134 | |
| 135 | /* Notify changes in virtqueue */ |
| 136 | void (*virtqueue_kick)(QVirtioDevice *d, QVirtQueue *vq); |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 137 | } QVirtioBus; |
| 138 | |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 139 | static inline uint32_t qvring_size(uint32_t num, uint32_t align) |
| 140 | { |
| 141 | return ((sizeof(struct QVRingDesc) * num + sizeof(uint16_t) * (3 + num) |
| 142 | + align - 1) & ~(align - 1)) |
| 143 | + sizeof(uint16_t) * 3 + sizeof(struct QVRingUsedElem) * num; |
| 144 | } |
| 145 | |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 146 | uint8_t qvirtio_config_readb(const QVirtioBus *bus, QVirtioDevice *d, |
Marc Marí | 728312b | 2015-02-24 22:21:51 +0100 | [diff] [blame^] | 147 | uint64_t addr); |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 148 | uint16_t qvirtio_config_readw(const QVirtioBus *bus, QVirtioDevice *d, |
Marc Marí | 728312b | 2015-02-24 22:21:51 +0100 | [diff] [blame^] | 149 | uint64_t addr); |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 150 | uint32_t qvirtio_config_readl(const QVirtioBus *bus, QVirtioDevice *d, |
Marc Marí | 728312b | 2015-02-24 22:21:51 +0100 | [diff] [blame^] | 151 | uint64_t addr); |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 152 | uint64_t qvirtio_config_readq(const QVirtioBus *bus, QVirtioDevice *d, |
Marc Marí | 728312b | 2015-02-24 22:21:51 +0100 | [diff] [blame^] | 153 | uint64_t addr); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 154 | uint32_t qvirtio_get_features(const QVirtioBus *bus, QVirtioDevice *d); |
| 155 | void qvirtio_set_features(const QVirtioBus *bus, QVirtioDevice *d, |
| 156 | uint32_t features); |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 157 | |
| 158 | void qvirtio_reset(const QVirtioBus *bus, QVirtioDevice *d); |
| 159 | void qvirtio_set_acknowledge(const QVirtioBus *bus, QVirtioDevice *d); |
| 160 | void qvirtio_set_driver(const QVirtioBus *bus, QVirtioDevice *d); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 161 | void qvirtio_set_driver_ok(const QVirtioBus *bus, QVirtioDevice *d); |
| 162 | |
Stefan Hajnoczi | 7055626 | 2014-09-29 16:40:12 +0100 | [diff] [blame] | 163 | void qvirtio_wait_queue_isr(const QVirtioBus *bus, QVirtioDevice *d, |
| 164 | QVirtQueue *vq, gint64 timeout_us); |
Stefan Hajnoczi | e8c81b4 | 2014-09-29 16:40:11 +0100 | [diff] [blame] | 165 | uint8_t qvirtio_wait_status_byte_no_isr(const QVirtioBus *bus, |
| 166 | QVirtioDevice *d, |
| 167 | QVirtQueue *vq, |
| 168 | uint64_t addr, |
| 169 | gint64 timeout_us); |
Stefan Hajnoczi | 7055626 | 2014-09-29 16:40:12 +0100 | [diff] [blame] | 170 | void qvirtio_wait_config_isr(const QVirtioBus *bus, QVirtioDevice *d, |
| 171 | gint64 timeout_us); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 172 | QVirtQueue *qvirtqueue_setup(const QVirtioBus *bus, QVirtioDevice *d, |
| 173 | QGuestAllocator *alloc, uint16_t index); |
| 174 | |
| 175 | void qvring_init(const QGuestAllocator *alloc, QVirtQueue *vq, uint64_t addr); |
Marc Marí | f294b02 | 2014-09-01 12:07:57 +0200 | [diff] [blame] | 176 | QVRingIndirectDesc *qvring_indirect_desc_setup(QVirtioDevice *d, |
| 177 | QGuestAllocator *alloc, uint16_t elem); |
| 178 | void qvring_indirect_desc_add(QVRingIndirectDesc *indirect, uint64_t data, |
| 179 | uint32_t len, bool write); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 180 | uint32_t qvirtqueue_add(QVirtQueue *vq, uint64_t data, uint32_t len, bool write, |
| 181 | bool next); |
Marc Marí | f294b02 | 2014-09-01 12:07:57 +0200 | [diff] [blame] | 182 | uint32_t qvirtqueue_add_indirect(QVirtQueue *vq, QVRingIndirectDesc *indirect); |
Marc Marí | bf3c63d | 2014-09-01 12:07:56 +0200 | [diff] [blame] | 183 | void qvirtqueue_kick(const QVirtioBus *bus, QVirtioDevice *d, QVirtQueue *vq, |
| 184 | uint32_t free_head); |
Marc Marí | 46e0cf7 | 2014-09-01 12:07:55 +0200 | [diff] [blame] | 185 | |
Marc Marí | 1053587 | 2014-09-01 12:08:00 +0200 | [diff] [blame] | 186 | void qvirtqueue_set_used_event(QVirtQueue *vq, uint16_t idx); |
Marc Marí | 311e666 | 2014-09-01 12:07:54 +0200 | [diff] [blame] | 187 | #endif |