diff options
author | Timos Ampelikiotis <t.ampelikiotis@virtualopensystems.com> | 2022-11-25 13:39:49 +0100 |
---|---|---|
committer | Angelos Mouzakitis <a.mouzakitis@virtualopensystems.com> | 2023-10-03 15:18:54 +0300 |
commit | e6d6621bced8b62dc9f98a09870b527535cbb158 (patch) | |
tree | a0779840e12602ad199e4ee45a7976e89e39b129 /virtio_loopback.c | |
parent | a3fcee5911bf760f9f4522e94cb9e6ab22a7eb95 (diff) |
Virtio-loopback-adapter Beta version:
- Build a single virtio-loopback-adapter binary
- Mmap the vrings with dynamic defined size
- Add set_mem_table mechanism (Not used into this release)
Signed-off-by: Timos Ampelikiotis <t.ampelikiotis@virtualopensystems.com>
Diffstat (limited to 'virtio_loopback.c')
-rw-r--r-- | virtio_loopback.c | 93 |
1 files changed, 83 insertions, 10 deletions
diff --git a/virtio_loopback.c b/virtio_loopback.c index 90458cc..b9fd353 100644 --- a/virtio_loopback.c +++ b/virtio_loopback.c @@ -71,10 +71,10 @@ /* Global variables */ +int s; /* To be deleted */ int efd; /* Eventfd file descriptor */ uint64_t eftd_ctr; fd_set rfds; -int s; int fd; int loopback_fd; @@ -146,6 +146,8 @@ int virtio_set_status(VirtIODevice *vdev, uint8_t val) virtio_set_started(vdev, val & VIRTIO_CONFIG_S_DRIVER_OK); } + DBG("set vdev->status :%u \n", vdev->status); + if (k->set_status) { DBG("k->set_status\n"); k->set_status(vdev, val); @@ -407,6 +409,8 @@ void virtio_set_isr(VirtIODevice *vdev, int value) if ((old & value) != value) { vdev->isr |= value; } + + DBG("Update isr: %d\n", vdev->isr); } static void virtio_irq(VirtQueue *vq) @@ -439,6 +443,7 @@ void virtio_notify(VirtIODevice *vdev, VirtQueue *vq) DBG("Do not notify!\n"); return; } + DBG("Go on and notify!\n"); virtio_irq(vq); } @@ -749,6 +754,44 @@ bool virtqueue_get_head(VirtQueue *vq, unsigned int idx, return true; } +uint32_t get_vqs_max_size(VirtIODevice *vdev) +{ + uint32_t vq_max_size = VIRTQUEUE_MAX_SIZE; + uint32_t total_size, temp_size, total_p2 = 1; + int i, log_res = 0; + + total_size = VIRTQUEUE_MAX_SIZE * sizeof(VRingDesc); + total_size += offsetof(VRingAvail, ring) + + VIRTQUEUE_MAX_SIZE * sizeof(uint16_t); + total_size += offsetof(VRingUsed, ring) + + VIRTQUEUE_MAX_SIZE * sizeof(uint16_t); + + temp_size = total_size; + + /* Compute log2 of total_size (Needs to be power of 2) */ + while ((temp_size /= 2) > 0) { + log_res++; + total_p2 *= 2; + } + + /* if total_size is not a power of 2: (total_size > 8) -> 16 */ + if (total_size > total_p2) { + total_size = 2 * total_p2; + } + + /* + * Align to page size: This needed only in case total_size + * is less than 4096 (PAGE_SIZE) + */ + if (total_size % PAGE_SIZE > 0) { + total_size = (total_size / PAGE_SIZE) * PAGE_SIZE + PAGE_SIZE; + } + + DBG("Total vqs size to mmap is: %u\n", total_size); + + return total_size; +} + int virtqueue_num_heads(VirtQueue *vq, unsigned int idx) { uint16_t num_heads = vring_avail_idx(vq) - idx; @@ -1037,6 +1080,8 @@ static void virtio_queue_guest_notifier_read(EventNotifier *n) int vhost_user_loopback_eventfd = 0; +int eventfd_count = 0; + void *loopback_event_select(void *wfd) { int retval; @@ -1066,7 +1111,10 @@ void *loopback_event_select(void *wfd) exit(1); } else { DBG("\n\nEvent has come from the vhost-user-device " - "(eventfd: %d)\n\n", *(int *)wfd); + "(eventfd: %d) -> event_count: %d\n\n", + *(int *)wfd, eventfd_count); + + eventfd_count++; virtio_irq(global_vdev->vq); } @@ -1243,6 +1291,17 @@ bool virtio_device_disabled(VirtIODevice *vdev) int prev_level = 0; +void *my_notify(void *data) { + + int irq_num = 44; + (void) data; + (void) ioctl(fd, IRQ, &irq_num); + + pthread_exit(NULL); +} + +int int_count = 0; + void virtio_loopback_update_irq(VirtIODevice *vdev) { int level, irq_num = 44; @@ -1254,6 +1313,9 @@ void virtio_loopback_update_irq(VirtIODevice *vdev) level = (vdev->isr != 0); + DBG("level: %d\n", level); + DBG("prev_level: %d\n", prev_level); + if (!((level == 1) && (prev_level == 0))) { prev_level = level; return; @@ -1261,9 +1323,13 @@ void virtio_loopback_update_irq(VirtIODevice *vdev) prev_level = level; DBG("Trigger interrupt (ioctl)\n"); - ioctl(fd, IRQ, &irq_num); + DBG("Interrupt counter: %d\n", int_count++); + + (void)pthread_create(&my_thread_id, NULL, my_notify, NULL); + } +bool enable_virtio_interrupt = false; /* virtio device */ void virtio_notify_vector(VirtIODevice *vdev) @@ -1296,15 +1362,17 @@ void virtio_queue_notify(VirtIODevice *vdev, int n) { VirtQueue *vq = &vdev->vq[n]; - DBG("virtio_queue_notify(...)\n"); + DBG("virtio_queue_notify(..., vq_n: %d)\n", n); if (!vq->vring.desc || vdev->broken) { + DBG("virtio_queue_notify: broken\n"); return; } if (vq->host_notifier_enabled) { event_notifier_set(&vq->host_notifier); } else if (vq->handle_output) { + DBG("vq->handle_output\n"); vq->handle_output(vdev, vq); if (vdev->start_on_kick) { @@ -1569,6 +1637,8 @@ static uint64_t virtio_loopback_read(VirtIODevice *vdev, uint64_t offset, return 0; } +uint64_t vring_phys_addrs[2] = {0}; +uint32_t vring_phys_addrs_idx = 0; void virtio_loopback_write(VirtIODevice *vdev, uint64_t offset, uint64_t value, unsigned size) @@ -1688,9 +1758,14 @@ void virtio_loopback_write(VirtIODevice *vdev, uint64_t offset, if (value == 0) { /* TODO: To be implemented */ } else { - (void)value; + + DBG("desc_addr: 0x%lx\n", value); + vring_phys_addrs[vring_phys_addrs_idx++] = value; + uint64_t desc_addr; - desc_addr = (uint64_t)mmap(NULL, 10 * PAGE_SIZE, + uint32_t vqs_size = get_vqs_max_size(global_vdev); + + desc_addr = (uint64_t)mmap(NULL, vqs_size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0); @@ -1733,6 +1808,8 @@ void virtio_loopback_write(VirtIODevice *vdev, uint64_t offset, virtio_set_status(vdev, value & 0xff); + DBG("STATUS -> %ld\n", value); + /* * TODO: Check if this is still needed * @@ -1843,7 +1920,6 @@ void *driver_event_select(void *data) (void) data; DBG("\nWaiting for loopback read/write events\n"); - fflush(stdout); FD_ZERO(&rfds); FD_SET(efd, &rfds); @@ -2021,9 +2097,6 @@ int virtio_loopback_start(void) exit(1); } - /* Fille the device info */ - create_rng_struct(); - /* Start loopback transport */ (void)ioctl(fd, START_LOOPBACK, &device_info); |