xref: /openbmc/qemu/io/channel-websock.c (revision 5aa62804)
1 /*
2  * QEMU I/O channels driver websockets
3  *
4  * Copyright (c) 2015 Red Hat, Inc.
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with this library; if not, see <http://www.gnu.org/licenses/>.
18  *
19  */
20 
21 #include "qemu/osdep.h"
22 #include "qapi/error.h"
23 #include "qemu/bswap.h"
24 #include "io/channel-websock.h"
25 #include "crypto/hash.h"
26 #include "trace.h"
27 #include "qemu/iov.h"
28 #include "qemu/module.h"
29 
30 /* Max amount to allow in rawinput/encoutput buffers */
31 #define QIO_CHANNEL_WEBSOCK_MAX_BUFFER 8192
32 
33 #define QIO_CHANNEL_WEBSOCK_CLIENT_KEY_LEN 24
34 #define QIO_CHANNEL_WEBSOCK_GUID "258EAFA5-E914-47DA-95CA-C5AB0DC85B11"
35 #define QIO_CHANNEL_WEBSOCK_GUID_LEN strlen(QIO_CHANNEL_WEBSOCK_GUID)
36 
37 #define QIO_CHANNEL_WEBSOCK_HEADER_PROTOCOL "sec-websocket-protocol"
38 #define QIO_CHANNEL_WEBSOCK_HEADER_VERSION "sec-websocket-version"
39 #define QIO_CHANNEL_WEBSOCK_HEADER_KEY "sec-websocket-key"
40 #define QIO_CHANNEL_WEBSOCK_HEADER_UPGRADE "upgrade"
41 #define QIO_CHANNEL_WEBSOCK_HEADER_HOST "host"
42 #define QIO_CHANNEL_WEBSOCK_HEADER_CONNECTION "connection"
43 
44 #define QIO_CHANNEL_WEBSOCK_PROTOCOL_BINARY "binary"
45 #define QIO_CHANNEL_WEBSOCK_CONNECTION_UPGRADE "Upgrade"
46 #define QIO_CHANNEL_WEBSOCK_UPGRADE_WEBSOCKET "websocket"
47 
48 #define QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_COMMON \
49     "Server: QEMU VNC\r\n"                       \
50     "Date: %s\r\n"
51 
52 #define QIO_CHANNEL_WEBSOCK_HANDSHAKE_WITH_PROTO_RES_OK \
53     "HTTP/1.1 101 Switching Protocols\r\n"              \
54     QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_COMMON            \
55     "Upgrade: websocket\r\n"                            \
56     "Connection: Upgrade\r\n"                           \
57     "Sec-WebSocket-Accept: %s\r\n"                      \
58     "Sec-WebSocket-Protocol: binary\r\n"                \
59     "\r\n"
60 #define QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_OK    \
61     "HTTP/1.1 101 Switching Protocols\r\n"      \
62     QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_COMMON    \
63     "Upgrade: websocket\r\n"                    \
64     "Connection: Upgrade\r\n"                   \
65     "Sec-WebSocket-Accept: %s\r\n"              \
66     "\r\n"
67 #define QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_NOT_FOUND \
68     "HTTP/1.1 404 Not Found\r\n"                    \
69     QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_COMMON        \
70     "Connection: close\r\n"                         \
71     "\r\n"
72 #define QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_BAD_REQUEST \
73     "HTTP/1.1 400 Bad Request\r\n"                    \
74     QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_COMMON          \
75     "Connection: close\r\n"                           \
76     "Sec-WebSocket-Version: "                         \
77     QIO_CHANNEL_WEBSOCK_SUPPORTED_VERSION             \
78     "\r\n"
79 #define QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_SERVER_ERR \
80     "HTTP/1.1 500 Internal Server Error\r\n"         \
81     QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_COMMON         \
82     "Connection: close\r\n"                          \
83     "\r\n"
84 #define QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_TOO_LARGE  \
85     "HTTP/1.1 403 Request Entity Too Large\r\n"      \
86     QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_COMMON         \
87     "Connection: close\r\n"                          \
88     "\r\n"
89 #define QIO_CHANNEL_WEBSOCK_HANDSHAKE_DELIM "\r\n"
90 #define QIO_CHANNEL_WEBSOCK_HANDSHAKE_END "\r\n\r\n"
91 #define QIO_CHANNEL_WEBSOCK_SUPPORTED_VERSION "13"
92 #define QIO_CHANNEL_WEBSOCK_HTTP_METHOD "GET"
93 #define QIO_CHANNEL_WEBSOCK_HTTP_PATH "/"
94 #define QIO_CHANNEL_WEBSOCK_HTTP_VERSION "HTTP/1.1"
95 
96 /* The websockets packet header is variable length
97  * depending on the size of the payload... */
98 
99 /* ...length when using 7-bit payload length */
100 #define QIO_CHANNEL_WEBSOCK_HEADER_LEN_7_BIT 6
101 /* ...length when using 16-bit payload length */
102 #define QIO_CHANNEL_WEBSOCK_HEADER_LEN_16_BIT 8
103 /* ...length when using 64-bit payload length */
104 #define QIO_CHANNEL_WEBSOCK_HEADER_LEN_64_BIT 14
105 
106 /* Length of the optional data mask field in header */
107 #define QIO_CHANNEL_WEBSOCK_HEADER_LEN_MASK 4
108 
109 /* Maximum length that can fit in 7-bit payload size */
110 #define QIO_CHANNEL_WEBSOCK_PAYLOAD_LEN_THRESHOLD_7_BIT 126
111 /* Maximum length that can fit in 16-bit payload size */
112 #define QIO_CHANNEL_WEBSOCK_PAYLOAD_LEN_THRESHOLD_16_BIT 65536
113 
114 /* Magic 7-bit length to indicate use of 16-bit payload length */
115 #define QIO_CHANNEL_WEBSOCK_PAYLOAD_LEN_MAGIC_16_BIT 126
116 /* Magic 7-bit length to indicate use of 64-bit payload length */
117 #define QIO_CHANNEL_WEBSOCK_PAYLOAD_LEN_MAGIC_64_BIT 127
118 
119 /* Bitmasks for accessing header fields */
120 #define QIO_CHANNEL_WEBSOCK_HEADER_FIELD_FIN 0x80
121 #define QIO_CHANNEL_WEBSOCK_HEADER_FIELD_OPCODE 0x0f
122 #define QIO_CHANNEL_WEBSOCK_HEADER_FIELD_HAS_MASK 0x80
123 #define QIO_CHANNEL_WEBSOCK_HEADER_FIELD_PAYLOAD_LEN 0x7f
124 #define QIO_CHANNEL_WEBSOCK_CONTROL_OPCODE_MASK 0x8
125 
126 typedef struct QIOChannelWebsockHeader QIOChannelWebsockHeader;
127 
128 struct QEMU_PACKED QIOChannelWebsockHeader {
129     unsigned char b0;
130     unsigned char b1;
131     union {
132         struct QEMU_PACKED {
133             uint16_t l16;
134             QIOChannelWebsockMask m16;
135         } s16;
136         struct QEMU_PACKED {
137             uint64_t l64;
138             QIOChannelWebsockMask m64;
139         } s64;
140         QIOChannelWebsockMask m;
141     } u;
142 };
143 
144 typedef struct QIOChannelWebsockHTTPHeader QIOChannelWebsockHTTPHeader;
145 
146 struct QIOChannelWebsockHTTPHeader {
147     char *name;
148     char *value;
149 };
150 
151 enum {
152     QIO_CHANNEL_WEBSOCK_OPCODE_CONTINUATION = 0x0,
153     QIO_CHANNEL_WEBSOCK_OPCODE_TEXT_FRAME = 0x1,
154     QIO_CHANNEL_WEBSOCK_OPCODE_BINARY_FRAME = 0x2,
155     QIO_CHANNEL_WEBSOCK_OPCODE_CLOSE = 0x8,
156     QIO_CHANNEL_WEBSOCK_OPCODE_PING = 0x9,
157     QIO_CHANNEL_WEBSOCK_OPCODE_PONG = 0xA
158 };
159 
160 static void GCC_FMT_ATTR(2, 3)
161 qio_channel_websock_handshake_send_res(QIOChannelWebsock *ioc,
162                                        const char *resmsg,
163                                        ...)
164 {
165     va_list vargs;
166     char *response;
167     size_t responselen;
168 
169     va_start(vargs, resmsg);
170     response = g_strdup_vprintf(resmsg, vargs);
171     responselen = strlen(response);
172     buffer_reserve(&ioc->encoutput, responselen);
173     buffer_append(&ioc->encoutput, response, responselen);
174     g_free(response);
175     va_end(vargs);
176 }
177 
178 static gchar *qio_channel_websock_date_str(void)
179 {
180     struct tm tm;
181     time_t now = time(NULL);
182     char datebuf[128];
183 
184     gmtime_r(&now, &tm);
185 
186     strftime(datebuf, sizeof(datebuf), "%a, %d %b %Y %H:%M:%S GMT", &tm);
187 
188     return g_strdup(datebuf);
189 }
190 
191 static void qio_channel_websock_handshake_send_res_err(QIOChannelWebsock *ioc,
192                                                        const char *resdata)
193 {
194     char *date = qio_channel_websock_date_str();
195     qio_channel_websock_handshake_send_res(ioc, resdata, date);
196     g_free(date);
197 }
198 
199 enum {
200     QIO_CHANNEL_WEBSOCK_STATUS_NORMAL = 1000,
201     QIO_CHANNEL_WEBSOCK_STATUS_PROTOCOL_ERR = 1002,
202     QIO_CHANNEL_WEBSOCK_STATUS_INVALID_DATA = 1003,
203     QIO_CHANNEL_WEBSOCK_STATUS_POLICY = 1008,
204     QIO_CHANNEL_WEBSOCK_STATUS_TOO_LARGE = 1009,
205     QIO_CHANNEL_WEBSOCK_STATUS_SERVER_ERR = 1011,
206 };
207 
208 static size_t
209 qio_channel_websock_extract_headers(QIOChannelWebsock *ioc,
210                                     char *buffer,
211                                     QIOChannelWebsockHTTPHeader *hdrs,
212                                     size_t nhdrsalloc,
213                                     Error **errp)
214 {
215     char *nl, *sep, *tmp;
216     size_t nhdrs = 0;
217 
218     /*
219      * First parse the HTTP protocol greeting of format:
220      *
221      *   $METHOD $PATH $VERSION
222      *
223      * e.g.
224      *
225      *   GET / HTTP/1.1
226      */
227 
228     nl = strstr(buffer, QIO_CHANNEL_WEBSOCK_HANDSHAKE_DELIM);
229     if (!nl) {
230         error_setg(errp, "Missing HTTP header delimiter");
231         goto bad_request;
232     }
233     *nl = '\0';
234     trace_qio_channel_websock_http_greeting(ioc, buffer);
235 
236     tmp = strchr(buffer, ' ');
237     if (!tmp) {
238         error_setg(errp, "Missing HTTP path delimiter");
239         return 0;
240     }
241     *tmp = '\0';
242 
243     if (!g_str_equal(buffer, QIO_CHANNEL_WEBSOCK_HTTP_METHOD)) {
244         error_setg(errp, "Unsupported HTTP method %s", buffer);
245         goto bad_request;
246     }
247 
248     buffer = tmp + 1;
249     tmp = strchr(buffer, ' ');
250     if (!tmp) {
251         error_setg(errp, "Missing HTTP version delimiter");
252         goto bad_request;
253     }
254     *tmp = '\0';
255 
256     if (!g_str_equal(buffer, QIO_CHANNEL_WEBSOCK_HTTP_PATH)) {
257         qio_channel_websock_handshake_send_res_err(
258             ioc, QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_NOT_FOUND);
259         error_setg(errp, "Unexpected HTTP path %s", buffer);
260         return 0;
261     }
262 
263     buffer = tmp + 1;
264 
265     if (!g_str_equal(buffer, QIO_CHANNEL_WEBSOCK_HTTP_VERSION)) {
266         error_setg(errp, "Unsupported HTTP version %s", buffer);
267         goto bad_request;
268     }
269 
270     buffer = nl + strlen(QIO_CHANNEL_WEBSOCK_HANDSHAKE_DELIM);
271 
272     /*
273      * Now parse all the header fields of format
274      *
275      *   $NAME: $VALUE
276      *
277      * e.g.
278      *
279      *   Cache-control: no-cache
280      */
281     do {
282         QIOChannelWebsockHTTPHeader *hdr;
283 
284         nl = strstr(buffer, QIO_CHANNEL_WEBSOCK_HANDSHAKE_DELIM);
285         if (nl) {
286             *nl = '\0';
287         }
288 
289         sep = strchr(buffer, ':');
290         if (!sep) {
291             error_setg(errp, "Malformed HTTP header");
292             goto bad_request;
293         }
294         *sep = '\0';
295         sep++;
296         while (*sep == ' ') {
297             sep++;
298         }
299 
300         if (nhdrs >= nhdrsalloc) {
301             error_setg(errp, "Too many HTTP headers");
302             goto bad_request;
303         }
304 
305         hdr = &hdrs[nhdrs++];
306         hdr->name = buffer;
307         hdr->value = sep;
308 
309         /* Canonicalize header name for easier identification later */
310         for (tmp = hdr->name; *tmp; tmp++) {
311             *tmp = g_ascii_tolower(*tmp);
312         }
313 
314         if (nl) {
315             buffer = nl + strlen(QIO_CHANNEL_WEBSOCK_HANDSHAKE_DELIM);
316         }
317     } while (nl != NULL);
318 
319     return nhdrs;
320 
321  bad_request:
322     qio_channel_websock_handshake_send_res_err(
323         ioc, QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_BAD_REQUEST);
324     return 0;
325 }
326 
327 static const char *
328 qio_channel_websock_find_header(QIOChannelWebsockHTTPHeader *hdrs,
329                                 size_t nhdrs,
330                                 const char *name)
331 {
332     size_t i;
333 
334     for (i = 0; i < nhdrs; i++) {
335         if (g_str_equal(hdrs[i].name, name)) {
336             return hdrs[i].value;
337         }
338     }
339 
340     return NULL;
341 }
342 
343 
344 static void qio_channel_websock_handshake_send_res_ok(QIOChannelWebsock *ioc,
345                                                       const char *key,
346                                                       const bool use_protocols,
347                                                       Error **errp)
348 {
349     char combined_key[QIO_CHANNEL_WEBSOCK_CLIENT_KEY_LEN +
350                       QIO_CHANNEL_WEBSOCK_GUID_LEN + 1];
351     char *accept = NULL;
352     char *date = NULL;
353 
354     g_strlcpy(combined_key, key, QIO_CHANNEL_WEBSOCK_CLIENT_KEY_LEN + 1);
355     g_strlcat(combined_key, QIO_CHANNEL_WEBSOCK_GUID,
356               QIO_CHANNEL_WEBSOCK_CLIENT_KEY_LEN +
357               QIO_CHANNEL_WEBSOCK_GUID_LEN + 1);
358 
359     /* hash and encode it */
360     if (qcrypto_hash_base64(QCRYPTO_HASH_ALG_SHA1,
361                             combined_key,
362                             QIO_CHANNEL_WEBSOCK_CLIENT_KEY_LEN +
363                             QIO_CHANNEL_WEBSOCK_GUID_LEN,
364                             &accept,
365                             errp) < 0) {
366         qio_channel_websock_handshake_send_res_err(
367             ioc, QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_SERVER_ERR);
368         return;
369     }
370 
371     date = qio_channel_websock_date_str();
372     if (use_protocols) {
373             qio_channel_websock_handshake_send_res(
374                 ioc, QIO_CHANNEL_WEBSOCK_HANDSHAKE_WITH_PROTO_RES_OK,
375                 date, accept);
376     } else {
377             qio_channel_websock_handshake_send_res(
378                 ioc, QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_OK, date, accept);
379     }
380 
381     g_free(date);
382     g_free(accept);
383 }
384 
385 static void qio_channel_websock_handshake_process(QIOChannelWebsock *ioc,
386                                                   char *buffer,
387                                                   Error **errp)
388 {
389     QIOChannelWebsockHTTPHeader hdrs[32];
390     size_t nhdrs = G_N_ELEMENTS(hdrs);
391     const char *protocols = NULL, *version = NULL, *key = NULL,
392         *host = NULL, *connection = NULL, *upgrade = NULL;
393     char **connectionv;
394     bool upgraded = false;
395     size_t i;
396 
397     nhdrs = qio_channel_websock_extract_headers(ioc, buffer, hdrs, nhdrs, errp);
398     if (!nhdrs) {
399         return;
400     }
401 
402     protocols = qio_channel_websock_find_header(
403         hdrs, nhdrs, QIO_CHANNEL_WEBSOCK_HEADER_PROTOCOL);
404 
405     version = qio_channel_websock_find_header(
406         hdrs, nhdrs, QIO_CHANNEL_WEBSOCK_HEADER_VERSION);
407     if (!version) {
408         error_setg(errp, "Missing websocket version header data");
409         goto bad_request;
410     }
411 
412     key = qio_channel_websock_find_header(
413         hdrs, nhdrs, QIO_CHANNEL_WEBSOCK_HEADER_KEY);
414     if (!key) {
415         error_setg(errp, "Missing websocket key header data");
416         goto bad_request;
417     }
418 
419     host = qio_channel_websock_find_header(
420         hdrs, nhdrs, QIO_CHANNEL_WEBSOCK_HEADER_HOST);
421     if (!host) {
422         error_setg(errp, "Missing websocket host header data");
423         goto bad_request;
424     }
425 
426     connection = qio_channel_websock_find_header(
427         hdrs, nhdrs, QIO_CHANNEL_WEBSOCK_HEADER_CONNECTION);
428     if (!connection) {
429         error_setg(errp, "Missing websocket connection header data");
430         goto bad_request;
431     }
432 
433     upgrade = qio_channel_websock_find_header(
434         hdrs, nhdrs, QIO_CHANNEL_WEBSOCK_HEADER_UPGRADE);
435     if (!upgrade) {
436         error_setg(errp, "Missing websocket upgrade header data");
437         goto bad_request;
438     }
439 
440     trace_qio_channel_websock_http_request(ioc, protocols, version,
441                                            host, connection, upgrade, key);
442 
443     if (protocols) {
444             if (!g_strrstr(protocols, QIO_CHANNEL_WEBSOCK_PROTOCOL_BINARY)) {
445                 error_setg(errp, "No '%s' protocol is supported by client '%s'",
446                            QIO_CHANNEL_WEBSOCK_PROTOCOL_BINARY, protocols);
447                 goto bad_request;
448             }
449     }
450 
451     if (!g_str_equal(version, QIO_CHANNEL_WEBSOCK_SUPPORTED_VERSION)) {
452         error_setg(errp, "Version '%s' is not supported by client '%s'",
453                    QIO_CHANNEL_WEBSOCK_SUPPORTED_VERSION, version);
454         goto bad_request;
455     }
456 
457     if (strlen(key) != QIO_CHANNEL_WEBSOCK_CLIENT_KEY_LEN) {
458         error_setg(errp, "Key length '%zu' was not as expected '%d'",
459                    strlen(key), QIO_CHANNEL_WEBSOCK_CLIENT_KEY_LEN);
460         goto bad_request;
461     }
462 
463     connectionv = g_strsplit(connection, ",", 0);
464     for (i = 0; connectionv != NULL && connectionv[i] != NULL; i++) {
465         g_strstrip(connectionv[i]);
466         if (strcasecmp(connectionv[i],
467                        QIO_CHANNEL_WEBSOCK_CONNECTION_UPGRADE) == 0) {
468             upgraded = true;
469         }
470     }
471     g_strfreev(connectionv);
472     if (!upgraded) {
473         error_setg(errp, "No connection upgrade requested '%s'", connection);
474         goto bad_request;
475     }
476 
477     if (strcasecmp(upgrade, QIO_CHANNEL_WEBSOCK_UPGRADE_WEBSOCKET) != 0) {
478         error_setg(errp, "Incorrect upgrade method '%s'", upgrade);
479         goto bad_request;
480     }
481 
482     qio_channel_websock_handshake_send_res_ok(ioc, key, !!protocols, errp);
483     return;
484 
485  bad_request:
486     qio_channel_websock_handshake_send_res_err(
487         ioc, QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_BAD_REQUEST);
488 }
489 
490 static int qio_channel_websock_handshake_read(QIOChannelWebsock *ioc,
491                                               Error **errp)
492 {
493     char *handshake_end;
494     ssize_t ret;
495     /* Typical HTTP headers from novnc are 512 bytes, so limiting
496      * total header size to 4096 is easily enough. */
497     size_t want = 4096 - ioc->encinput.offset;
498     buffer_reserve(&ioc->encinput, want);
499     ret = qio_channel_read(ioc->master,
500                            (char *)buffer_end(&ioc->encinput), want, errp);
501     if (ret < 0) {
502         return -1;
503     }
504     ioc->encinput.offset += ret;
505 
506     handshake_end = g_strstr_len((char *)ioc->encinput.buffer,
507                                  ioc->encinput.offset,
508                                  QIO_CHANNEL_WEBSOCK_HANDSHAKE_END);
509     if (!handshake_end) {
510         if (ioc->encinput.offset >= 4096) {
511             qio_channel_websock_handshake_send_res_err(
512                 ioc, QIO_CHANNEL_WEBSOCK_HANDSHAKE_RES_TOO_LARGE);
513             error_setg(errp,
514                        "End of headers not found in first 4096 bytes");
515             return 1;
516         } else if (ret == 0) {
517             error_setg(errp,
518                        "End of headers not found before connection closed");
519             return -1;
520         }
521         return 0;
522     }
523     *handshake_end = '\0';
524 
525     qio_channel_websock_handshake_process(ioc,
526                                           (char *)ioc->encinput.buffer,
527                                           errp);
528 
529     buffer_advance(&ioc->encinput,
530                    handshake_end - (char *)ioc->encinput.buffer +
531                    strlen(QIO_CHANNEL_WEBSOCK_HANDSHAKE_END));
532     return 1;
533 }
534 
535 static gboolean qio_channel_websock_handshake_send(QIOChannel *ioc,
536                                                    GIOCondition condition,
537                                                    gpointer user_data)
538 {
539     QIOTask *task = user_data;
540     QIOChannelWebsock *wioc = QIO_CHANNEL_WEBSOCK(
541         qio_task_get_source(task));
542     Error *err = NULL;
543     ssize_t ret;
544 
545     ret = qio_channel_write(wioc->master,
546                             (char *)wioc->encoutput.buffer,
547                             wioc->encoutput.offset,
548                             &err);
549 
550     if (ret < 0) {
551         trace_qio_channel_websock_handshake_fail(ioc, error_get_pretty(err));
552         qio_task_set_error(task, err);
553         qio_task_complete(task);
554         return FALSE;
555     }
556 
557     buffer_advance(&wioc->encoutput, ret);
558     if (wioc->encoutput.offset == 0) {
559         if (wioc->io_err) {
560             trace_qio_channel_websock_handshake_fail(
561                 ioc, error_get_pretty(wioc->io_err));
562             qio_task_set_error(task, wioc->io_err);
563             wioc->io_err = NULL;
564             qio_task_complete(task);
565         } else {
566             trace_qio_channel_websock_handshake_complete(ioc);
567             qio_task_complete(task);
568         }
569         return FALSE;
570     }
571     trace_qio_channel_websock_handshake_pending(ioc, G_IO_OUT);
572     return TRUE;
573 }
574 
575 static gboolean qio_channel_websock_handshake_io(QIOChannel *ioc,
576                                                  GIOCondition condition,
577                                                  gpointer user_data)
578 {
579     QIOTask *task = user_data;
580     QIOChannelWebsock *wioc = QIO_CHANNEL_WEBSOCK(
581         qio_task_get_source(task));
582     Error *err = NULL;
583     int ret;
584 
585     ret = qio_channel_websock_handshake_read(wioc, &err);
586     if (ret < 0) {
587         /*
588          * We only take this path on a fatal I/O error reading from
589          * client connection, as most of the time we have an
590          * HTTP 4xx err response to send instead
591          */
592         trace_qio_channel_websock_handshake_fail(ioc, error_get_pretty(err));
593         qio_task_set_error(task, err);
594         qio_task_complete(task);
595         return FALSE;
596     }
597     if (ret == 0) {
598         trace_qio_channel_websock_handshake_pending(ioc, G_IO_IN);
599         /* need more data still */
600         return TRUE;
601     }
602 
603     error_propagate(&wioc->io_err, err);
604 
605     trace_qio_channel_websock_handshake_reply(ioc);
606     qio_channel_add_watch(
607         wioc->master,
608         G_IO_OUT,
609         qio_channel_websock_handshake_send,
610         task,
611         NULL);
612     return FALSE;
613 }
614 
615 
616 static void qio_channel_websock_encode(QIOChannelWebsock *ioc,
617                                        uint8_t opcode,
618                                        const struct iovec *iov,
619                                        size_t niov,
620                                        size_t size)
621 {
622     size_t header_size;
623     size_t i;
624     union {
625         char buf[QIO_CHANNEL_WEBSOCK_HEADER_LEN_64_BIT];
626         QIOChannelWebsockHeader ws;
627     } header;
628 
629     assert(size <= iov_size(iov, niov));
630 
631     header.ws.b0 = QIO_CHANNEL_WEBSOCK_HEADER_FIELD_FIN |
632         (opcode & QIO_CHANNEL_WEBSOCK_HEADER_FIELD_OPCODE);
633     if (size < QIO_CHANNEL_WEBSOCK_PAYLOAD_LEN_THRESHOLD_7_BIT) {
634         header.ws.b1 = (uint8_t)size;
635         header_size = QIO_CHANNEL_WEBSOCK_HEADER_LEN_7_BIT;
636     } else if (size < QIO_CHANNEL_WEBSOCK_PAYLOAD_LEN_THRESHOLD_16_BIT) {
637         header.ws.b1 = QIO_CHANNEL_WEBSOCK_PAYLOAD_LEN_MAGIC_16_BIT;
638         header.ws.u.s16.l16 = cpu_to_be16((uint16_t)size);
639         header_size = QIO_CHANNEL_WEBSOCK_HEADER_LEN_16_BIT;
640     } else {
641         header.ws.b1 = QIO_CHANNEL_WEBSOCK_PAYLOAD_LEN_MAGIC_64_BIT;
642         header.ws.u.s64.l64 = cpu_to_be64(size);
643         header_size = QIO_CHANNEL_WEBSOCK_HEADER_LEN_64_BIT;
644     }
645     header_size -= QIO_CHANNEL_WEBSOCK_HEADER_LEN_MASK;
646 
647     trace_qio_channel_websock_encode(ioc, opcode, header_size, size);
648     buffer_reserve(&ioc->encoutput, header_size + size);
649     buffer_append(&ioc->encoutput, header.buf, header_size);
650     for (i = 0; i < niov && size != 0; i++) {
651         size_t want = iov[i].iov_len;
652         if (want > size) {
653             want = size;
654         }
655         buffer_append(&ioc->encoutput, iov[i].iov_base, want);
656         size -= want;
657     }
658 }
659 
660 
661 static ssize_t qio_channel_websock_write_wire(QIOChannelWebsock *, Error **);
662 
663 
664 static void qio_channel_websock_write_close(QIOChannelWebsock *ioc,
665                                             uint16_t code, const char *reason)
666 {
667     struct iovec iov[2] = {
668         { .iov_base = &code, .iov_len = sizeof(code) },
669     };
670     size_t niov = 1;
671     size_t size = iov[0].iov_len;
672 
673     cpu_to_be16s(&code);
674 
675     if (reason) {
676         iov[1].iov_base = (void *)reason;
677         iov[1].iov_len = strlen(reason);
678         size += iov[1].iov_len;
679         niov++;
680     }
681     qio_channel_websock_encode(ioc, QIO_CHANNEL_WEBSOCK_OPCODE_CLOSE,
682                                iov, niov, size);
683     qio_channel_websock_write_wire(ioc, NULL);
684     qio_channel_shutdown(ioc->master, QIO_CHANNEL_SHUTDOWN_BOTH, NULL);
685 }
686 
687 
688 static int qio_channel_websock_decode_header(QIOChannelWebsock *ioc,
689                                              Error **errp)
690 {
691     unsigned char opcode, fin, has_mask;
692     size_t header_size;
693     size_t payload_len;
694     QIOChannelWebsockHeader *header =
695         (QIOChannelWebsockHeader *)ioc->encinput.buffer;
696 
697     if (ioc->payload_remain) {
698         error_setg(errp,
699                    "Decoding header but %zu bytes of payload remain",
700                    ioc->payload_remain);
701         qio_channel_websock_write_close(
702             ioc, QIO_CHANNEL_WEBSOCK_STATUS_SERVER_ERR,
703             "internal server error");
704         return -1;
705     }
706     if (ioc->encinput.offset < QIO_CHANNEL_WEBSOCK_HEADER_LEN_7_BIT) {
707         /* header not complete */
708         return QIO_CHANNEL_ERR_BLOCK;
709     }
710 
711     fin = header->b0 & QIO_CHANNEL_WEBSOCK_HEADER_FIELD_FIN;
712     opcode = header->b0 & QIO_CHANNEL_WEBSOCK_HEADER_FIELD_OPCODE;
713     has_mask = header->b1 & QIO_CHANNEL_WEBSOCK_HEADER_FIELD_HAS_MASK;
714     payload_len = header->b1 & QIO_CHANNEL_WEBSOCK_HEADER_FIELD_PAYLOAD_LEN;
715 
716     /* Save or restore opcode. */
717     if (opcode) {
718         ioc->opcode = opcode;
719     } else {
720         opcode = ioc->opcode;
721     }
722 
723     trace_qio_channel_websock_header_partial_decode(ioc, payload_len,
724                                                     fin, opcode, (int)has_mask);
725 
726     if (opcode == QIO_CHANNEL_WEBSOCK_OPCODE_CLOSE) {
727         /* disconnect */
728         return 0;
729     }
730 
731     /* Websocket frame sanity check:
732      * * Fragmentation is only supported for binary frames.
733      * * All frames sent by a client MUST be masked.
734      * * Only binary and ping/pong encoding is supported.
735      */
736     if (!fin) {
737         if (opcode != QIO_CHANNEL_WEBSOCK_OPCODE_BINARY_FRAME) {
738             error_setg(errp, "only binary websocket frames may be fragmented");
739             qio_channel_websock_write_close(
740                 ioc, QIO_CHANNEL_WEBSOCK_STATUS_POLICY ,
741                 "only binary frames may be fragmented");
742             return -1;
743         }
744     } else {
745         if (opcode != QIO_CHANNEL_WEBSOCK_OPCODE_BINARY_FRAME &&
746             opcode != QIO_CHANNEL_WEBSOCK_OPCODE_CLOSE &&
747             opcode != QIO_CHANNEL_WEBSOCK_OPCODE_PING &&
748             opcode != QIO_CHANNEL_WEBSOCK_OPCODE_PONG) {
749             error_setg(errp, "unsupported opcode: %#04x; only binary, close, "
750                        "ping, and pong websocket frames are supported", opcode);
751             qio_channel_websock_write_close(
752                 ioc, QIO_CHANNEL_WEBSOCK_STATUS_INVALID_DATA ,
753                 "only binary, close, ping, and pong frames are supported");
754             return -1;
755         }
756     }
757     if (!has_mask) {
758         error_setg(errp, "client websocket frames must be masked");
759         qio_channel_websock_write_close(
760             ioc, QIO_CHANNEL_WEBSOCK_STATUS_PROTOCOL_ERR,
761             "client frames must be masked");
762         return -1;
763     }
764 
765     if (payload_len < QIO_CHANNEL_WEBSOCK_PAYLOAD_LEN_MAGIC_16_BIT) {
766         ioc->payload_remain = payload_len;
767         header_size = QIO_CHANNEL_WEBSOCK_HEADER_LEN_7_BIT;
768         ioc->mask = header->u.m;
769     } else if (opcode & QIO_CHANNEL_WEBSOCK_CONTROL_OPCODE_MASK) {
770         error_setg(errp, "websocket control frame is too large");
771         qio_channel_websock_write_close(
772             ioc, QIO_CHANNEL_WEBSOCK_STATUS_PROTOCOL_ERR,
773             "control frame is too large");
774         return -1;
775     } else if (payload_len == QIO_CHANNEL_WEBSOCK_PAYLOAD_LEN_MAGIC_16_BIT &&
776                ioc->encinput.offset >= QIO_CHANNEL_WEBSOCK_HEADER_LEN_16_BIT) {
777         ioc->payload_remain = be16_to_cpu(header->u.s16.l16);
778         header_size = QIO_CHANNEL_WEBSOCK_HEADER_LEN_16_BIT;
779         ioc->mask = header->u.s16.m16;
780     } else if (payload_len == QIO_CHANNEL_WEBSOCK_PAYLOAD_LEN_MAGIC_64_BIT &&
781                ioc->encinput.offset >= QIO_CHANNEL_WEBSOCK_HEADER_LEN_64_BIT) {
782         ioc->payload_remain = be64_to_cpu(header->u.s64.l64);
783         header_size = QIO_CHANNEL_WEBSOCK_HEADER_LEN_64_BIT;
784         ioc->mask = header->u.s64.m64;
785     } else {
786         /* header not complete */
787         return QIO_CHANNEL_ERR_BLOCK;
788     }
789 
790     trace_qio_channel_websock_header_full_decode(
791         ioc, header_size, ioc->payload_remain, ioc->mask.u);
792     buffer_advance(&ioc->encinput, header_size);
793     return 0;
794 }
795 
796 
797 static int qio_channel_websock_decode_payload(QIOChannelWebsock *ioc,
798                                               Error **errp)
799 {
800     size_t i;
801     size_t payload_len = 0;
802     uint32_t *payload32;
803 
804     if (ioc->payload_remain) {
805         /* If we aren't at the end of the payload, then drop
806          * off the last bytes, so we're always multiple of 4
807          * for purpose of unmasking, except at end of payload
808          */
809         if (ioc->encinput.offset < ioc->payload_remain) {
810             /* Wait for the entire payload before processing control frames
811              * because the payload will most likely be echoed back. */
812             if (ioc->opcode & QIO_CHANNEL_WEBSOCK_CONTROL_OPCODE_MASK) {
813                 return QIO_CHANNEL_ERR_BLOCK;
814             }
815             payload_len = ioc->encinput.offset - (ioc->encinput.offset % 4);
816         } else {
817             payload_len = ioc->payload_remain;
818         }
819         if (payload_len == 0) {
820             return QIO_CHANNEL_ERR_BLOCK;
821         }
822 
823         ioc->payload_remain -= payload_len;
824 
825         /* unmask frame */
826         /* process 1 frame (32 bit op) */
827         payload32 = (uint32_t *)ioc->encinput.buffer;
828         for (i = 0; i < payload_len / 4; i++) {
829             payload32[i] ^= ioc->mask.u;
830         }
831         /* process the remaining bytes (if any) */
832         for (i *= 4; i < payload_len; i++) {
833             ioc->encinput.buffer[i] ^= ioc->mask.c[i % 4];
834         }
835     }
836 
837     trace_qio_channel_websock_payload_decode(
838         ioc, ioc->opcode, ioc->payload_remain);
839 
840     if (ioc->opcode == QIO_CHANNEL_WEBSOCK_OPCODE_BINARY_FRAME) {
841         if (payload_len) {
842             /* binary frames are passed on */
843             buffer_reserve(&ioc->rawinput, payload_len);
844             buffer_append(&ioc->rawinput, ioc->encinput.buffer, payload_len);
845         }
846     } else if (ioc->opcode == QIO_CHANNEL_WEBSOCK_OPCODE_CLOSE) {
847         /* close frames are echoed back */
848         error_setg(errp, "websocket closed by peer");
849         if (payload_len) {
850             /* echo client status */
851             struct iovec iov = { .iov_base = ioc->encinput.buffer,
852                                  .iov_len = ioc->encinput.offset };
853             qio_channel_websock_encode(ioc, QIO_CHANNEL_WEBSOCK_OPCODE_CLOSE,
854                                        &iov, 1, iov.iov_len);
855             qio_channel_websock_write_wire(ioc, NULL);
856             qio_channel_shutdown(ioc->master, QIO_CHANNEL_SHUTDOWN_BOTH, NULL);
857         } else {
858             /* send our own status */
859             qio_channel_websock_write_close(
860                 ioc, QIO_CHANNEL_WEBSOCK_STATUS_NORMAL, "peer requested close");
861         }
862         return -1;
863     } else if (ioc->opcode == QIO_CHANNEL_WEBSOCK_OPCODE_PING) {
864         /* ping frames produce an immediate reply, as long as we've not still
865          * got a previous pong queued, in which case we drop the new pong */
866         if (ioc->pong_remain == 0) {
867             struct iovec iov = { .iov_base = ioc->encinput.buffer,
868                                  .iov_len = ioc->encinput.offset };
869             qio_channel_websock_encode(ioc, QIO_CHANNEL_WEBSOCK_OPCODE_PONG,
870                                        &iov, 1, iov.iov_len);
871             ioc->pong_remain = ioc->encoutput.offset;
872         }
873     }   /* pong frames are ignored */
874 
875     if (payload_len) {
876         buffer_advance(&ioc->encinput, payload_len);
877     }
878     return 0;
879 }
880 
881 
882 QIOChannelWebsock *
883 qio_channel_websock_new_server(QIOChannel *master)
884 {
885     QIOChannelWebsock *wioc;
886     QIOChannel *ioc;
887 
888     wioc = QIO_CHANNEL_WEBSOCK(object_new(TYPE_QIO_CHANNEL_WEBSOCK));
889     ioc = QIO_CHANNEL(wioc);
890 
891     wioc->master = master;
892     if (qio_channel_has_feature(master, QIO_CHANNEL_FEATURE_SHUTDOWN)) {
893         qio_channel_set_feature(ioc, QIO_CHANNEL_FEATURE_SHUTDOWN);
894     }
895     object_ref(OBJECT(master));
896 
897     trace_qio_channel_websock_new_server(wioc, master);
898     return wioc;
899 }
900 
901 void qio_channel_websock_handshake(QIOChannelWebsock *ioc,
902                                    QIOTaskFunc func,
903                                    gpointer opaque,
904                                    GDestroyNotify destroy)
905 {
906     QIOTask *task;
907 
908     task = qio_task_new(OBJECT(ioc),
909                         func,
910                         opaque,
911                         destroy);
912 
913     trace_qio_channel_websock_handshake_start(ioc);
914     trace_qio_channel_websock_handshake_pending(ioc, G_IO_IN);
915     qio_channel_add_watch(ioc->master,
916                           G_IO_IN,
917                           qio_channel_websock_handshake_io,
918                           task,
919                           NULL);
920 }
921 
922 
923 static void qio_channel_websock_finalize(Object *obj)
924 {
925     QIOChannelWebsock *ioc = QIO_CHANNEL_WEBSOCK(obj);
926 
927     buffer_free(&ioc->encinput);
928     buffer_free(&ioc->encoutput);
929     buffer_free(&ioc->rawinput);
930     object_unref(OBJECT(ioc->master));
931     if (ioc->io_tag) {
932         g_source_remove(ioc->io_tag);
933     }
934     if (ioc->io_err) {
935         error_free(ioc->io_err);
936     }
937 }
938 
939 
940 static ssize_t qio_channel_websock_read_wire(QIOChannelWebsock *ioc,
941                                              Error **errp)
942 {
943     ssize_t ret;
944 
945     if (ioc->encinput.offset < 4096) {
946         size_t want = 4096 - ioc->encinput.offset;
947 
948         buffer_reserve(&ioc->encinput, want);
949         ret = qio_channel_read(ioc->master,
950                                (char *)ioc->encinput.buffer +
951                                ioc->encinput.offset,
952                                want,
953                                errp);
954         if (ret < 0) {
955             return ret;
956         }
957         if (ret == 0 && ioc->encinput.offset == 0) {
958             ioc->io_eof = TRUE;
959             return 0;
960         }
961         ioc->encinput.offset += ret;
962     }
963 
964     while (ioc->encinput.offset != 0) {
965         if (ioc->payload_remain == 0) {
966             ret = qio_channel_websock_decode_header(ioc, errp);
967             if (ret < 0) {
968                 return ret;
969             }
970         }
971 
972         ret = qio_channel_websock_decode_payload(ioc, errp);
973         if (ret < 0) {
974             return ret;
975         }
976     }
977     return 1;
978 }
979 
980 
981 static ssize_t qio_channel_websock_write_wire(QIOChannelWebsock *ioc,
982                                               Error **errp)
983 {
984     ssize_t ret;
985     ssize_t done = 0;
986 
987     while (ioc->encoutput.offset > 0) {
988         ret = qio_channel_write(ioc->master,
989                                 (char *)ioc->encoutput.buffer,
990                                 ioc->encoutput.offset,
991                                 errp);
992         if (ret < 0) {
993             if (ret == QIO_CHANNEL_ERR_BLOCK &&
994                 done > 0) {
995                 return done;
996             } else {
997                 return ret;
998             }
999         }
1000         buffer_advance(&ioc->encoutput, ret);
1001         done += ret;
1002         if (ioc->pong_remain < ret) {
1003             ioc->pong_remain = 0;
1004         } else {
1005             ioc->pong_remain -= ret;
1006         }
1007     }
1008     return done;
1009 }
1010 
1011 
1012 static void qio_channel_websock_flush_free(gpointer user_data)
1013 {
1014     QIOChannelWebsock *wioc = QIO_CHANNEL_WEBSOCK(user_data);
1015     object_unref(OBJECT(wioc));
1016 }
1017 
1018 static void qio_channel_websock_set_watch(QIOChannelWebsock *ioc);
1019 
1020 static gboolean qio_channel_websock_flush(QIOChannel *ioc,
1021                                           GIOCondition condition,
1022                                           gpointer user_data)
1023 {
1024     QIOChannelWebsock *wioc = QIO_CHANNEL_WEBSOCK(user_data);
1025     ssize_t ret;
1026 
1027     if (condition & G_IO_OUT) {
1028         ret = qio_channel_websock_write_wire(wioc, &wioc->io_err);
1029         if (ret < 0) {
1030             goto cleanup;
1031         }
1032     }
1033 
1034     if (condition & G_IO_IN) {
1035         ret = qio_channel_websock_read_wire(wioc, &wioc->io_err);
1036         if (ret < 0) {
1037             goto cleanup;
1038         }
1039     }
1040 
1041  cleanup:
1042     qio_channel_websock_set_watch(wioc);
1043     return FALSE;
1044 }
1045 
1046 
1047 static void qio_channel_websock_unset_watch(QIOChannelWebsock *ioc)
1048 {
1049     if (ioc->io_tag) {
1050         g_source_remove(ioc->io_tag);
1051         ioc->io_tag = 0;
1052     }
1053 }
1054 
1055 static void qio_channel_websock_set_watch(QIOChannelWebsock *ioc)
1056 {
1057     GIOCondition cond = 0;
1058 
1059     qio_channel_websock_unset_watch(ioc);
1060 
1061     if (ioc->io_err) {
1062         return;
1063     }
1064 
1065     if (ioc->encoutput.offset) {
1066         cond |= G_IO_OUT;
1067     }
1068     if (ioc->encinput.offset < QIO_CHANNEL_WEBSOCK_MAX_BUFFER &&
1069         !ioc->io_eof) {
1070         cond |= G_IO_IN;
1071     }
1072 
1073     if (cond) {
1074         object_ref(OBJECT(ioc));
1075         ioc->io_tag =
1076             qio_channel_add_watch(ioc->master,
1077                                   cond,
1078                                   qio_channel_websock_flush,
1079                                   ioc,
1080                                   qio_channel_websock_flush_free);
1081     }
1082 }
1083 
1084 
1085 static ssize_t qio_channel_websock_readv(QIOChannel *ioc,
1086                                          const struct iovec *iov,
1087                                          size_t niov,
1088                                          int **fds,
1089                                          size_t *nfds,
1090                                          Error **errp)
1091 {
1092     QIOChannelWebsock *wioc = QIO_CHANNEL_WEBSOCK(ioc);
1093     size_t i;
1094     ssize_t got = 0;
1095     ssize_t ret;
1096 
1097     if (wioc->io_err) {
1098         error_propagate(errp, error_copy(wioc->io_err));
1099         return -1;
1100     }
1101 
1102     if (!wioc->rawinput.offset) {
1103         ret = qio_channel_websock_read_wire(QIO_CHANNEL_WEBSOCK(ioc), errp);
1104         if (ret < 0) {
1105             return ret;
1106         }
1107     }
1108 
1109     for (i = 0 ; i < niov ; i++) {
1110         size_t want = iov[i].iov_len;
1111         if (want > (wioc->rawinput.offset - got)) {
1112             want = (wioc->rawinput.offset - got);
1113         }
1114 
1115         memcpy(iov[i].iov_base,
1116                wioc->rawinput.buffer + got,
1117                want);
1118         got += want;
1119 
1120         if (want < iov[i].iov_len) {
1121             break;
1122         }
1123     }
1124 
1125     buffer_advance(&wioc->rawinput, got);
1126     qio_channel_websock_set_watch(wioc);
1127     return got;
1128 }
1129 
1130 
1131 static ssize_t qio_channel_websock_writev(QIOChannel *ioc,
1132                                           const struct iovec *iov,
1133                                           size_t niov,
1134                                           int *fds,
1135                                           size_t nfds,
1136                                           Error **errp)
1137 {
1138     QIOChannelWebsock *wioc = QIO_CHANNEL_WEBSOCK(ioc);
1139     ssize_t want = iov_size(iov, niov);
1140     ssize_t avail;
1141     ssize_t ret;
1142 
1143     if (wioc->io_err) {
1144         error_propagate(errp, error_copy(wioc->io_err));
1145         return -1;
1146     }
1147 
1148     if (wioc->io_eof) {
1149         error_setg(errp, "%s", "Broken pipe");
1150         return -1;
1151     }
1152 
1153     avail = wioc->encoutput.offset >= QIO_CHANNEL_WEBSOCK_MAX_BUFFER ?
1154         0 : (QIO_CHANNEL_WEBSOCK_MAX_BUFFER - wioc->encoutput.offset);
1155     if (want > avail) {
1156         want = avail;
1157     }
1158 
1159     if (want) {
1160         qio_channel_websock_encode(wioc,
1161                                    QIO_CHANNEL_WEBSOCK_OPCODE_BINARY_FRAME,
1162                                    iov, niov, want);
1163     }
1164 
1165     /* Even if want == 0, we'll try write_wire in case there's
1166      * pending data we could usefully flush out
1167      */
1168     ret = qio_channel_websock_write_wire(wioc, errp);
1169     if (ret < 0 &&
1170         ret != QIO_CHANNEL_ERR_BLOCK) {
1171         qio_channel_websock_unset_watch(wioc);
1172         return -1;
1173     }
1174 
1175     qio_channel_websock_set_watch(wioc);
1176 
1177     if (want == 0) {
1178         return QIO_CHANNEL_ERR_BLOCK;
1179     }
1180 
1181     return want;
1182 }
1183 
1184 static int qio_channel_websock_set_blocking(QIOChannel *ioc,
1185                                             bool enabled,
1186                                             Error **errp)
1187 {
1188     QIOChannelWebsock *wioc = QIO_CHANNEL_WEBSOCK(ioc);
1189 
1190     qio_channel_set_blocking(wioc->master, enabled, errp);
1191     return 0;
1192 }
1193 
1194 static void qio_channel_websock_set_delay(QIOChannel *ioc,
1195                                           bool enabled)
1196 {
1197     QIOChannelWebsock *tioc = QIO_CHANNEL_WEBSOCK(ioc);
1198 
1199     qio_channel_set_delay(tioc->master, enabled);
1200 }
1201 
1202 static void qio_channel_websock_set_cork(QIOChannel *ioc,
1203                                          bool enabled)
1204 {
1205     QIOChannelWebsock *tioc = QIO_CHANNEL_WEBSOCK(ioc);
1206 
1207     qio_channel_set_cork(tioc->master, enabled);
1208 }
1209 
1210 static int qio_channel_websock_shutdown(QIOChannel *ioc,
1211                                         QIOChannelShutdown how,
1212                                         Error **errp)
1213 {
1214     QIOChannelWebsock *tioc = QIO_CHANNEL_WEBSOCK(ioc);
1215 
1216     return qio_channel_shutdown(tioc->master, how, errp);
1217 }
1218 
1219 static int qio_channel_websock_close(QIOChannel *ioc,
1220                                      Error **errp)
1221 {
1222     QIOChannelWebsock *wioc = QIO_CHANNEL_WEBSOCK(ioc);
1223 
1224     trace_qio_channel_websock_close(ioc);
1225     return qio_channel_close(wioc->master, errp);
1226 }
1227 
1228 typedef struct QIOChannelWebsockSource QIOChannelWebsockSource;
1229 struct QIOChannelWebsockSource {
1230     GSource parent;
1231     QIOChannelWebsock *wioc;
1232     GIOCondition condition;
1233 };
1234 
1235 static gboolean
1236 qio_channel_websock_source_check(GSource *source)
1237 {
1238     QIOChannelWebsockSource *wsource = (QIOChannelWebsockSource *)source;
1239     GIOCondition cond = 0;
1240 
1241     if (wsource->wioc->rawinput.offset) {
1242         cond |= G_IO_IN;
1243     }
1244     if (wsource->wioc->encoutput.offset < QIO_CHANNEL_WEBSOCK_MAX_BUFFER) {
1245         cond |= G_IO_OUT;
1246     }
1247     if (wsource->wioc->io_eof) {
1248         cond |= G_IO_HUP;
1249     }
1250     if (wsource->wioc->io_err) {
1251         cond |= G_IO_ERR;
1252     }
1253 
1254     return cond & wsource->condition;
1255 }
1256 
1257 static gboolean
1258 qio_channel_websock_source_prepare(GSource *source,
1259                                    gint *timeout)
1260 {
1261     *timeout = -1;
1262     return qio_channel_websock_source_check(source);
1263 }
1264 
1265 static gboolean
1266 qio_channel_websock_source_dispatch(GSource *source,
1267                                     GSourceFunc callback,
1268                                     gpointer user_data)
1269 {
1270     QIOChannelFunc func = (QIOChannelFunc)callback;
1271     QIOChannelWebsockSource *wsource = (QIOChannelWebsockSource *)source;
1272 
1273     return (*func)(QIO_CHANNEL(wsource->wioc),
1274                    qio_channel_websock_source_check(source),
1275                    user_data);
1276 }
1277 
1278 static void
1279 qio_channel_websock_source_finalize(GSource *source)
1280 {
1281     QIOChannelWebsockSource *ssource = (QIOChannelWebsockSource *)source;
1282 
1283     object_unref(OBJECT(ssource->wioc));
1284 }
1285 
1286 GSourceFuncs qio_channel_websock_source_funcs = {
1287     qio_channel_websock_source_prepare,
1288     qio_channel_websock_source_check,
1289     qio_channel_websock_source_dispatch,
1290     qio_channel_websock_source_finalize
1291 };
1292 
1293 static GSource *qio_channel_websock_create_watch(QIOChannel *ioc,
1294                                                  GIOCondition condition)
1295 {
1296     QIOChannelWebsock *wioc = QIO_CHANNEL_WEBSOCK(ioc);
1297     QIOChannelWebsockSource *ssource;
1298     GSource *source;
1299 
1300     source = g_source_new(&qio_channel_websock_source_funcs,
1301                           sizeof(QIOChannelWebsockSource));
1302     ssource = (QIOChannelWebsockSource *)source;
1303 
1304     ssource->wioc = wioc;
1305     object_ref(OBJECT(wioc));
1306 
1307     ssource->condition = condition;
1308 
1309     qio_channel_websock_set_watch(wioc);
1310     return source;
1311 }
1312 
1313 static void qio_channel_websock_class_init(ObjectClass *klass,
1314                                            void *class_data G_GNUC_UNUSED)
1315 {
1316     QIOChannelClass *ioc_klass = QIO_CHANNEL_CLASS(klass);
1317 
1318     ioc_klass->io_writev = qio_channel_websock_writev;
1319     ioc_klass->io_readv = qio_channel_websock_readv;
1320     ioc_klass->io_set_blocking = qio_channel_websock_set_blocking;
1321     ioc_klass->io_set_cork = qio_channel_websock_set_cork;
1322     ioc_klass->io_set_delay = qio_channel_websock_set_delay;
1323     ioc_klass->io_close = qio_channel_websock_close;
1324     ioc_klass->io_shutdown = qio_channel_websock_shutdown;
1325     ioc_klass->io_create_watch = qio_channel_websock_create_watch;
1326 }
1327 
1328 static const TypeInfo qio_channel_websock_info = {
1329     .parent = TYPE_QIO_CHANNEL,
1330     .name = TYPE_QIO_CHANNEL_WEBSOCK,
1331     .instance_size = sizeof(QIOChannelWebsock),
1332     .instance_finalize = qio_channel_websock_finalize,
1333     .class_init = qio_channel_websock_class_init,
1334 };
1335 
1336 static void qio_channel_websock_register_types(void)
1337 {
1338     type_register_static(&qio_channel_websock_info);
1339 }
1340 
1341 type_init(qio_channel_websock_register_types);
1342