virtio.c 25.7 KB
Newer Older
A
aliguori 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
/*
 * Virtio Support
 *
 * Copyright IBM, Corp. 2007
 *
 * Authors:
 *  Anthony Liguori   <aliguori@us.ibm.com>
 *
 * This work is licensed under the terms of the GNU GPL, version 2.  See
 * the COPYING file in the top-level directory.
 *
 */

#include <inttypes.h>

16
#include "trace.h"
17
#include "qemu-error.h"
A
aliguori 已提交
18 19
#include "virtio.h"

20 21 22 23
/* The alignment to use between consumer and producer parts of vring.
 * x86 pagesize again. */
#define VIRTIO_PCI_VRING_ALIGN         4096

A
aliguori 已提交
24 25 26
/* QEMU doesn't strictly need write barriers since everything runs in
 * lock-step.  We'll leave the calls to wmb() in though to make it obvious for
 * KVM or if kqemu gets SMP support.
27 28
 * In any case, we must prevent the compiler from reordering the code.
 * TODO: we likely need some rmb()/mb() as well.
A
aliguori 已提交
29
 */
30 31

#define wmb() __asm__ __volatile__("": : :"memory")
A
aliguori 已提交
32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63

typedef struct VRingDesc
{
    uint64_t addr;
    uint32_t len;
    uint16_t flags;
    uint16_t next;
} VRingDesc;

typedef struct VRingAvail
{
    uint16_t flags;
    uint16_t idx;
    uint16_t ring[0];
} VRingAvail;

typedef struct VRingUsedElem
{
    uint32_t id;
    uint32_t len;
} VRingUsedElem;

typedef struct VRingUsed
{
    uint16_t flags;
    uint16_t idx;
    VRingUsedElem ring[0];
} VRingUsed;

typedef struct VRing
{
    unsigned int num;
A
Anthony Liguori 已提交
64 65 66
    target_phys_addr_t desc;
    target_phys_addr_t avail;
    target_phys_addr_t used;
A
aliguori 已提交
67 68 69 70 71
} VRing;

struct VirtQueue
{
    VRing vring;
A
Anthony Liguori 已提交
72
    target_phys_addr_t pa;
A
aliguori 已提交
73
    uint16_t last_avail_idx;
M
Michael S. Tsirkin 已提交
74 75 76 77 78 79 80 81 82
    /* Last used index value we have signalled on */
    uint16_t signalled_used;

    /* Last used index value we have signalled on */
    bool signalled_used_valid;

    /* Notification enabled? */
    bool notification;

A
aliguori 已提交
83
    int inuse;
M
Michael S. Tsirkin 已提交
84

85
    uint16_t vector;
A
aliguori 已提交
86
    void (*handle_output)(VirtIODevice *vdev, VirtQueue *vq);
87 88 89
    VirtIODevice *vdev;
    EventNotifier guest_notifier;
    EventNotifier host_notifier;
A
aliguori 已提交
90 91 92
};

/* virt queue functions */
P
Paul Brook 已提交
93
static void virtqueue_init(VirtQueue *vq)
A
aliguori 已提交
94
{
A
Anthony Liguori 已提交
95
    target_phys_addr_t pa = vq->pa;
P
Paul Brook 已提交
96

A
aliguori 已提交
97 98
    vq->vring.desc = pa;
    vq->vring.avail = pa + vq->vring.num * sizeof(VRingDesc);
99 100 101
    vq->vring.used = vring_align(vq->vring.avail +
                                 offsetof(VRingAvail, ring[vq->vring.num]),
                                 VIRTIO_PCI_VRING_ALIGN);
A
aliguori 已提交
102 103
}

A
Anthony Liguori 已提交
104
static inline uint64_t vring_desc_addr(target_phys_addr_t desc_pa, int i)
A
aliguori 已提交
105
{
A
Anthony Liguori 已提交
106
    target_phys_addr_t pa;
107
    pa = desc_pa + sizeof(VRingDesc) * i + offsetof(VRingDesc, addr);
A
aliguori 已提交
108 109 110
    return ldq_phys(pa);
}

A
Anthony Liguori 已提交
111
static inline uint32_t vring_desc_len(target_phys_addr_t desc_pa, int i)
A
aliguori 已提交
112
{
A
Anthony Liguori 已提交
113
    target_phys_addr_t pa;
114
    pa = desc_pa + sizeof(VRingDesc) * i + offsetof(VRingDesc, len);
A
aliguori 已提交
115 116 117
    return ldl_phys(pa);
}

A
Anthony Liguori 已提交
118
static inline uint16_t vring_desc_flags(target_phys_addr_t desc_pa, int i)
A
aliguori 已提交
119
{
A
Anthony Liguori 已提交
120
    target_phys_addr_t pa;
121
    pa = desc_pa + sizeof(VRingDesc) * i + offsetof(VRingDesc, flags);
A
aliguori 已提交
122 123 124
    return lduw_phys(pa);
}

A
Anthony Liguori 已提交
125
static inline uint16_t vring_desc_next(target_phys_addr_t desc_pa, int i)
A
aliguori 已提交
126
{
A
Anthony Liguori 已提交
127
    target_phys_addr_t pa;
128
    pa = desc_pa + sizeof(VRingDesc) * i + offsetof(VRingDesc, next);
A
aliguori 已提交
129 130 131 132 133
    return lduw_phys(pa);
}

static inline uint16_t vring_avail_flags(VirtQueue *vq)
{
A
Anthony Liguori 已提交
134
    target_phys_addr_t pa;
A
aliguori 已提交
135 136 137 138 139 140
    pa = vq->vring.avail + offsetof(VRingAvail, flags);
    return lduw_phys(pa);
}

static inline uint16_t vring_avail_idx(VirtQueue *vq)
{
A
Anthony Liguori 已提交
141
    target_phys_addr_t pa;
A
aliguori 已提交
142 143 144 145 146 147
    pa = vq->vring.avail + offsetof(VRingAvail, idx);
    return lduw_phys(pa);
}

static inline uint16_t vring_avail_ring(VirtQueue *vq, int i)
{
A
Anthony Liguori 已提交
148
    target_phys_addr_t pa;
A
aliguori 已提交
149 150 151 152
    pa = vq->vring.avail + offsetof(VRingAvail, ring[i]);
    return lduw_phys(pa);
}

M
Michael S. Tsirkin 已提交
153 154 155 156 157
static inline uint16_t vring_used_event(VirtQueue *vq)
{
    return vring_avail_ring(vq, vq->vring.num);
}

A
aliguori 已提交
158 159
static inline void vring_used_ring_id(VirtQueue *vq, int i, uint32_t val)
{
A
Anthony Liguori 已提交
160
    target_phys_addr_t pa;
A
aliguori 已提交
161 162 163 164 165 166
    pa = vq->vring.used + offsetof(VRingUsed, ring[i].id);
    stl_phys(pa, val);
}

static inline void vring_used_ring_len(VirtQueue *vq, int i, uint32_t val)
{
A
Anthony Liguori 已提交
167
    target_phys_addr_t pa;
A
aliguori 已提交
168 169 170 171 172 173
    pa = vq->vring.used + offsetof(VRingUsed, ring[i].len);
    stl_phys(pa, val);
}

static uint16_t vring_used_idx(VirtQueue *vq)
{
A
Anthony Liguori 已提交
174
    target_phys_addr_t pa;
A
aliguori 已提交
175 176 177 178
    pa = vq->vring.used + offsetof(VRingUsed, idx);
    return lduw_phys(pa);
}

M
Michael S. Tsirkin 已提交
179
static inline void vring_used_idx_set(VirtQueue *vq, uint16_t val)
A
aliguori 已提交
180
{
A
Anthony Liguori 已提交
181
    target_phys_addr_t pa;
A
aliguori 已提交
182
    pa = vq->vring.used + offsetof(VRingUsed, idx);
M
Michael S. Tsirkin 已提交
183
    stw_phys(pa, val);
A
aliguori 已提交
184 185 186 187
}

static inline void vring_used_flags_set_bit(VirtQueue *vq, int mask)
{
A
Anthony Liguori 已提交
188
    target_phys_addr_t pa;
A
aliguori 已提交
189 190 191 192 193 194
    pa = vq->vring.used + offsetof(VRingUsed, flags);
    stw_phys(pa, lduw_phys(pa) | mask);
}

static inline void vring_used_flags_unset_bit(VirtQueue *vq, int mask)
{
A
Anthony Liguori 已提交
195
    target_phys_addr_t pa;
A
aliguori 已提交
196 197 198 199
    pa = vq->vring.used + offsetof(VRingUsed, flags);
    stw_phys(pa, lduw_phys(pa) & ~mask);
}

M
Michael S. Tsirkin 已提交
200 201 202 203 204 205 206 207 208 209
static inline void vring_avail_event(VirtQueue *vq, uint16_t val)
{
    target_phys_addr_t pa;
    if (!vq->notification) {
        return;
    }
    pa = vq->vring.used + offsetof(VRingUsed, ring[vq->vring.num]);
    stw_phys(pa, val);
}

A
aliguori 已提交
210 211
void virtio_queue_set_notification(VirtQueue *vq, int enable)
{
M
Michael S. Tsirkin 已提交
212 213 214 215
    vq->notification = enable;
    if (vq->vdev->guest_features & (1 << VIRTIO_RING_F_EVENT_IDX)) {
        vring_avail_event(vq, vring_avail_idx(vq));
    } else if (enable) {
A
aliguori 已提交
216
        vring_used_flags_unset_bit(vq, VRING_USED_F_NO_NOTIFY);
M
Michael S. Tsirkin 已提交
217
    } else {
A
aliguori 已提交
218
        vring_used_flags_set_bit(vq, VRING_USED_F_NO_NOTIFY);
M
Michael S. Tsirkin 已提交
219
    }
A
aliguori 已提交
220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237
}

int virtio_queue_ready(VirtQueue *vq)
{
    return vq->vring.avail != 0;
}

int virtio_queue_empty(VirtQueue *vq)
{
    return vring_avail_idx(vq) == vq->last_avail_idx;
}

void virtqueue_fill(VirtQueue *vq, const VirtQueueElement *elem,
                    unsigned int len, unsigned int idx)
{
    unsigned int offset;
    int i;

238 239
    trace_virtqueue_fill(vq, elem, len, idx);

A
aliguori 已提交
240 241 242 243
    offset = 0;
    for (i = 0; i < elem->in_num; i++) {
        size_t size = MIN(len - offset, elem->in_sg[i].iov_len);

244 245 246
        cpu_physical_memory_unmap(elem->in_sg[i].iov_base,
                                  elem->in_sg[i].iov_len,
                                  1, size);
A
aliguori 已提交
247

248
        offset += elem->in_sg[i].iov_len;
A
aliguori 已提交
249 250
    }

251 252 253 254 255
    for (i = 0; i < elem->out_num; i++)
        cpu_physical_memory_unmap(elem->out_sg[i].iov_base,
                                  elem->out_sg[i].iov_len,
                                  0, elem->out_sg[i].iov_len);

A
aliguori 已提交
256 257 258 259 260 261 262 263 264
    idx = (idx + vring_used_idx(vq)) % vq->vring.num;

    /* Get a pointer to the next entry in the used ring. */
    vring_used_ring_id(vq, idx, elem->index);
    vring_used_ring_len(vq, idx, len);
}

void virtqueue_flush(VirtQueue *vq, unsigned int count)
{
M
Michael S. Tsirkin 已提交
265
    uint16_t old, new;
A
aliguori 已提交
266 267
    /* Make sure buffer is written before we update index. */
    wmb();
268
    trace_virtqueue_flush(vq, count);
M
Michael S. Tsirkin 已提交
269 270 271
    old = vring_used_idx(vq);
    new = old + count;
    vring_used_idx_set(vq, new);
A
aliguori 已提交
272
    vq->inuse -= count;
M
Michael S. Tsirkin 已提交
273 274
    if (unlikely((int16_t)(new - vq->signalled_used) < (uint16_t)(new - old)))
        vq->signalled_used_valid = false;
A
aliguori 已提交
275 276 277 278 279 280 281 282 283 284 285 286 287 288
}

void virtqueue_push(VirtQueue *vq, const VirtQueueElement *elem,
                    unsigned int len)
{
    virtqueue_fill(vq, elem, len, 0);
    virtqueue_flush(vq, 1);
}

static int virtqueue_num_heads(VirtQueue *vq, unsigned int idx)
{
    uint16_t num_heads = vring_avail_idx(vq) - idx;

    /* Check it isn't doing very strange things with descriptor numbers. */
A
aliguori 已提交
289
    if (num_heads > vq->vring.num) {
290 291
        error_report("Guest moved used index from %u to %u",
                     idx, vring_avail_idx(vq));
A
aliguori 已提交
292 293
        exit(1);
    }
A
aliguori 已提交
294 295 296 297 298 299 300 301 302 303 304 305 306

    return num_heads;
}

static unsigned int virtqueue_get_head(VirtQueue *vq, unsigned int idx)
{
    unsigned int head;

    /* Grab the next descriptor number they're advertising, and increment
     * the index we've seen. */
    head = vring_avail_ring(vq, idx % vq->vring.num);

    /* If their number is silly, that's a fatal mistake. */
A
aliguori 已提交
307
    if (head >= vq->vring.num) {
308
        error_report("Guest says index %u is available", head);
A
aliguori 已提交
309 310
        exit(1);
    }
A
aliguori 已提交
311 312 313 314

    return head;
}

A
Anthony Liguori 已提交
315
static unsigned virtqueue_next_desc(target_phys_addr_t desc_pa,
316
                                    unsigned int i, unsigned int max)
A
aliguori 已提交
317 318 319 320
{
    unsigned int next;

    /* If this descriptor says it doesn't chain, we're done. */
321 322
    if (!(vring_desc_flags(desc_pa, i) & VRING_DESC_F_NEXT))
        return max;
A
aliguori 已提交
323 324

    /* Check they're not leading us off end of descriptors. */
325
    next = vring_desc_next(desc_pa, i);
A
aliguori 已提交
326 327 328
    /* Make sure compiler knows to grab that: we don't want it changing! */
    wmb();

329
    if (next >= max) {
330
        error_report("Desc next is %u", next);
A
aliguori 已提交
331 332
        exit(1);
    }
A
aliguori 已提交
333 334 335 336 337 338

    return next;
}

int virtqueue_avail_bytes(VirtQueue *vq, int in_bytes, int out_bytes)
{
339 340
    unsigned int idx;
    int total_bufs, in_total, out_total;
A
aliguori 已提交
341 342 343

    idx = vq->last_avail_idx;

344
    total_bufs = in_total = out_total = 0;
A
aliguori 已提交
345
    while (virtqueue_num_heads(vq, idx)) {
346
        unsigned int max, num_bufs, indirect = 0;
A
Anthony Liguori 已提交
347
        target_phys_addr_t desc_pa;
A
aliguori 已提交
348 349
        int i;

350 351
        max = vq->vring.num;
        num_bufs = total_bufs;
A
aliguori 已提交
352
        i = virtqueue_get_head(vq, idx++);
353 354 355 356
        desc_pa = vq->vring.desc;

        if (vring_desc_flags(desc_pa, i) & VRING_DESC_F_INDIRECT) {
            if (vring_desc_len(desc_pa, i) % sizeof(VRingDesc)) {
357
                error_report("Invalid size for indirect buffer table");
358 359 360 361 362
                exit(1);
            }

            /* If we've got too many, that implies a descriptor loop. */
            if (num_bufs >= max) {
363
                error_report("Looped descriptor");
364 365 366 367 368 369 370 371 372 373
                exit(1);
            }

            /* loop over the indirect descriptor table */
            indirect = 1;
            max = vring_desc_len(desc_pa, i) / sizeof(VRingDesc);
            num_bufs = i = 0;
            desc_pa = vring_desc_addr(desc_pa, i);
        }

A
aliguori 已提交
374 375
        do {
            /* If we've got too many, that implies a descriptor loop. */
376
            if (++num_bufs > max) {
377
                error_report("Looped descriptor");
A
aliguori 已提交
378 379
                exit(1);
            }
A
aliguori 已提交
380

381
            if (vring_desc_flags(desc_pa, i) & VRING_DESC_F_WRITE) {
A
aliguori 已提交
382
                if (in_bytes > 0 &&
383
                    (in_total += vring_desc_len(desc_pa, i)) >= in_bytes)
A
aliguori 已提交
384 385 386
                    return 1;
            } else {
                if (out_bytes > 0 &&
387
                    (out_total += vring_desc_len(desc_pa, i)) >= out_bytes)
A
aliguori 已提交
388 389
                    return 1;
            }
390
        } while ((i = virtqueue_next_desc(desc_pa, i, max)) != max);
391 392 393 394 395

        if (!indirect)
            total_bufs = num_bufs;
        else
            total_bufs++;
A
aliguori 已提交
396 397 398 399 400
    }

    return 0;
}

K
Kevin Wolf 已提交
401 402 403 404 405 406 407 408 409 410
void virtqueue_map_sg(struct iovec *sg, target_phys_addr_t *addr,
    size_t num_sg, int is_write)
{
    unsigned int i;
    target_phys_addr_t len;

    for (i = 0; i < num_sg; i++) {
        len = sg[i].iov_len;
        sg[i].iov_base = cpu_physical_memory_map(addr[i], &len, is_write);
        if (sg[i].iov_base == NULL || len != sg[i].iov_len) {
411
            error_report("virtio: trying to map MMIO memory");
K
Kevin Wolf 已提交
412 413 414 415 416
            exit(1);
        }
    }
}

A
aliguori 已提交
417 418
int virtqueue_pop(VirtQueue *vq, VirtQueueElement *elem)
{
419
    unsigned int i, head, max;
A
Anthony Liguori 已提交
420
    target_phys_addr_t desc_pa = vq->vring.desc;
A
aliguori 已提交
421 422 423 424 425 426 427

    if (!virtqueue_num_heads(vq, vq->last_avail_idx))
        return 0;

    /* When we start there are none of either input nor output. */
    elem->out_num = elem->in_num = 0;

428 429
    max = vq->vring.num;

A
aliguori 已提交
430
    i = head = virtqueue_get_head(vq, vq->last_avail_idx++);
M
Michael S. Tsirkin 已提交
431 432 433
    if (vq->vdev->guest_features & (1 << VIRTIO_RING_F_EVENT_IDX)) {
        vring_avail_event(vq, vring_avail_idx(vq));
    }
434 435 436

    if (vring_desc_flags(desc_pa, i) & VRING_DESC_F_INDIRECT) {
        if (vring_desc_len(desc_pa, i) % sizeof(VRingDesc)) {
437
            error_report("Invalid size for indirect buffer table");
438 439 440 441 442 443 444 445 446
            exit(1);
        }

        /* loop over the indirect descriptor table */
        max = vring_desc_len(desc_pa, i) / sizeof(VRingDesc);
        desc_pa = vring_desc_addr(desc_pa, i);
        i = 0;
    }

K
Kevin Wolf 已提交
447
    /* Collect all the descriptors */
A
aliguori 已提交
448 449 450
    do {
        struct iovec *sg;

451
        if (vring_desc_flags(desc_pa, i) & VRING_DESC_F_WRITE) {
452 453 454 455
            if (elem->in_num >= ARRAY_SIZE(elem->in_sg)) {
                error_report("Too many write descriptors in indirect table");
                exit(1);
            }
456
            elem->in_addr[elem->in_num] = vring_desc_addr(desc_pa, i);
A
aliguori 已提交
457
            sg = &elem->in_sg[elem->in_num++];
K
Kevin Wolf 已提交
458
        } else {
459 460 461 462
            if (elem->out_num >= ARRAY_SIZE(elem->out_sg)) {
                error_report("Too many read descriptors in indirect table");
                exit(1);
            }
K
Kevin Wolf 已提交
463
            elem->out_addr[elem->out_num] = vring_desc_addr(desc_pa, i);
A
aliguori 已提交
464
            sg = &elem->out_sg[elem->out_num++];
K
Kevin Wolf 已提交
465
        }
A
aliguori 已提交
466

467
        sg->iov_len = vring_desc_len(desc_pa, i);
A
aliguori 已提交
468 469

        /* If we've got too many, that implies a descriptor loop. */
470
        if ((elem->in_num + elem->out_num) > max) {
471
            error_report("Looped descriptor");
A
aliguori 已提交
472 473
            exit(1);
        }
474
    } while ((i = virtqueue_next_desc(desc_pa, i, max)) != max);
A
aliguori 已提交
475

K
Kevin Wolf 已提交
476 477 478 479
    /* Now map what we have collected */
    virtqueue_map_sg(elem->in_sg, elem->in_addr, elem->in_num, 1);
    virtqueue_map_sg(elem->out_sg, elem->out_addr, elem->out_num, 0);

A
aliguori 已提交
480 481 482 483
    elem->index = head;

    vq->inuse++;

484
    trace_virtqueue_pop(vq, elem, elem->in_num, elem->out_num);
A
aliguori 已提交
485 486 487 488
    return elem->in_num + elem->out_num;
}

/* virtio device */
489 490 491 492 493 494
static void virtio_notify_vector(VirtIODevice *vdev, uint16_t vector)
{
    if (vdev->binding->notify) {
        vdev->binding->notify(vdev->binding_opaque, vector);
    }
}
A
aliguori 已提交
495

P
Paul Brook 已提交
496
void virtio_update_irq(VirtIODevice *vdev)
A
aliguori 已提交
497
{
498
    virtio_notify_vector(vdev, VIRTIO_NO_VECTOR);
A
aliguori 已提交
499 500
}

501 502 503 504 505 506 507 508 509 510
void virtio_set_status(VirtIODevice *vdev, uint8_t val)
{
    trace_virtio_set_status(vdev, val);

    if (vdev->set_status) {
        vdev->set_status(vdev, val);
    }
    vdev->status = val;
}

P
Paul Brook 已提交
511
void virtio_reset(void *opaque)
A
aliguori 已提交
512 513 514 515
{
    VirtIODevice *vdev = opaque;
    int i;

516 517
    virtio_set_status(vdev, 0);

A
aliguori 已提交
518 519 520
    if (vdev->reset)
        vdev->reset(vdev);

521
    vdev->guest_features = 0;
A
aliguori 已提交
522 523 524
    vdev->queue_sel = 0;
    vdev->status = 0;
    vdev->isr = 0;
525 526
    vdev->config_vector = VIRTIO_NO_VECTOR;
    virtio_notify_vector(vdev, vdev->config_vector);
A
aliguori 已提交
527 528 529 530 531 532

    for(i = 0; i < VIRTIO_PCI_QUEUE_MAX; i++) {
        vdev->vq[i].vring.desc = 0;
        vdev->vq[i].vring.avail = 0;
        vdev->vq[i].vring.used = 0;
        vdev->vq[i].last_avail_idx = 0;
P
Paul Brook 已提交
533
        vdev->vq[i].pa = 0;
534
        vdev->vq[i].vector = VIRTIO_NO_VECTOR;
M
Michael S. Tsirkin 已提交
535 536 537
        vdev->vq[i].signalled_used = 0;
        vdev->vq[i].signalled_used_valid = false;
        vdev->vq[i].notification = true;
A
aliguori 已提交
538 539 540
    }
}

P
Paul Brook 已提交
541
uint32_t virtio_config_readb(VirtIODevice *vdev, uint32_t addr)
A
aliguori 已提交
542 543 544 545 546 547 548 549 550 551 552 553
{
    uint8_t val;

    vdev->get_config(vdev, vdev->config);

    if (addr > (vdev->config_len - sizeof(val)))
        return (uint32_t)-1;

    memcpy(&val, vdev->config + addr, sizeof(val));
    return val;
}

P
Paul Brook 已提交
554
uint32_t virtio_config_readw(VirtIODevice *vdev, uint32_t addr)
A
aliguori 已提交
555 556 557 558 559 560 561 562 563 564 565 566
{
    uint16_t val;

    vdev->get_config(vdev, vdev->config);

    if (addr > (vdev->config_len - sizeof(val)))
        return (uint32_t)-1;

    memcpy(&val, vdev->config + addr, sizeof(val));
    return val;
}

P
Paul Brook 已提交
567
uint32_t virtio_config_readl(VirtIODevice *vdev, uint32_t addr)
A
aliguori 已提交
568 569 570 571 572 573 574 575 576 577 578 579
{
    uint32_t val;

    vdev->get_config(vdev, vdev->config);

    if (addr > (vdev->config_len - sizeof(val)))
        return (uint32_t)-1;

    memcpy(&val, vdev->config + addr, sizeof(val));
    return val;
}

P
Paul Brook 已提交
580
void virtio_config_writeb(VirtIODevice *vdev, uint32_t addr, uint32_t data)
A
aliguori 已提交
581 582 583 584 585 586 587 588 589 590 591 592
{
    uint8_t val = data;

    if (addr > (vdev->config_len - sizeof(val)))
        return;

    memcpy(vdev->config + addr, &val, sizeof(val));

    if (vdev->set_config)
        vdev->set_config(vdev, vdev->config);
}

P
Paul Brook 已提交
593
void virtio_config_writew(VirtIODevice *vdev, uint32_t addr, uint32_t data)
A
aliguori 已提交
594 595 596 597 598 599 600 601 602 603 604 605
{
    uint16_t val = data;

    if (addr > (vdev->config_len - sizeof(val)))
        return;

    memcpy(vdev->config + addr, &val, sizeof(val));

    if (vdev->set_config)
        vdev->set_config(vdev, vdev->config);
}

P
Paul Brook 已提交
606
void virtio_config_writel(VirtIODevice *vdev, uint32_t addr, uint32_t data)
A
aliguori 已提交
607 608 609 610 611 612 613 614 615 616 617 618
{
    uint32_t val = data;

    if (addr > (vdev->config_len - sizeof(val)))
        return;

    memcpy(vdev->config + addr, &val, sizeof(val));

    if (vdev->set_config)
        vdev->set_config(vdev, vdev->config);
}

A
Anthony Liguori 已提交
619
void virtio_queue_set_addr(VirtIODevice *vdev, int n, target_phys_addr_t addr)
A
aliguori 已提交
620
{
621 622
    vdev->vq[n].pa = addr;
    virtqueue_init(&vdev->vq[n]);
P
Paul Brook 已提交
623 624
}

A
Anthony Liguori 已提交
625
target_phys_addr_t virtio_queue_get_addr(VirtIODevice *vdev, int n)
P
Paul Brook 已提交
626 627 628 629 630 631 632 633
{
    return vdev->vq[n].pa;
}

int virtio_queue_get_num(VirtIODevice *vdev, int n)
{
    return vdev->vq[n].vring.num;
}
A
aliguori 已提交
634

635 636 637 638 639 640 641 642 643
void virtio_queue_notify_vq(VirtQueue *vq)
{
    if (vq->vring.desc) {
        VirtIODevice *vdev = vq->vdev;
        trace_virtio_queue_notify(vdev, vq - vdev->vq, vq);
        vq->handle_output(vdev, vq);
    }
}

P
Paul Brook 已提交
644 645
void virtio_queue_notify(VirtIODevice *vdev, int n)
{
646
    virtio_queue_notify_vq(&vdev->vq[n]);
A
aliguori 已提交
647 648
}

649 650 651 652 653 654 655 656 657 658 659 660
uint16_t virtio_queue_vector(VirtIODevice *vdev, int n)
{
    return n < VIRTIO_PCI_QUEUE_MAX ? vdev->vq[n].vector :
        VIRTIO_NO_VECTOR;
}

void virtio_queue_set_vector(VirtIODevice *vdev, int n, uint16_t vector)
{
    if (n < VIRTIO_PCI_QUEUE_MAX)
        vdev->vq[n].vector = vector;
}

A
aliguori 已提交
661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679
VirtQueue *virtio_add_queue(VirtIODevice *vdev, int queue_size,
                            void (*handle_output)(VirtIODevice *, VirtQueue *))
{
    int i;

    for (i = 0; i < VIRTIO_PCI_QUEUE_MAX; i++) {
        if (vdev->vq[i].vring.num == 0)
            break;
    }

    if (i == VIRTIO_PCI_QUEUE_MAX || queue_size > VIRTQUEUE_MAX_SIZE)
        abort();

    vdev->vq[i].vring.num = queue_size;
    vdev->vq[i].handle_output = handle_output;

    return &vdev->vq[i];
}

680 681
void virtio_irq(VirtQueue *vq)
{
682
    trace_virtio_irq(vq);
683 684 685 686
    vq->vdev->isr |= 0x01;
    virtio_notify_vector(vq->vdev, vq->vector);
}

M
Michael S. Tsirkin 已提交
687 688 689 690
/* Assuming a given event_idx value from the other size, if
 * we have just incremented index from old to new_idx,
 * should we trigger an event? */
static inline int vring_need_event(uint16_t event, uint16_t new, uint16_t old)
A
aliguori 已提交
691
{
M
Michael S. Tsirkin 已提交
692 693 694 695 696 697 698 699 700 701 702 703
	/* Note: Xen has similar logic for notification hold-off
	 * in include/xen/interface/io/ring.h with req_event and req_prod
	 * corresponding to event_idx + 1 and new respectively.
	 * Note also that req_event and req_prod in Xen start at 1,
	 * event indexes in virtio start at 0. */
	return (uint16_t)(new - event - 1) < (uint16_t)(new - old);
}

static bool vring_notify(VirtIODevice *vdev, VirtQueue *vq)
{
    uint16_t old, new;
    bool v;
704
    /* Always notify when queue is empty (when feature acknowledge) */
M
Michael S. Tsirkin 已提交
705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723
    if (((vdev->guest_features & (1 << VIRTIO_F_NOTIFY_ON_EMPTY)) &&
         !vq->inuse && vring_avail_idx(vq) == vq->last_avail_idx)) {
        return true;
    }

    if (!(vdev->guest_features & (1 << VIRTIO_RING_F_EVENT_IDX))) {
        return !(vring_avail_flags(vq) & VRING_AVAIL_F_NO_INTERRUPT);
    }

    v = vq->signalled_used_valid;
    vq->signalled_used_valid = true;
    old = vq->signalled_used;
    new = vq->signalled_used = vring_used_idx(vq);
    return !v || vring_need_event(vring_used_event(vq), new, old);
}

void virtio_notify(VirtIODevice *vdev, VirtQueue *vq)
{
    if (!vring_notify(vdev, vq)) {
A
aliguori 已提交
724
        return;
M
Michael S. Tsirkin 已提交
725
    }
A
aliguori 已提交
726

727
    trace_virtio_notify(vdev, vq);
A
aliguori 已提交
728
    vdev->isr |= 0x01;
729
    virtio_notify_vector(vdev, vq->vector);
A
aliguori 已提交
730 731 732 733
}

void virtio_notify_config(VirtIODevice *vdev)
{
734 735 736
    if (!(vdev->status & VIRTIO_CONFIG_S_DRIVER_OK))
        return;

A
aliguori 已提交
737
    vdev->isr |= 0x03;
738
    virtio_notify_vector(vdev, vdev->config_vector);
A
aliguori 已提交
739 740 741 742 743 744
}

void virtio_save(VirtIODevice *vdev, QEMUFile *f)
{
    int i;

745 746
    if (vdev->binding->save_config)
        vdev->binding->save_config(vdev->binding_opaque, f);
A
aliguori 已提交
747 748 749 750

    qemu_put_8s(f, &vdev->status);
    qemu_put_8s(f, &vdev->isr);
    qemu_put_be16s(f, &vdev->queue_sel);
751
    qemu_put_be32s(f, &vdev->guest_features);
A
aliguori 已提交
752 753 754 755 756 757 758 759 760 761 762 763 764 765 766
    qemu_put_be32(f, vdev->config_len);
    qemu_put_buffer(f, vdev->config, vdev->config_len);

    for (i = 0; i < VIRTIO_PCI_QUEUE_MAX; i++) {
        if (vdev->vq[i].vring.num == 0)
            break;
    }

    qemu_put_be32(f, i);

    for (i = 0; i < VIRTIO_PCI_QUEUE_MAX; i++) {
        if (vdev->vq[i].vring.num == 0)
            break;

        qemu_put_be32(f, vdev->vq[i].vring.num);
P
Paul Brook 已提交
767
        qemu_put_be64(f, vdev->vq[i].pa);
A
aliguori 已提交
768
        qemu_put_be16s(f, &vdev->vq[i].last_avail_idx);
769 770
        if (vdev->binding->save_queue)
            vdev->binding->save_queue(vdev->binding_opaque, i, f);
A
aliguori 已提交
771 772 773
    }
}

774
int virtio_load(VirtIODevice *vdev, QEMUFile *f)
A
aliguori 已提交
775
{
776
    int num, i, ret;
777
    uint32_t features;
778
    uint32_t supported_features =
779
        vdev->binding->get_features(vdev->binding_opaque);
A
aliguori 已提交
780

781 782 783 784 785
    if (vdev->binding->load_config) {
        ret = vdev->binding->load_config(vdev->binding_opaque, f);
        if (ret)
            return ret;
    }
A
aliguori 已提交
786 787 788 789

    qemu_get_8s(f, &vdev->status);
    qemu_get_8s(f, &vdev->isr);
    qemu_get_be16s(f, &vdev->queue_sel);
790 791
    qemu_get_be32s(f, &features);
    if (features & ~supported_features) {
792 793
        error_report("Features 0x%x unsupported. Allowed features: 0x%x",
                     features, supported_features);
794 795
        return -1;
    }
796 797
    if (vdev->set_features)
        vdev->set_features(vdev, features);
798
    vdev->guest_features = features;
A
aliguori 已提交
799 800 801 802 803 804 805
    vdev->config_len = qemu_get_be32(f);
    qemu_get_buffer(f, vdev->config, vdev->config_len);

    num = qemu_get_be32(f);

    for (i = 0; i < num; i++) {
        vdev->vq[i].vring.num = qemu_get_be32(f);
P
Paul Brook 已提交
806
        vdev->vq[i].pa = qemu_get_be64(f);
A
aliguori 已提交
807
        qemu_get_be16s(f, &vdev->vq[i].last_avail_idx);
M
Michael S. Tsirkin 已提交
808 809
        vdev->vq[i].signalled_used_valid = false;
        vdev->vq[i].notification = true;
A
aliguori 已提交
810

P
Paul Brook 已提交
811
        if (vdev->vq[i].pa) {
M
Michael S. Tsirkin 已提交
812
            uint16_t nheads;
P
Paul Brook 已提交
813
            virtqueue_init(&vdev->vq[i]);
M
Michael S. Tsirkin 已提交
814 815 816 817
            nheads = vring_avail_idx(&vdev->vq[i]) - vdev->vq[i].last_avail_idx;
            /* Check it isn't doing very strange things with descriptor numbers. */
            if (nheads > vdev->vq[i].vring.num) {
                error_report("VQ %d size 0x%x Guest index 0x%x "
818
                             "inconsistent with Host index 0x%x: delta 0x%x",
M
Michael S. Tsirkin 已提交
819 820 821 822 823 824 825
                             i, vdev->vq[i].vring.num,
                             vring_avail_idx(&vdev->vq[i]),
                             vdev->vq[i].last_avail_idx, nheads);
                return -1;
            }
        } else if (vdev->vq[i].last_avail_idx) {
            error_report("VQ %d address 0x0 "
826
                         "inconsistent with Host index 0x%x",
M
Michael S. Tsirkin 已提交
827 828
                         i, vdev->vq[i].last_avail_idx);
                return -1;
829
	}
830 831 832 833
        if (vdev->binding->load_queue) {
            ret = vdev->binding->load_queue(vdev->binding_opaque, i, f);
            if (ret)
                return ret;
834
        }
A
aliguori 已提交
835 836
    }

837
    virtio_notify_vector(vdev, VIRTIO_NO_VECTOR);
838
    return 0;
A
aliguori 已提交
839 840
}

841 842
void virtio_cleanup(VirtIODevice *vdev)
{
843
    qemu_del_vm_change_state_handler(vdev->vmstate);
844
    if (vdev->config)
845 846 847
        g_free(vdev->config);
    g_free(vdev->vq);
    g_free(vdev);
848 849
}

850
static void virtio_vmstate_change(void *opaque, int running, RunState state)
851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868
{
    VirtIODevice *vdev = opaque;
    bool backend_run = running && (vdev->status & VIRTIO_CONFIG_S_DRIVER_OK);
    vdev->vm_running = running;

    if (backend_run) {
        virtio_set_status(vdev, vdev->status);
    }

    if (vdev->binding->vmstate_change) {
        vdev->binding->vmstate_change(vdev->binding_opaque, backend_run);
    }

    if (!backend_run) {
        virtio_set_status(vdev, vdev->status);
    }
}

P
Paul Brook 已提交
869 870
VirtIODevice *virtio_common_init(const char *name, uint16_t device_id,
                                 size_t config_size, size_t struct_size)
A
aliguori 已提交
871 872
{
    VirtIODevice *vdev;
873
    int i;
A
aliguori 已提交
874

875
    vdev = g_malloc0(struct_size);
A
aliguori 已提交
876

P
Paul Brook 已提交
877
    vdev->device_id = device_id;
A
aliguori 已提交
878 879 880
    vdev->status = 0;
    vdev->isr = 0;
    vdev->queue_sel = 0;
881
    vdev->config_vector = VIRTIO_NO_VECTOR;
882
    vdev->vq = g_malloc0(sizeof(VirtQueue) * VIRTIO_PCI_QUEUE_MAX);
883
    vdev->vm_running = runstate_is_running();
884
    for(i = 0; i < VIRTIO_PCI_QUEUE_MAX; i++) {
885
        vdev->vq[i].vector = VIRTIO_NO_VECTOR;
886 887
        vdev->vq[i].vdev = vdev;
    }
A
aliguori 已提交
888 889 890 891

    vdev->name = name;
    vdev->config_len = config_size;
    if (vdev->config_len)
892
        vdev->config = g_malloc0(config_size);
A
aliguori 已提交
893 894 895
    else
        vdev->config = NULL;

896 897
    vdev->vmstate = qemu_add_vm_change_state_handler(virtio_vmstate_change, vdev);

A
aliguori 已提交
898 899
    return vdev;
}
P
Paul Brook 已提交
900 901 902 903 904 905 906

void virtio_bind_device(VirtIODevice *vdev, const VirtIOBindings *binding,
                        void *opaque)
{
    vdev->binding = binding;
    vdev->binding_opaque = opaque;
}
907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935

target_phys_addr_t virtio_queue_get_desc_addr(VirtIODevice *vdev, int n)
{
    return vdev->vq[n].vring.desc;
}

target_phys_addr_t virtio_queue_get_avail_addr(VirtIODevice *vdev, int n)
{
    return vdev->vq[n].vring.avail;
}

target_phys_addr_t virtio_queue_get_used_addr(VirtIODevice *vdev, int n)
{
    return vdev->vq[n].vring.used;
}

target_phys_addr_t virtio_queue_get_ring_addr(VirtIODevice *vdev, int n)
{
    return vdev->vq[n].vring.desc;
}

target_phys_addr_t virtio_queue_get_desc_size(VirtIODevice *vdev, int n)
{
    return sizeof(VRingDesc) * vdev->vq[n].vring.num;
}

target_phys_addr_t virtio_queue_get_avail_size(VirtIODevice *vdev, int n)
{
    return offsetof(VRingAvail, ring) +
936
        sizeof(uint64_t) * vdev->vq[n].vring.num;
937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973
}

target_phys_addr_t virtio_queue_get_used_size(VirtIODevice *vdev, int n)
{
    return offsetof(VRingUsed, ring) +
        sizeof(VRingUsedElem) * vdev->vq[n].vring.num;
}

target_phys_addr_t virtio_queue_get_ring_size(VirtIODevice *vdev, int n)
{
    return vdev->vq[n].vring.used - vdev->vq[n].vring.desc +
	    virtio_queue_get_used_size(vdev, n);
}

uint16_t virtio_queue_get_last_avail_idx(VirtIODevice *vdev, int n)
{
    return vdev->vq[n].last_avail_idx;
}

void virtio_queue_set_last_avail_idx(VirtIODevice *vdev, int n, uint16_t idx)
{
    vdev->vq[n].last_avail_idx = idx;
}

VirtQueue *virtio_get_queue(VirtIODevice *vdev, int n)
{
    return vdev->vq + n;
}

EventNotifier *virtio_queue_get_guest_notifier(VirtQueue *vq)
{
    return &vq->guest_notifier;
}
EventNotifier *virtio_queue_get_host_notifier(VirtQueue *vq)
{
    return &vq->host_notifier;
}