|
|
1.1 root 1: #ifndef _VIRTIO_RING_H
2: #define _VIRTIO_RING_H
3:
4: #include "types.h" // u64
5: #include "memmap.h" // PAGE_SIZE
6:
7: #define PAGE_SHIFT 12
8: #define PAGE_MASK (PAGE_SIZE-1)
9:
10: #define virt_to_phys(v) (unsigned long)(v)
11: #define phys_to_virt(p) (void*)(p)
12: /* Compiler barrier is enough as an x86 CPU does not reorder reads or writes */
13: #define smp_rmb() barrier()
14: #define smp_wmb() barrier()
15:
16: /* Status byte for guest to report progress, and synchronize features. */
17: /* We have seen device and processed generic fields (VIRTIO_CONFIG_F_VIRTIO) */
18: #define VIRTIO_CONFIG_S_ACKNOWLEDGE 1
19: /* We have found a driver for the device. */
20: #define VIRTIO_CONFIG_S_DRIVER 2
21: /* Driver has used its parts of the config, and is happy */
22: #define VIRTIO_CONFIG_S_DRIVER_OK 4
23: /* We've given up on this device. */
24: #define VIRTIO_CONFIG_S_FAILED 0x80
25:
26: #define MAX_QUEUE_NUM (128)
27:
28: #define VRING_DESC_F_NEXT 1
29: #define VRING_DESC_F_WRITE 2
30:
31: #define VRING_AVAIL_F_NO_INTERRUPT 1
32:
33: #define VRING_USED_F_NO_NOTIFY 1
34:
35: struct vring_desc
36: {
37: u64 addr;
38: u32 len;
39: u16 flags;
40: u16 next;
41: };
42:
43: struct vring_avail
44: {
45: u16 flags;
46: u16 idx;
47: u16 ring[0];
48: };
49:
50: struct vring_used_elem
51: {
52: u32 id;
53: u32 len;
54: };
55:
56: struct vring_used
57: {
58: u16 flags;
59: u16 idx;
60: struct vring_used_elem ring[];
61: };
62:
63: struct vring {
64: unsigned int num;
65: struct vring_desc *desc;
66: struct vring_avail *avail;
67: struct vring_used *used;
68: };
69:
70: #define vring_size(num) \
71: (((((sizeof(struct vring_desc) * num) + \
72: (sizeof(struct vring_avail) + sizeof(u16) * num)) \
73: + PAGE_MASK) & ~PAGE_MASK) + \
74: (sizeof(struct vring_used) + sizeof(struct vring_used_elem) * num))
75:
76: typedef unsigned char virtio_queue_t[vring_size(MAX_QUEUE_NUM)];
77:
78: struct vring_virtqueue {
79: virtio_queue_t queue;
80: struct vring vring;
81: u16 free_head;
82: u16 last_used_idx;
83: u16 vdata[MAX_QUEUE_NUM];
84: /* PCI */
85: int queue_index;
86: };
87:
88: struct vring_list {
89: char *addr;
90: unsigned int length;
91: };
92:
93: static inline void vring_init(struct vring *vr,
94: unsigned int num, unsigned char *queue)
95: {
96: unsigned int i;
97: unsigned long pa;
98:
99: ASSERT32FLAT();
100: vr->num = num;
101:
102: /* physical address of desc must be page aligned */
103:
104: pa = virt_to_phys(queue);
105: pa = (pa + PAGE_MASK) & ~PAGE_MASK;
106: vr->desc = phys_to_virt(pa);
107:
108: vr->avail = (struct vring_avail *)&vr->desc[num];
109: /* disable interrupts */
110: vr->avail->flags |= VRING_AVAIL_F_NO_INTERRUPT;
111:
112: /* physical address of used must be page aligned */
113:
114: pa = virt_to_phys(&vr->avail->ring[num]);
115: pa = (pa + PAGE_MASK) & ~PAGE_MASK;
116: vr->used = phys_to_virt(pa);
117:
118: for (i = 0; i < num - 1; i++)
119: vr->desc[i].next = i + 1;
120: vr->desc[i].next = 0;
121: }
122:
123: int vring_more_used(struct vring_virtqueue *vq);
124: void vring_detach(struct vring_virtqueue *vq, unsigned int head);
125: int vring_get_buf(struct vring_virtqueue *vq, unsigned int *len);
126: void vring_add_buf(struct vring_virtqueue *vq, struct vring_list list[],
127: unsigned int out, unsigned int in,
128: int index, int num_added);
129: void vring_kick(unsigned int ioaddr, struct vring_virtqueue *vq, int num_added);
130:
131: #endif /* _VIRTIO_RING_H_ */
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.