L4Re Operating System Framework
Interface and Usage Documentation
Loading...
Searching...
No Matches
virtqueue
1// vi:set ft=cpp: -*- Mode: C++ -*-
2/* SPDX-License-Identifier: MIT */
3/*
4 * (c) 2014 Alexander Warg <warg@os.inf.tu-dresden.de>
5 */
6
7#include <l4/re/util/debug>
8#include <l4/sys/types.h>
9#include <l4/sys/err.h>
10#include <l4/cxx/bitfield>
11#include <l4/cxx/exceptions>
12#include <cstdint>
13
14#pragma once
15
16namespace L4virtio {
17
18#if defined(__ARM_ARCH) && __ARM_ARCH == 7
19static inline void wmb() { asm volatile ("dmb ishst" : : : "memory"); }
20static inline void rmb() { asm volatile ("dmb ish" : : : "memory"); }
21static inline void mb() { asm volatile ("dmb ish" : : : "memory"); }
22#elif defined(__ARM_ARCH) && __ARM_ARCH >= 8
23static inline void wmb() { asm volatile ("dmb ishst" : : : "memory"); }
24static inline void rmb() { asm volatile ("dmb ishld" : : : "memory"); }
25static inline void mb() { asm volatile ("dmb ish" : : : "memory"); }
26#elif defined(__mips__)
27static inline void wmb() { asm volatile ("sync" : : : "memory"); }
28static inline void rmb() { asm volatile ("sync" : : : "memory"); }
29static inline void mb() { asm volatile ("sync" : : : "memory"); }
30#elif defined(__amd64__) || defined(__i386__) || defined(__i686__)
31static inline void wmb() { asm volatile ("sfence" : : : "memory"); }
32static inline void rmb() { asm volatile ("lfence" : : : "memory"); }
33static inline void mb() { asm volatile ("mfence" : : : "memory"); }
34#elif defined(__riscv)
35static inline void wmb() { asm volatile ("fence ow, ow" : : : "memory"); }
36static inline void rmb() { asm volatile ("fence ir, ir" : : : "memory"); }
37static inline void mb() { asm volatile ("fence iorw, iorw" : : : "memory"); }
38#else
39#warning Missing proper memory write barrier
40static inline void wmb() { asm volatile ("" : : : "memory"); }
41static inline void rmb() { asm volatile ("" : : : "memory"); }
42static inline void mb() { asm volatile ("" : : : "memory"); }
43#endif
44
45
52template< typename T >
53class Ptr
54{
55public:
58
59 Ptr() = default;
60
62 Ptr(Invalid_type) : _p(~0ULL) {}
63
65 explicit Ptr(l4_uint64_t vm_addr) : _p(vm_addr) {}
66
68 l4_uint64_t get() const { return _p; }
69
71 bool is_valid() const { return _p != ~0ULL; }
72
73private:
74 l4_uint64_t _p;
75};
76
77
87{
88public:
92 class Desc
93 {
94 public:
98 struct Flags
99 {
101 Flags() = default;
102
104 explicit Flags(l4_uint16_t v) : raw(v) {}
105
107 CXX_BITFIELD_MEMBER( 0, 0, next, raw);
109 CXX_BITFIELD_MEMBER( 1, 1, write, raw);
111 CXX_BITFIELD_MEMBER( 2, 2, indirect, raw);
112 };
113
118
122 void dump(unsigned idx) const
123 {
124 L4Re::Util::Dbg().printf("D[%04x]: %08llx (%x) f=%04x n=%04x\n",
125 idx, addr.get(),
126 len, static_cast<unsigned>(flags.raw),
127 static_cast<unsigned>(next));
128 }
129 };
130
134 class Avail
135 {
136 public:
140 struct Flags
141 {
143 Flags() = default;
144
146 explicit Flags(l4_uint16_t v) : raw(v) {}
147
149 CXX_BITFIELD_MEMBER( 0, 0, no_irq, raw);
150 };
151
155 };
156
160 struct Used_elem
161 {
162 Used_elem() = default;
163
174 };
175
179 class Used
180 {
181 public:
185 struct Flags
186 {
188 Flags() = default;
189
191 explicit Flags(l4_uint16_t v) : raw(v) {}
192
194 CXX_BITFIELD_MEMBER( 0, 0, no_notify, raw);
195 };
196
200 };
201
202protected:
203 Desc *_desc = nullptr;
204 Avail *_avail = nullptr;
205 Used *_used = nullptr;
206
209
215
219 Virtqueue() = default;
220
221 Virtqueue(Virtqueue const &) = delete;
222 ~Virtqueue() = default;
223
224public:
230 void disable()
231 { _desc = 0; }
232
236 enum
237 {
238 Desc_align = 4, //< Alignment of the descriptor table.
239 Avail_align = 1, //< Alignment of the available ring.
240 Used_align = 2, //< Alignment of the used ring.
241 };
242
251 static unsigned long total_size(unsigned num)
252 {
253 static_assert(Desc_align >= Avail_align,
254 "virtqueue alignment assumptions broken");
255 return l4_round_size(desc_size(num) + avail_size(num), Used_align)
256 + used_size(num);
257 }
258
267 static unsigned long desc_size(unsigned num)
268 { return num * 16; }
269
275 static unsigned long desc_align()
276 { return Desc_align; }
277
285 static unsigned long avail_size(unsigned num)
286 { return 2 * num + 6; }
287
293 static unsigned long avail_align()
294 { return Avail_align; }
295
304 static unsigned long used_size(unsigned num)
305 { return 8 * num + 6; }
306
312 static unsigned long used_align()
313 { return Used_align; }
314
320 unsigned long total_size() const
321 {
322 return (reinterpret_cast<char *>(_used) - reinterpret_cast<char *>(_desc))
323 + used_size(num());
324 }
325
329 unsigned long avail_offset() const
330 { return reinterpret_cast<char *>(_avail) - reinterpret_cast<char *>(_desc); }
331
335 unsigned long used_offset() const
336 { return reinterpret_cast<char *>(_used) - reinterpret_cast<char *>(_desc); }
337
355 void setup(unsigned num, void *desc, void *avail, void *used)
356 {
357 if (num > 0x10000)
358 throw L4::Runtime_error(-L4_EINVAL, "Queue too large.");
359
360 _idx_mask = num - 1;
361 _desc = static_cast<Desc*>(desc);
362 _avail = static_cast<Avail*>(avail);
363 _used = static_cast<Used*>(used);
364
365 _current_avail = 0;
366
367 L4Re::Util::Dbg().printf("VQ[%p]: num=%d d:%p a:%p u:%p\n",
368 this, num, _desc, _avail, _used);
369 }
370
384 void setup_simple(unsigned num, void *ring)
385 {
386 l4_addr_t desc = reinterpret_cast<l4_addr_t>(ring);
387 l4_addr_t avail = l4_round_size(desc + desc_size(num), Avail_align);
388 void *used = reinterpret_cast<void *>(
389 l4_round_size(avail + avail_size(num), Used_align));
390 setup(num, ring, reinterpret_cast<void *>(avail), used);
391 }
392
398 void dump(Desc const *d) const
399 { d->dump(d - _desc); }
400
406 bool ready() const
407 { return L4_LIKELY(_desc != 0); }
408
410 unsigned num() const
411 { return _idx_mask + 1; }
412
420 bool no_notify_guest() const
421 {
422 mb(); // All queue updates must be visible before the flag can be read!
423 return _avail->flags.no_irq();
424 }
425
433 bool no_notify_host() const
434 {
435 return _used->flags.no_notify();
436 }
437
443 void no_notify_host(bool value)
444 {
445 _used->flags.no_notify() = value;
446 }
447
456 l4_uint16_t get_avail_idx() const { return _avail->idx; }
457
464
465};
466
467namespace Driver {
468
477class Virtqueue : public L4virtio::Virtqueue
478{
479private:
481 l4_uint16_t _next_free;
482
483public:
484 enum End_of_queue
485 {
486 // Indicates the end of the queue.
487 Eoq = 0xFFFF
488 };
489
490 Virtqueue() : _next_free(Eoq) {}
491
501 void initialize_rings(unsigned num)
502 {
503 _used->idx = 0;
504 _avail->idx = 0;
505
506 // setup the freelist
507 for (l4_uint16_t d = 0; d < num - 1; ++d)
508 _desc[d].next = d + 1;
509 _desc[num - 1].next = Eoq;
510 _next_free = 0;
511 }
512
529 void init_queue(unsigned num, void *desc, void *avail, void *used)
530 {
531 setup(num, desc, avail, used);
533 }
534
544 void init_queue(unsigned num, void *base)
545 {
546 setup_simple(num, base);
548 }
549
550
566 {
567 l4_uint16_t idx = _next_free;
568 if (idx == Eoq)
569 return Eoq;
570
571 _next_free = _desc[idx].next;
572
573 return idx;
574 }
575
582 {
583 if (descno > _idx_mask)
584 throw L4::Bounds_error();
585
586 _avail->ring[_avail->idx & _idx_mask] = descno; // _avail->idx expected to wrap
587 wmb();
588 ++_avail->idx;
589 }
590
598 {
599 if (descno > _idx_mask)
600 throw L4::Bounds_error();
601
602 return _desc[descno];
603 }
604
617 {
618 if (_current_avail == _used->idx)
619 return Eoq;
620
621 auto elem = _used->ring[_current_avail++ & _idx_mask];
622
623 if (len)
624 *len = elem.len;
625
626 return elem.id;
627 }
628
639 {
640 if (head > _idx_mask || tail > _idx_mask)
641 throw L4::Bounds_error();
642
643 _desc[tail].next = _next_free;
644 _next_free = head;
645 }
646};
647
648}
649} // namespace L4virtio
Access out of bounds.
Definition exceptions:279
Exception for an abstract runtime error.
Definition exceptions:129
void free_descriptor(l4_uint16_t head, l4_uint16_t tail)
Free a chained list of descriptors in the descriptor queue.
Definition virtqueue:638
void enqueue_descriptor(l4_uint16_t descno)
Enqueue a descriptor in the available ring.
Definition virtqueue:581
void init_queue(unsigned num, void *base)
Initialize this virtqueue.
Definition virtqueue:544
l4_uint16_t alloc_descriptor()
Allocate and return an unused descriptor from the descriptor table.
Definition virtqueue:565
l4_uint16_t find_next_used(l4_uint32_t *len=nullptr)
Return the next finished block.
Definition virtqueue:616
void init_queue(unsigned num, void *desc, void *avail, void *used)
Initialize this virtqueue.
Definition virtqueue:529
Desc & desc(l4_uint16_t descno)
Return a reference to a descriptor in the descriptor table.
Definition virtqueue:597
void initialize_rings(unsigned num)
Initialize the descriptor table and the index structures of this queue.
Definition virtqueue:501
Pointer used in virtio descriptors.
Definition virtqueue:54
Ptr(l4_uint64_t vm_addr)
Make a Ptr from a raw 64bit address.
Definition virtqueue:65
l4_uint64_t get() const
Definition virtqueue:68
Invalid_type
Type for making an invalid (NULL) Ptr.
Definition virtqueue:57
@ Invalid
Use to set a Ptr to invalid (NULL).
Definition virtqueue:57
bool is_valid() const
Definition virtqueue:71
Ptr(Invalid_type)
Make and invalid Ptr.
Definition virtqueue:62
Type of available ring, this is read-only for the host.
Definition virtqueue:135
l4_uint16_t ring[]
array of available descriptor indexes.
Definition virtqueue:154
Flags flags
flags of available ring
Definition virtqueue:152
l4_uint16_t idx
available index written by guest
Definition virtqueue:153
Descriptor in the descriptor table.
Definition virtqueue:93
l4_uint16_t next
Index of the next chained descriptor.
Definition virtqueue:117
l4_uint32_t len
Length of described buffer.
Definition virtqueue:115
Flags flags
Descriptor flags.
Definition virtqueue:116
void dump(unsigned idx) const
Dump a single descriptor.
Definition virtqueue:122
Ptr< void > addr
Address stored in descriptor.
Definition virtqueue:114
Used_elem ring[]
array of used descriptors.
Definition virtqueue:199
l4_uint16_t idx
index of the last entry in the ring.
Definition virtqueue:198
Flags flags
flags of the used ring.
Definition virtqueue:197
Low-level Virtqueue.
Definition virtqueue:87
void no_notify_host(bool value)
Set the no-notify flag for this queue.
Definition virtqueue:443
void disable()
Completely disable the queue.
Definition virtqueue:230
void setup(unsigned num, void *desc, void *avail, void *used)
Enable this queue.
Definition virtqueue:355
static unsigned long desc_size(unsigned num)
Calculate the size of the descriptor table for num entries.
Definition virtqueue:267
l4_uint16_t get_tail_avail_idx() const
Get tail-available index stored in local state (for debugging).
Definition virtqueue:463
Used * _used
pointer to used ring.
Definition virtqueue:205
bool no_notify_guest() const
Get the no IRQ flag of this queue.
Definition virtqueue:420
static unsigned long avail_align()
Get the alignment in zero LSBs needed for the available ring.
Definition virtqueue:293
void dump(Desc const *d) const
Dump descriptors for this queue.
Definition virtqueue:398
void setup_simple(unsigned num, void *ring)
Enable this queue.
Definition virtqueue:384
unsigned long avail_offset() const
Get the offset of the available ring from the descriptor table.
Definition virtqueue:329
static unsigned long used_align()
Get the alignment in zero LSBs needed for the used ring.
Definition virtqueue:312
static unsigned long total_size(unsigned num)
Calculate the total size for a virtqueue of the given dimensions.
Definition virtqueue:251
static unsigned long desc_align()
Get the alignment in zero LSBs needed for the descriptor table.
Definition virtqueue:275
static unsigned long used_size(unsigned num)
Calculate the size of the used ring for num entries.
Definition virtqueue:304
Virtqueue()=default
Create a disabled virtqueue.
unsigned long used_offset() const
Get the offset of the used ring from the descriptor table.
Definition virtqueue:335
static unsigned long avail_size(unsigned num)
Calculate the size of the available ring for num entries.
Definition virtqueue:285
unsigned long total_size() const
Calculate the total size of this virtqueue.
Definition virtqueue:320
bool ready() const
Test if this queue is in working state.
Definition virtqueue:406
l4_uint16_t _idx_mask
mask used for indexing into the descriptor table and the rings.
Definition virtqueue:214
Desc * _desc
pointer to descriptor table, NULL if queue is off.
Definition virtqueue:203
l4_uint16_t get_avail_idx() const
Get available index from available ring (for debugging).
Definition virtqueue:456
bool no_notify_host() const
Get the no notify flag of this queue.
Definition virtqueue:433
Avail * _avail
pointer to available ring.
Definition virtqueue:204
l4_uint16_t _current_avail
The life counter for the queue.
Definition virtqueue:208
unsigned num() const
Definition virtqueue:410
Error codes.
Base exceptions.
unsigned long l4_addr_t
Address type.
Definition l4int.h:34
unsigned int l4_uint32_t
Unsigned 32bit value.
Definition l4int.h:29
unsigned short int l4_uint16_t
Unsigned 16bit value.
Definition l4int.h:27
unsigned long long l4_uint64_t
Unsigned 64bit value.
Definition l4int.h:31
@ L4_EINVAL
Invalid argument.
Definition err.h:47
l4_addr_t l4_round_size(l4_addr_t value, unsigned char bits) L4_NOTHROW
Round value up to the next alignment with bits size.
Definition consts.h:503
#define L4_LIKELY(x)
Expression is likely to execute.
Definition compiler.h:294
Common L4 ABI Data Types.
L4-VIRTIO Transport C++ API.
Definition l4virtio:26
Flags of the available ring.
Definition virtqueue:141
constexpr no_irq_bfm_t::Val no_irq() const
Get the no_irq bits (0 to 0) of raw.
Definition virtqueue:149
Flags(l4_uint16_t v)
Make Flags from the raw value.
Definition virtqueue:146
l4_uint16_t raw
raw 16bit flags value of the available ring.
Definition virtqueue:142
Type for descriptor flags.
Definition virtqueue:99
constexpr next_bfm_t::Val next() const
Get the next bits (0 to 0) of raw.
Definition virtqueue:107
constexpr write_bfm_t::Val write() const
Get the write bits (1 to 1) of raw.
Definition virtqueue:109
Flags(l4_uint16_t v)
Make Flags from raw 16bit value.
Definition virtqueue:104
l4_uint16_t raw
raw flags value of a virtio descriptor.
Definition virtqueue:100
constexpr indirect_bfm_t::Val indirect() const
Get the indirect bits (2 to 2) of raw.
Definition virtqueue:111
flags for the used ring.
Definition virtqueue:186
constexpr no_notify_bfm_t::Val no_notify() const
Get the no_notify bits (0 to 0) of raw.
Definition virtqueue:194
l4_uint16_t raw
raw flags value as specified by virtio.
Definition virtqueue:187
Flags(l4_uint16_t v)
make Flags from raw value
Definition virtqueue:191
Type of an element of the used ring.
Definition virtqueue:161
l4_uint32_t id
descriptor index
Definition virtqueue:172
l4_uint32_t len
length field
Definition virtqueue:173
Used_elem(l4_uint16_t id, l4_uint32_t len)
Initialize a used ring element.
Definition virtqueue:171