summary refs log tree commit diff stats
diff options
context:
space:
mode:
authorMark McLoughlin <markmc@redhat.com>2009-04-29 13:30:24 +0100
committerMark McLoughlin <markmc@redhat.com>2009-06-09 11:38:50 +0100
commite19eb22486f258a421108ac22b8380a4e2f16b97 (patch)
treeeb4eb3fc252294521c32d1de75769b4939568d67
parentf3b6c7fcf8fca857b3c3ba0aa5b3a06d7ce0ac37 (diff)
downloadfocaccia-qemu-e19eb22486f258a421108ac22b8380a4e2f16b97.tar.gz
focaccia-qemu-e19eb22486f258a421108ac22b8380a4e2f16b97.zip
net: make use of async packet sending API in tap client
If a packet is queued by qemu_send_packet(), remove I/O
handler for the tap fd until we get notification that the
packet has been sent.

A not insignificant side effect of this is we can now
drain the tap send queue in one go without fear of packets
being dropped.

Signed-off-by: Mark McLoughlin <markmc@redhat.com>
-rw-r--r--net.c24
1 files changed, 20 insertions, 4 deletions
diff --git a/net.c b/net.c
index e7f5138343..4cf27be99c 100644
--- a/net.c
+++ b/net.c
@@ -1079,15 +1079,31 @@ static ssize_t tap_read_packet(int tapfd, uint8_t *buf, int maxlen)
 }
 #endif
 
+static void tap_send(void *opaque);
+
+static void tap_send_completed(VLANClientState *vc)
+{
+    TAPState *s = vc->opaque;
+
+    qemu_set_fd_handler2(s->fd, tap_can_send, tap_send, NULL, s);
+}
+
 static void tap_send(void *opaque)
 {
     TAPState *s = opaque;
     int size;
 
-    size = tap_read_packet(s->fd, s->buf, sizeof(s->buf));
-    if (size > 0) {
-        qemu_send_packet(s->vc, s->buf, size);
-    }
+    do {
+        size = tap_read_packet(s->fd, s->buf, sizeof(s->buf));
+        if (size <= 0) {
+            break;
+        }
+
+        size = qemu_send_packet_async(s->vc, s->buf, size, tap_send_completed);
+        if (size == 0) {
+            qemu_set_fd_handler2(s->fd, NULL, NULL, NULL, NULL);
+        }
+    } while (size > 0);
 }
 
 static void tap_cleanup(VLANClientState *vc)