17536ee4bSTim Hardeck /*
27536ee4bSTim Hardeck * QEMU VNC display driver: Websockets support
37536ee4bSTim Hardeck *
47536ee4bSTim Hardeck * Copyright (C) 2010 Joel Martin
57536ee4bSTim Hardeck * Copyright (C) 2012 Tim Hardeck
67536ee4bSTim Hardeck *
77536ee4bSTim Hardeck * This is free software; you can redistribute it and/or modify
87536ee4bSTim Hardeck * it under the terms of the GNU General Public License as published by
97536ee4bSTim Hardeck * the Free Software Foundation; either version 2 of the License, or
107536ee4bSTim Hardeck * (at your option) any later version.
117536ee4bSTim Hardeck *
127536ee4bSTim Hardeck * This software is distributed in the hope that it will be useful,
137536ee4bSTim Hardeck * but WITHOUT ANY WARRANTY; without even the implied warranty of
147536ee4bSTim Hardeck * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
157536ee4bSTim Hardeck * GNU General Public License for more details.
167536ee4bSTim Hardeck *
177536ee4bSTim Hardeck * You should have received a copy of the GNU General Public License
187536ee4bSTim Hardeck * along with this software; if not, see <http://www.gnu.org/licenses/>.
197536ee4bSTim Hardeck */
207536ee4bSTim Hardeck
21e16f4c87SPeter Maydell #include "qemu/osdep.h"
22da34e65cSMarkus Armbruster #include "qapi/error.h"
237536ee4bSTim Hardeck #include "vnc.h"
24d5f04223SDaniel P. Berrange #include "io/channel-websock.h"
2558369e22SPaolo Bonzini #include "qemu/bswap.h"
26ad6374c4SDaniel P. Berrange #include "trace.h"
272cc45228SDaniel P. Berrange
vncws_tls_handshake_done(QIOTask * task,gpointer user_data)2860e705c5SDaniel P. Berrange static void vncws_tls_handshake_done(QIOTask *task,
292cc45228SDaniel P. Berrange gpointer user_data)
300057a0d5STim Hardeck {
312cc45228SDaniel P. Berrange VncState *vs = user_data;
3260e705c5SDaniel P. Berrange Error *err = NULL;
330057a0d5STim Hardeck
3460e705c5SDaniel P. Berrange if (qio_task_propagate_error(task, &err)) {
353e305e4aSDaniel P. Berrange VNC_DEBUG("Handshake failed %s\n", error_get_pretty(err));
363e305e4aSDaniel P. Berrange vnc_client_error(vs);
3760e705c5SDaniel P. Berrange error_free(err);
382cc45228SDaniel P. Berrange } else {
39d5f04223SDaniel P. Berrange VNC_DEBUG("TLS handshake complete, starting websocket handshake\n");
40a75d6f07SBrandon Carpenter if (vs->ioc_tag) {
41a75d6f07SBrandon Carpenter g_source_remove(vs->ioc_tag);
42a75d6f07SBrandon Carpenter }
43*7d5b0d68SPhilippe Mathieu-Daudé vs->ioc_tag = qio_channel_add_watch(vs->ioc,
44*7d5b0d68SPhilippe Mathieu-Daudé G_IO_IN | G_IO_HUP | G_IO_ERR,
452ddafce7SDing Hui vncws_handshake_io, vs, NULL);
460057a0d5STim Hardeck }
472cc45228SDaniel P. Berrange }
482cc45228SDaniel P. Berrange
490057a0d5STim Hardeck
vncws_tls_handshake_io(QIOChannel * ioc G_GNUC_UNUSED,GIOCondition condition,void * opaque)5004d2529dSDaniel P. Berrange gboolean vncws_tls_handshake_io(QIOChannel *ioc G_GNUC_UNUSED,
512ddafce7SDing Hui GIOCondition condition,
5204d2529dSDaniel P. Berrange void *opaque)
530057a0d5STim Hardeck {
542cc45228SDaniel P. Berrange VncState *vs = opaque;
552cc45228SDaniel P. Berrange QIOChannelTLS *tls;
563e305e4aSDaniel P. Berrange Error *err = NULL;
570057a0d5STim Hardeck
582cc45228SDaniel P. Berrange if (vs->ioc_tag) {
592cc45228SDaniel P. Berrange g_source_remove(vs->ioc_tag);
602cc45228SDaniel P. Berrange vs->ioc_tag = 0;
612cc45228SDaniel P. Berrange }
622cc45228SDaniel P. Berrange
632ddafce7SDing Hui if (condition & (G_IO_HUP | G_IO_ERR)) {
642ddafce7SDing Hui vnc_client_error(vs);
652ddafce7SDing Hui return TRUE;
662ddafce7SDing Hui }
672ddafce7SDing Hui
682cc45228SDaniel P. Berrange tls = qio_channel_tls_new_server(
692cc45228SDaniel P. Berrange vs->ioc,
702cc45228SDaniel P. Berrange vs->vd->tlscreds,
71b76806d4SDaniel P. Berrange vs->vd->tlsauthzid,
723e305e4aSDaniel P. Berrange &err);
732cc45228SDaniel P. Berrange if (!tls) {
742cc45228SDaniel P. Berrange VNC_DEBUG("Failed to setup TLS %s\n", error_get_pretty(err));
753e305e4aSDaniel P. Berrange error_free(err);
763e305e4aSDaniel P. Berrange vnc_client_error(vs);
7704d2529dSDaniel P. Berrange return TRUE;
7851941e46SDaniel P. Berrange }
793e305e4aSDaniel P. Berrange
8010bcfe58SDaniel P. Berrange qio_channel_set_name(QIO_CHANNEL(tls), "vnc-ws-server-tls");
8110bcfe58SDaniel P. Berrange
822cc45228SDaniel P. Berrange object_unref(OBJECT(vs->ioc));
832cc45228SDaniel P. Berrange vs->ioc = QIO_CHANNEL(tls);
84ad6374c4SDaniel P. Berrange trace_vnc_client_io_wrap(vs, vs->ioc, "tls");
852cc45228SDaniel P. Berrange vs->tls = qio_channel_tls_get_session(tls);
862cc45228SDaniel P. Berrange
872cc45228SDaniel P. Berrange qio_channel_tls_handshake(tls,
882cc45228SDaniel P. Berrange vncws_tls_handshake_done,
892cc45228SDaniel P. Berrange vs,
901939ccdaSPeter Xu NULL,
912cc45228SDaniel P. Berrange NULL);
922cc45228SDaniel P. Berrange
9304d2529dSDaniel P. Berrange return TRUE;
940057a0d5STim Hardeck }
950057a0d5STim Hardeck
96d5f04223SDaniel P. Berrange
vncws_handshake_done(QIOTask * task,gpointer user_data)9760e705c5SDaniel P. Berrange static void vncws_handshake_done(QIOTask *task,
98d5f04223SDaniel P. Berrange gpointer user_data)
997536ee4bSTim Hardeck {
100d5f04223SDaniel P. Berrange VncState *vs = user_data;
10160e705c5SDaniel P. Berrange Error *err = NULL;
1027536ee4bSTim Hardeck
10360e705c5SDaniel P. Berrange if (qio_task_propagate_error(task, &err)) {
104d5f04223SDaniel P. Berrange VNC_DEBUG("Websock handshake failed %s\n", error_get_pretty(err));
105d5f04223SDaniel P. Berrange vnc_client_error(vs);
10660e705c5SDaniel P. Berrange error_free(err);
107d5f04223SDaniel P. Berrange } else {
108d5f04223SDaniel P. Berrange VNC_DEBUG("Websock handshake complete, starting VNC protocol\n");
109dbee9897SDaniel P. Berrange vnc_start_protocol(vs);
110a75d6f07SBrandon Carpenter if (vs->ioc_tag) {
111a75d6f07SBrandon Carpenter g_source_remove(vs->ioc_tag);
112a75d6f07SBrandon Carpenter }
11304d2529dSDaniel P. Berrange vs->ioc_tag = qio_channel_add_watch(
1142ddafce7SDing Hui vs->ioc, G_IO_IN | G_IO_HUP | G_IO_ERR,
1152ddafce7SDing Hui vnc_client_io, vs, NULL);
1167536ee4bSTim Hardeck }
1177536ee4bSTim Hardeck }
1187536ee4bSTim Hardeck
1197536ee4bSTim Hardeck
vncws_handshake_io(QIOChannel * ioc G_GNUC_UNUSED,GIOCondition condition,void * opaque)12004d2529dSDaniel P. Berrange gboolean vncws_handshake_io(QIOChannel *ioc G_GNUC_UNUSED,
1212ddafce7SDing Hui GIOCondition condition,
12204d2529dSDaniel P. Berrange void *opaque)
12304d2529dSDaniel P. Berrange {
12404d2529dSDaniel P. Berrange VncState *vs = opaque;
125d5f04223SDaniel P. Berrange QIOChannelWebsock *wioc;
12604d2529dSDaniel P. Berrange
12704d2529dSDaniel P. Berrange if (vs->ioc_tag) {
12804d2529dSDaniel P. Berrange g_source_remove(vs->ioc_tag);
129d5f04223SDaniel P. Berrange vs->ioc_tag = 0;
1307536ee4bSTim Hardeck }
1317536ee4bSTim Hardeck
1322ddafce7SDing Hui if (condition & (G_IO_HUP | G_IO_ERR)) {
1332ddafce7SDing Hui vnc_client_error(vs);
1342ddafce7SDing Hui return TRUE;
1352ddafce7SDing Hui }
1362ddafce7SDing Hui
137d5f04223SDaniel P. Berrange wioc = qio_channel_websock_new_server(vs->ioc);
13810bcfe58SDaniel P. Berrange qio_channel_set_name(QIO_CHANNEL(wioc), "vnc-ws-server-websock");
1397536ee4bSTim Hardeck
140d5f04223SDaniel P. Berrange object_unref(OBJECT(vs->ioc));
141d5f04223SDaniel P. Berrange vs->ioc = QIO_CHANNEL(wioc);
142ad6374c4SDaniel P. Berrange trace_vnc_client_io_wrap(vs, vs->ioc, "websock");
1437536ee4bSTim Hardeck
144d5f04223SDaniel P. Berrange qio_channel_websock_handshake(wioc,
145d5f04223SDaniel P. Berrange vncws_handshake_done,
146d5f04223SDaniel P. Berrange vs,
147d5f04223SDaniel P. Berrange NULL);
1487536ee4bSTim Hardeck
149d5f04223SDaniel P. Berrange return TRUE;
1507536ee4bSTim Hardeck }
151