Blame SOURCES/kvm-io-Small-updates-in-preparation-for-websocket-change.patch

4a2fec
From 977ae916454f470780f7562b61c98718e2d0d49c Mon Sep 17 00:00:00 2001
4a2fec
From: "Daniel P. Berrange" <berrange@redhat.com>
4a2fec
Date: Wed, 20 Dec 2017 17:56:47 +0100
4a2fec
Subject: [PATCH 07/42] io: Small updates in preparation for websocket changes
4a2fec
4a2fec
RH-Author: Daniel P. Berrange <berrange@redhat.com>
4a2fec
Message-id: <20171220175702.29663-6-berrange@redhat.com>
4a2fec
Patchwork-id: 78459
4a2fec
O-Subject: [RHV-7.5 qemu-kvm-rhev PATCH v2 05/20] io: Small updates in preparation for websocket changes
4a2fec
Bugzilla: 1518649
4a2fec
RH-Acked-by: John Snow <jsnow@redhat.com>
4a2fec
RH-Acked-by: Jeffrey Cody <jcody@redhat.com>
4a2fec
RH-Acked-by: Miroslav Rezanina <mrezanin@redhat.com>
4a2fec
4a2fec
From: Brandon Carpenter <brandon.carpenter@cypherpath.com>
4a2fec
4a2fec
Gets rid of unnecessary bit shifting and performs proper EOF checking to
4a2fec
avoid a large number of repeated calls to recvmsg() when a client
4a2fec
abruptly terminates a connection (bug fix).
4a2fec
4a2fec
Signed-off-by: Brandon Carpenter <brandon.carpenter@cypherpath.com>
4a2fec
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
4a2fec
(cherry picked from commit eefa3d8ef649f9055611361e2201cca49f8c3433)
4a2fec
Signed-off-by: Miroslav Rezanina <mrezanin@redhat.com>
4a2fec
---
4a2fec
 io/channel-websock.c | 64 ++++++++++++++++------------------------------------
4a2fec
 1 file changed, 19 insertions(+), 45 deletions(-)
4a2fec
4a2fec
diff --git a/io/channel-websock.c b/io/channel-websock.c
4a2fec
index 2258557..4e5afb2 100644
4a2fec
--- a/io/channel-websock.c
4a2fec
+++ b/io/channel-websock.c
4a2fec
@@ -110,13 +110,11 @@
4a2fec
 /* Magic 7-bit length to indicate use of 64-bit payload length */
4a2fec
 #define QIO_CHANNEL_WEBSOCK_PAYLOAD_LEN_MAGIC_64_BIT 127
4a2fec
 
4a2fec
-/* Bitmasks & shifts for accessing header fields */
4a2fec
+/* Bitmasks for accessing header fields */
4a2fec
 #define QIO_CHANNEL_WEBSOCK_HEADER_FIELD_FIN 0x80
4a2fec
 #define QIO_CHANNEL_WEBSOCK_HEADER_FIELD_OPCODE 0x0f
4a2fec
 #define QIO_CHANNEL_WEBSOCK_HEADER_FIELD_HAS_MASK 0x80
4a2fec
 #define QIO_CHANNEL_WEBSOCK_HEADER_FIELD_PAYLOAD_LEN 0x7f
4a2fec
-#define QIO_CHANNEL_WEBSOCK_HEADER_SHIFT_FIN 7
4a2fec
-#define QIO_CHANNEL_WEBSOCK_HEADER_SHIFT_HAS_MASK 7
4a2fec
 
4a2fec
 typedef struct QIOChannelWebsockHeader QIOChannelWebsockHeader;
4a2fec
 
4a2fec
@@ -586,7 +584,7 @@ static void qio_channel_websock_encode(QIOChannelWebsock *ioc)
4a2fec
         return;
4a2fec
     }
4a2fec
 
4a2fec
-    header.ws.b0 = (1 << QIO_CHANNEL_WEBSOCK_HEADER_SHIFT_FIN) |
4a2fec
+    header.ws.b0 = QIO_CHANNEL_WEBSOCK_HEADER_FIELD_FIN |
4a2fec
         (QIO_CHANNEL_WEBSOCK_OPCODE_BINARY_FRAME &
4a2fec
          QIO_CHANNEL_WEBSOCK_HEADER_FIELD_OPCODE);
4a2fec
     if (ioc->rawoutput.offset <
4a2fec
@@ -613,8 +611,8 @@ static void qio_channel_websock_encode(QIOChannelWebsock *ioc)
4a2fec
 }
4a2fec
 
4a2fec
 
4a2fec
-static ssize_t qio_channel_websock_decode_header(QIOChannelWebsock *ioc,
4a2fec
-                                                 Error **errp)
4a2fec
+static int qio_channel_websock_decode_header(QIOChannelWebsock *ioc,
4a2fec
+                                             Error **errp)
4a2fec
 {
4a2fec
     unsigned char opcode, fin, has_mask;
4a2fec
     size_t header_size;
4a2fec
@@ -633,11 +631,9 @@ static ssize_t qio_channel_websock_decode_header(QIOChannelWebsock *ioc,
4a2fec
         return QIO_CHANNEL_ERR_BLOCK;
4a2fec
     }
4a2fec
 
4a2fec
-    fin = (header->b0 & QIO_CHANNEL_WEBSOCK_HEADER_FIELD_FIN) >>
4a2fec
-        QIO_CHANNEL_WEBSOCK_HEADER_SHIFT_FIN;
4a2fec
+    fin = header->b0 & QIO_CHANNEL_WEBSOCK_HEADER_FIELD_FIN;
4a2fec
     opcode = header->b0 & QIO_CHANNEL_WEBSOCK_HEADER_FIELD_OPCODE;
4a2fec
-    has_mask = (header->b1 & QIO_CHANNEL_WEBSOCK_HEADER_FIELD_HAS_MASK) >>
4a2fec
-        QIO_CHANNEL_WEBSOCK_HEADER_SHIFT_HAS_MASK;
4a2fec
+    has_mask = header->b1 & QIO_CHANNEL_WEBSOCK_HEADER_FIELD_HAS_MASK;
4a2fec
     payload_len = header->b1 & QIO_CHANNEL_WEBSOCK_HEADER_FIELD_PAYLOAD_LEN;
4a2fec
 
4a2fec
     if (opcode == QIO_CHANNEL_WEBSOCK_OPCODE_CLOSE) {
4a2fec
@@ -655,7 +651,7 @@ static ssize_t qio_channel_websock_decode_header(QIOChannelWebsock *ioc,
4a2fec
         return -1;
4a2fec
     }
4a2fec
     if (!has_mask) {
4a2fec
-        error_setg(errp, "websocket frames must be masked");
4a2fec
+        error_setg(errp, "client websocket frames must be masked");
4a2fec
         return -1;
4a2fec
     }
4a2fec
     if (opcode != QIO_CHANNEL_WEBSOCK_OPCODE_BINARY_FRAME) {
4a2fec
@@ -687,8 +683,8 @@ static ssize_t qio_channel_websock_decode_header(QIOChannelWebsock *ioc,
4a2fec
 }
4a2fec
 
4a2fec
 
4a2fec
-static ssize_t qio_channel_websock_decode_payload(QIOChannelWebsock *ioc,
4a2fec
-                                                  Error **errp)
4a2fec
+static int qio_channel_websock_decode_payload(QIOChannelWebsock *ioc,
4a2fec
+                                              Error **errp)
4a2fec
 {
4a2fec
     size_t i;
4a2fec
     size_t payload_len;
4a2fec
@@ -729,7 +725,7 @@ static ssize_t qio_channel_websock_decode_payload(QIOChannelWebsock *ioc,
4a2fec
     buffer_reserve(&ioc->rawinput, payload_len);
4a2fec
     buffer_append(&ioc->rawinput, ioc->encinput.buffer, payload_len);
4a2fec
     buffer_advance(&ioc->encinput, payload_len);
4a2fec
-    return payload_len;
4a2fec
+    return 0;
4a2fec
 }
4a2fec
 
4a2fec
 
4a2fec
@@ -809,8 +805,8 @@ static ssize_t qio_channel_websock_read_wire(QIOChannelWebsock *ioc,
4a2fec
         if (ret < 0) {
4a2fec
             return ret;
4a2fec
         }
4a2fec
-        if (ret == 0 &&
4a2fec
-            ioc->encinput.offset == 0) {
4a2fec
+        if (ret == 0 && ioc->encinput.offset == 0) {
4a2fec
+            ioc->io_eof = TRUE;
4a2fec
             return 0;
4a2fec
         }
4a2fec
         ioc->encinput.offset += ret;
4a2fec
@@ -822,10 +818,6 @@ static ssize_t qio_channel_websock_read_wire(QIOChannelWebsock *ioc,
4a2fec
             if (ret < 0) {
4a2fec
                 return ret;
4a2fec
             }
4a2fec
-            if (ret == 0) {
4a2fec
-                ioc->io_eof = TRUE;
4a2fec
-                break;
4a2fec
-            }
4a2fec
         }
4a2fec
 
4a2fec
         ret = qio_channel_websock_decode_payload(ioc, errp);
4a2fec
@@ -1090,14 +1082,12 @@ struct QIOChannelWebsockSource {
4a2fec
 };
4a2fec
 
4a2fec
 static gboolean
4a2fec
-qio_channel_websock_source_prepare(GSource *source,
4a2fec
-                                   gint *timeout)
4a2fec
+qio_channel_websock_source_check(GSource *source)
4a2fec
 {
4a2fec
     QIOChannelWebsockSource *wsource = (QIOChannelWebsockSource *)source;
4a2fec
     GIOCondition cond = 0;
4a2fec
-    *timeout = -1;
4a2fec
 
4a2fec
-    if (wsource->wioc->rawinput.offset) {
4a2fec
+    if (wsource->wioc->rawinput.offset || wsource->wioc->io_eof) {
4a2fec
         cond |= G_IO_IN;
4a2fec
     }
4a2fec
     if (wsource->wioc->rawoutput.offset < QIO_CHANNEL_WEBSOCK_MAX_BUFFER) {
4a2fec
@@ -1108,19 +1098,11 @@ qio_channel_websock_source_prepare(GSource *source,
4a2fec
 }
4a2fec
 
4a2fec
 static gboolean
4a2fec
-qio_channel_websock_source_check(GSource *source)
4a2fec
+qio_channel_websock_source_prepare(GSource *source,
4a2fec
+                                   gint *timeout)
4a2fec
 {
4a2fec
-    QIOChannelWebsockSource *wsource = (QIOChannelWebsockSource *)source;
4a2fec
-    GIOCondition cond = 0;
4a2fec
-
4a2fec
-    if (wsource->wioc->rawinput.offset) {
4a2fec
-        cond |= G_IO_IN;
4a2fec
-    }
4a2fec
-    if (wsource->wioc->rawoutput.offset < QIO_CHANNEL_WEBSOCK_MAX_BUFFER) {
4a2fec
-        cond |= G_IO_OUT;
4a2fec
-    }
4a2fec
-
4a2fec
-    return cond & wsource->condition;
4a2fec
+    *timeout = -1;
4a2fec
+    return qio_channel_websock_source_check(source);
4a2fec
 }
4a2fec
 
4a2fec
 static gboolean
4a2fec
@@ -1130,17 +1112,9 @@ qio_channel_websock_source_dispatch(GSource *source,
4a2fec
 {
4a2fec
     QIOChannelFunc func = (QIOChannelFunc)callback;
4a2fec
     QIOChannelWebsockSource *wsource = (QIOChannelWebsockSource *)source;
4a2fec
-    GIOCondition cond = 0;
4a2fec
-
4a2fec
-    if (wsource->wioc->rawinput.offset) {
4a2fec
-        cond |= G_IO_IN;
4a2fec
-    }
4a2fec
-    if (wsource->wioc->rawoutput.offset < QIO_CHANNEL_WEBSOCK_MAX_BUFFER) {
4a2fec
-        cond |= G_IO_OUT;
4a2fec
-    }
4a2fec
 
4a2fec
     return (*func)(QIO_CHANNEL(wsource->wioc),
4a2fec
-                   (cond & wsource->condition),
4a2fec
+                   qio_channel_websock_source_check(source),
4a2fec
                    user_data);
4a2fec
 }
4a2fec
 
4a2fec
-- 
4a2fec
1.8.3.1
4a2fec