]> Git Repo - qemu.git/blame - tests/libqos/virtio-mmio.c
Merge remote-tracking branch 'remotes/vivier/tags/q800-branch-pull-request' into...
[qemu.git] / tests / libqos / virtio-mmio.c
CommitLineData
0a6ed700
MM
1/*
2 * libqos virtio MMIO 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
681c28a3 10#include "qemu/osdep.h"
0a6ed700 11#include "libqtest.h"
0b8fa32f 12#include "qemu/module.h"
0a6ed700
MM
13#include "libqos/virtio.h"
14#include "libqos/virtio-mmio.h"
15#include "libqos/malloc.h"
57ed038a 16#include "libqos/qgraph.h"
ee3b850a 17#include "standard-headers/linux/virtio_ring.h"
0a6ed700 18
246fc0fb 19static uint8_t qvirtio_mmio_config_readb(QVirtioDevice *d, uint64_t off)
0a6ed700 20{
57ed038a
EGE
21 QVirtioMMIODevice *dev = container_of(d, QVirtioMMIODevice, vdev);
22 return qtest_readb(dev->qts, dev->addr + QVIRTIO_MMIO_DEVICE_SPECIFIC + off);
0a6ed700
MM
23}
24
246fc0fb 25static uint16_t qvirtio_mmio_config_readw(QVirtioDevice *d, uint64_t off)
0a6ed700 26{
57ed038a
EGE
27 QVirtioMMIODevice *dev = container_of(d, QVirtioMMIODevice, vdev);
28 return qtest_readw(dev->qts, dev->addr + QVIRTIO_MMIO_DEVICE_SPECIFIC + off);
0a6ed700
MM
29}
30
246fc0fb 31static uint32_t qvirtio_mmio_config_readl(QVirtioDevice *d, uint64_t off)
0a6ed700 32{
57ed038a
EGE
33 QVirtioMMIODevice *dev = container_of(d, QVirtioMMIODevice, vdev);
34 return qtest_readl(dev->qts, dev->addr + QVIRTIO_MMIO_DEVICE_SPECIFIC + off);
0a6ed700
MM
35}
36
246fc0fb 37static uint64_t qvirtio_mmio_config_readq(QVirtioDevice *d, uint64_t off)
0a6ed700 38{
57ed038a
EGE
39 QVirtioMMIODevice *dev = container_of(d, QVirtioMMIODevice, vdev);
40 return qtest_readq(dev->qts, dev->addr + QVIRTIO_MMIO_DEVICE_SPECIFIC + off);
0a6ed700
MM
41}
42
a9340358 43static uint64_t qvirtio_mmio_get_features(QVirtioDevice *d)
0a6ed700 44{
57ed038a 45 QVirtioMMIODevice *dev = container_of(d, QVirtioMMIODevice, vdev);
a9340358
SH
46 uint64_t lo;
47 uint64_t hi = 0;
48
57ed038a 49 qtest_writel(dev->qts, dev->addr + QVIRTIO_MMIO_HOST_FEATURES_SEL, 0);
a9340358
SH
50 lo = qtest_readl(dev->qts, dev->addr + QVIRTIO_MMIO_HOST_FEATURES);
51
52 if (dev->version >= 2) {
53 qtest_writel(dev->qts, dev->addr + QVIRTIO_MMIO_HOST_FEATURES_SEL, 1);
54 hi = qtest_readl(dev->qts, dev->addr + QVIRTIO_MMIO_HOST_FEATURES);
55 }
56
57 return (hi << 32) | lo;
0a6ed700
MM
58}
59
a9340358 60static void qvirtio_mmio_set_features(QVirtioDevice *d, uint64_t features)
0a6ed700 61{
57ed038a 62 QVirtioMMIODevice *dev = container_of(d, QVirtioMMIODevice, vdev);
0a6ed700 63 dev->features = features;
57ed038a
EGE
64 qtest_writel(dev->qts, dev->addr + QVIRTIO_MMIO_GUEST_FEATURES_SEL, 0);
65 qtest_writel(dev->qts, dev->addr + QVIRTIO_MMIO_GUEST_FEATURES, features);
a9340358
SH
66
67 if (dev->version >= 2) {
68 qtest_writel(dev->qts, dev->addr + QVIRTIO_MMIO_GUEST_FEATURES_SEL, 1);
69 qtest_writel(dev->qts, dev->addr + QVIRTIO_MMIO_GUEST_FEATURES,
70 features >> 32);
71 }
0a6ed700
MM
72}
73
a9340358 74static uint64_t qvirtio_mmio_get_guest_features(QVirtioDevice *d)
0a6ed700 75{
57ed038a 76 QVirtioMMIODevice *dev = container_of(d, QVirtioMMIODevice, vdev);
0a6ed700
MM
77 return dev->features;
78}
79
80static uint8_t qvirtio_mmio_get_status(QVirtioDevice *d)
81{
57ed038a
EGE
82 QVirtioMMIODevice *dev = container_of(d, QVirtioMMIODevice, vdev);
83 return (uint8_t)qtest_readl(dev->qts, dev->addr + QVIRTIO_MMIO_DEVICE_STATUS);
0a6ed700
MM
84}
85
86static void qvirtio_mmio_set_status(QVirtioDevice *d, uint8_t status)
87{
57ed038a
EGE
88 QVirtioMMIODevice *dev = container_of(d, QVirtioMMIODevice, vdev);
89 qtest_writel(dev->qts, dev->addr + QVIRTIO_MMIO_DEVICE_STATUS, (uint32_t)status);
0a6ed700
MM
90}
91
92static bool qvirtio_mmio_get_queue_isr_status(QVirtioDevice *d, QVirtQueue *vq)
93{
57ed038a 94 QVirtioMMIODevice *dev = container_of(d, QVirtioMMIODevice, vdev);
0a6ed700
MM
95 uint32_t isr;
96
57ed038a 97 isr = qtest_readl(dev->qts, dev->addr + QVIRTIO_MMIO_INTERRUPT_STATUS) & 1;
0a6ed700 98 if (isr != 0) {
57ed038a 99 qtest_writel(dev->qts, dev->addr + QVIRTIO_MMIO_INTERRUPT_ACK, 1);
0a6ed700
MM
100 return true;
101 }
102
103 return false;
104}
105
106static bool qvirtio_mmio_get_config_isr_status(QVirtioDevice *d)
107{
57ed038a 108 QVirtioMMIODevice *dev = container_of(d, QVirtioMMIODevice, vdev);
0a6ed700
MM
109 uint32_t isr;
110
57ed038a 111 isr = qtest_readl(dev->qts, dev->addr + QVIRTIO_MMIO_INTERRUPT_STATUS) & 2;
0a6ed700 112 if (isr != 0) {
57ed038a 113 qtest_writel(dev->qts, dev->addr + QVIRTIO_MMIO_INTERRUPT_ACK, 2);
0a6ed700
MM
114 return true;
115 }
116
117 return false;
118}
119
b57ebd57
TH
120static void qvirtio_mmio_wait_config_isr_status(QVirtioDevice *d,
121 gint64 timeout_us)
122{
123 QVirtioMMIODevice *dev = container_of(d, QVirtioMMIODevice, vdev);
124 gint64 start_time = g_get_monotonic_time();
125
126 do {
127 g_assert(g_get_monotonic_time() - start_time <= timeout_us);
128 qtest_clock_step(dev->qts, 100);
129 } while (!qvirtio_mmio_get_config_isr_status(d));
130}
131
0a6ed700
MM
132static void qvirtio_mmio_queue_select(QVirtioDevice *d, uint16_t index)
133{
57ed038a
EGE
134 QVirtioMMIODevice *dev = container_of(d, QVirtioMMIODevice, vdev);
135 qtest_writel(dev->qts, dev->addr + QVIRTIO_MMIO_QUEUE_SEL, (uint32_t)index);
0a6ed700 136
57ed038a 137 g_assert_cmphex(qtest_readl(dev->qts, dev->addr + QVIRTIO_MMIO_QUEUE_PFN), ==, 0);
0a6ed700
MM
138}
139
140static uint16_t qvirtio_mmio_get_queue_size(QVirtioDevice *d)
141{
57ed038a
EGE
142 QVirtioMMIODevice *dev = container_of(d, QVirtioMMIODevice, vdev);
143 return (uint16_t)qtest_readl(dev->qts, dev->addr + QVIRTIO_MMIO_QUEUE_NUM_MAX);
0a6ed700
MM
144}
145
1e59a866 146static void qvirtio_mmio_set_queue_address(QVirtioDevice *d, QVirtQueue *vq)
0a6ed700 147{
57ed038a 148 QVirtioMMIODevice *dev = container_of(d, QVirtioMMIODevice, vdev);
1e59a866
SH
149 uint64_t pfn = vq->desc / dev->page_size;
150
57ed038a 151 qtest_writel(dev->qts, dev->addr + QVIRTIO_MMIO_QUEUE_PFN, pfn);
0a6ed700
MM
152}
153
154static QVirtQueue *qvirtio_mmio_virtqueue_setup(QVirtioDevice *d,
155 QGuestAllocator *alloc, uint16_t index)
156{
57ed038a 157 QVirtioMMIODevice *dev = container_of(d, QVirtioMMIODevice, vdev);
0a6ed700
MM
158 QVirtQueue *vq;
159 uint64_t addr;
160
161 vq = g_malloc0(sizeof(*vq));
bccd82b4 162 vq->vdev = d;
0a6ed700 163 qvirtio_mmio_queue_select(d, index);
57ed038a 164 qtest_writel(dev->qts, dev->addr + QVIRTIO_MMIO_QUEUE_ALIGN, dev->page_size);
0a6ed700
MM
165
166 vq->index = index;
167 vq->size = qvirtio_mmio_get_queue_size(d);
168 vq->free_head = 0;
169 vq->num_free = vq->size;
170 vq->align = dev->page_size;
a9340358
SH
171 vq->indirect = dev->features & (1ull << VIRTIO_RING_F_INDIRECT_DESC);
172 vq->event = dev->features & (1ull << VIRTIO_RING_F_EVENT_IDX);
0a6ed700 173
57ed038a 174 qtest_writel(dev->qts, dev->addr + QVIRTIO_MMIO_QUEUE_NUM, vq->size);
0a6ed700
MM
175
176 /* Check different than 0 */
177 g_assert_cmpint(vq->size, !=, 0);
178
179 /* Check power of 2 */
180 g_assert_cmpint(vq->size & (vq->size - 1), ==, 0);
181
182 addr = guest_alloc(alloc, qvring_size(vq->size, dev->page_size));
8b898f59 183 qvring_init(dev->qts, alloc, vq, addr);
1e59a866 184 qvirtio_mmio_set_queue_address(d, vq);
0a6ed700
MM
185
186 return vq;
187}
188
f1d3b991
SH
189static void qvirtio_mmio_virtqueue_cleanup(QVirtQueue *vq,
190 QGuestAllocator *alloc)
191{
192 guest_free(alloc, vq->desc);
193 g_free(vq);
194}
195
0a6ed700
MM
196static void qvirtio_mmio_virtqueue_kick(QVirtioDevice *d, QVirtQueue *vq)
197{
57ed038a
EGE
198 QVirtioMMIODevice *dev = container_of(d, QVirtioMMIODevice, vdev);
199 qtest_writel(dev->qts, dev->addr + QVIRTIO_MMIO_QUEUE_NOTIFY, vq->index);
0a6ed700
MM
200}
201
202const QVirtioBus qvirtio_mmio = {
203 .config_readb = qvirtio_mmio_config_readb,
204 .config_readw = qvirtio_mmio_config_readw,
205 .config_readl = qvirtio_mmio_config_readl,
206 .config_readq = qvirtio_mmio_config_readq,
207 .get_features = qvirtio_mmio_get_features,
208 .set_features = qvirtio_mmio_set_features,
209 .get_guest_features = qvirtio_mmio_get_guest_features,
210 .get_status = qvirtio_mmio_get_status,
211 .set_status = qvirtio_mmio_set_status,
212 .get_queue_isr_status = qvirtio_mmio_get_queue_isr_status,
b57ebd57 213 .wait_config_isr_status = qvirtio_mmio_wait_config_isr_status,
0a6ed700
MM
214 .queue_select = qvirtio_mmio_queue_select,
215 .get_queue_size = qvirtio_mmio_get_queue_size,
216 .set_queue_address = qvirtio_mmio_set_queue_address,
217 .virtqueue_setup = qvirtio_mmio_virtqueue_setup,
f1d3b991 218 .virtqueue_cleanup = qvirtio_mmio_virtqueue_cleanup,
0a6ed700
MM
219 .virtqueue_kick = qvirtio_mmio_virtqueue_kick,
220};
221
57ed038a 222static void *qvirtio_mmio_get_driver(void *obj, const char *interface)
0a6ed700 223{
57ed038a
EGE
224 QVirtioMMIODevice *virtio_mmio = obj;
225 if (!g_strcmp0(interface, "virtio-bus")) {
226 return &virtio_mmio->vdev;
227 }
228 fprintf(stderr, "%s not present in virtio-mmio\n", interface);
229 g_assert_not_reached();
230}
231
232static void qvirtio_mmio_start_hw(QOSGraphObject *obj)
233{
234 QVirtioMMIODevice *dev = (QVirtioMMIODevice *) obj;
235 qvirtio_start_device(&dev->vdev);
236}
0a6ed700 237
57ed038a
EGE
238void qvirtio_mmio_init_device(QVirtioMMIODevice *dev, QTestState *qts,
239 uint64_t addr, uint32_t page_size)
240{
241 uint32_t magic;
242 magic = qtest_readl(qts, addr + QVIRTIO_MMIO_MAGIC_VALUE);
0a6ed700
MM
243 g_assert(magic == ('v' | 'i' << 8 | 'r' << 16 | 't' << 24));
244
60ce3a40
SH
245 dev->version = qtest_readl(qts, addr + QVIRTIO_MMIO_VERSION);
246 g_assert(dev->version == 1 || dev->version == 2);
247
57ed038a 248 dev->qts = qts;
0a6ed700
MM
249 dev->addr = addr;
250 dev->page_size = page_size;
57ed038a 251 dev->vdev.device_type = qtest_readl(qts, addr + QVIRTIO_MMIO_DEVICE_ID);
6b9cdf4c 252 dev->vdev.bus = &qvirtio_mmio;
0a6ed700 253
57ed038a
EGE
254 qtest_writel(qts, addr + QVIRTIO_MMIO_GUEST_PAGE_SIZE, page_size);
255
256 dev->obj.get_driver = qvirtio_mmio_get_driver;
257 dev->obj.start_hw = qvirtio_mmio_start_hw;
258}
0a6ed700 259
57ed038a
EGE
260static void virtio_mmio_register_nodes(void)
261{
262 qos_node_create_driver("virtio-mmio", NULL);
263 qos_node_produces("virtio-mmio", "virtio-bus");
0a6ed700 264}
57ed038a
EGE
265
266libqos_init(virtio_mmio_register_nodes);
This page took 0.304934 seconds and 4 git commands to generate.