xref: /qemu/migration/channel.c (revision 4f8cf323)
1dd4339c5SJuan Quintela /*
2dd4339c5SJuan Quintela  * QEMU live migration channel operations
3dd4339c5SJuan Quintela  *
4dd4339c5SJuan Quintela  * Copyright Red Hat, Inc. 2016
5dd4339c5SJuan Quintela  *
6dd4339c5SJuan Quintela  * Authors:
7dd4339c5SJuan Quintela  *  Daniel P. Berrange <berrange@redhat.com>
8dd4339c5SJuan Quintela  *
9dd4339c5SJuan Quintela  * Contributions after 2012-01-13 are licensed under the terms of the
10dd4339c5SJuan Quintela  * GNU GPL, version 2 or (at your option) any later version.
11dd4339c5SJuan Quintela  */
12dd4339c5SJuan Quintela 
13dd4339c5SJuan Quintela #include "qemu/osdep.h"
14dd4339c5SJuan Quintela #include "channel.h"
1541d64227SJuan Quintela #include "tls.h"
166666c96aSJuan Quintela #include "migration.h"
1777ef2dc1SDaniel P. Berrangé #include "qemu-file.h"
18dd4339c5SJuan Quintela #include "trace.h"
19dd4339c5SJuan Quintela #include "qapi/error.h"
20dd4339c5SJuan Quintela #include "io/channel-tls.h"
21b5eea99eSLukas Straub #include "io/channel-socket.h"
22b5eea99eSLukas Straub #include "qemu/yank.h"
231a92d6d5SLukas Straub #include "yank_functions.h"
24dd4339c5SJuan Quintela 
258e1a1931SJuan Quintela /**
268e1a1931SJuan Quintela  * @migration_channel_process_incoming - Create new incoming migration channel
278e1a1931SJuan Quintela  *
288e1a1931SJuan Quintela  * Notice that TLS is special.  For it we listen in a listener socket,
298e1a1931SJuan Quintela  * and then create a new client socket from the TLS library.
308e1a1931SJuan Quintela  *
318e1a1931SJuan Quintela  * @ioc: Channel to which we are connecting
328e1a1931SJuan Quintela  */
migration_channel_process_incoming(QIOChannel * ioc)3354314711SJuan Quintela void migration_channel_process_incoming(QIOChannel *ioc)
34dd4339c5SJuan Quintela {
3554314711SJuan Quintela     MigrationState *s = migrate_get_current();
3649ed0d24SFei Li     Error *local_err = NULL;
3754314711SJuan Quintela 
38dd4339c5SJuan Quintela     trace_migration_set_incoming_channel(
39dd4339c5SJuan Quintela         ioc, object_get_typename(OBJECT(ioc)));
40dd4339c5SJuan Quintela 
4185a8578eSPeter Xu     if (migrate_channel_requires_tls_upgrade(ioc)) {
42dd4339c5SJuan Quintela         migration_tls_channel_process_incoming(s, ioc, &local_err);
4349ed0d24SFei Li     } else {
4418711405SPeter Xu         migration_ioc_register_yank(ioc);
4549ed0d24SFei Li         migration_ioc_process_incoming(ioc, &local_err);
4649ed0d24SFei Li     }
4749ed0d24SFei Li 
48dd4339c5SJuan Quintela     if (local_err) {
49dd4339c5SJuan Quintela         error_report_err(local_err);
50dd4339c5SJuan Quintela     }
51dd4339c5SJuan Quintela }
52dd4339c5SJuan Quintela 
53dd4339c5SJuan Quintela 
548e1a1931SJuan Quintela /**
558e1a1931SJuan Quintela  * @migration_channel_connect - Create new outgoing migration channel
568e1a1931SJuan Quintela  *
578e1a1931SJuan Quintela  * @s: Current migration state
588e1a1931SJuan Quintela  * @ioc: Channel to which we are connecting
598e1a1931SJuan Quintela  * @hostname: Where we want to connect
60688a3dcbSDr. David Alan Gilbert  * @error: Error indicating failure to connect, free'd here
618e1a1931SJuan Quintela  */
migration_channel_connect(MigrationState * s,QIOChannel * ioc,const char * hostname,Error * error)62dd4339c5SJuan Quintela void migration_channel_connect(MigrationState *s,
63dd4339c5SJuan Quintela                                QIOChannel *ioc,
64688a3dcbSDr. David Alan Gilbert                                const char *hostname,
65688a3dcbSDr. David Alan Gilbert                                Error *error)
66dd4339c5SJuan Quintela {
67dd4339c5SJuan Quintela     trace_migration_set_outgoing_channel(
68688a3dcbSDr. David Alan Gilbert         ioc, object_get_typename(OBJECT(ioc)), hostname, error);
69dd4339c5SJuan Quintela 
70688a3dcbSDr. David Alan Gilbert     if (!error) {
7185a8578eSPeter Xu         if (migrate_channel_requires_tls_upgrade(ioc)) {
72688a3dcbSDr. David Alan Gilbert             migration_tls_channel_connect(s, ioc, hostname, &error);
738b7bf2baSDr. David Alan Gilbert 
748b7bf2baSDr. David Alan Gilbert             if (!error) {
758b7bf2baSDr. David Alan Gilbert                 /* tls_channel_connect will call back to this
768b7bf2baSDr. David Alan Gilbert                  * function after the TLS handshake,
778b7bf2baSDr. David Alan Gilbert                  * so we mustn't call migrate_fd_connect until then
788b7bf2baSDr. David Alan Gilbert                  */
798b7bf2baSDr. David Alan Gilbert 
808b7bf2baSDr. David Alan Gilbert                 return;
818b7bf2baSDr. David Alan Gilbert             }
82dd4339c5SJuan Quintela         } else {
8377ef2dc1SDaniel P. Berrangé             QEMUFile *f = qemu_file_new_output(ioc);
84dd4339c5SJuan Quintela 
8518711405SPeter Xu             migration_ioc_register_yank(ioc);
867de2e856SLeonardo Bras 
8762df066fSPeter Xu             qemu_mutex_lock(&s->qemu_file_lock);
88dd4339c5SJuan Quintela             s->to_dst_file = f;
8962df066fSPeter Xu             qemu_mutex_unlock(&s->qemu_file_lock);
90dd4339c5SJuan Quintela         }
91dd4339c5SJuan Quintela     }
92688a3dcbSDr. David Alan Gilbert     migrate_fd_connect(s, error);
93688a3dcbSDr. David Alan Gilbert     error_free(error);
94688a3dcbSDr. David Alan Gilbert }
956720c2b3Smanish.mishra 
966720c2b3Smanish.mishra 
976720c2b3Smanish.mishra /**
986720c2b3Smanish.mishra  * @migration_channel_read_peek - Peek at migration channel, without
996720c2b3Smanish.mishra  *     actually removing it from channel buffer.
1006720c2b3Smanish.mishra  *
1016720c2b3Smanish.mishra  * @ioc: the channel object
1026720c2b3Smanish.mishra  * @buf: the memory region to read data into
1036720c2b3Smanish.mishra  * @buflen: the number of bytes to read in @buf
1046720c2b3Smanish.mishra  * @errp: pointer to a NULL-initialized error object
1056720c2b3Smanish.mishra  *
1066720c2b3Smanish.mishra  * Returns 0 if successful, returns -1 and sets @errp if fails.
1076720c2b3Smanish.mishra  */
migration_channel_read_peek(QIOChannel * ioc,const char * buf,const size_t buflen,Error ** errp)1086720c2b3Smanish.mishra int migration_channel_read_peek(QIOChannel *ioc,
1096720c2b3Smanish.mishra                                 const char *buf,
1106720c2b3Smanish.mishra                                 const size_t buflen,
1116720c2b3Smanish.mishra                                 Error **errp)
1126720c2b3Smanish.mishra {
1136720c2b3Smanish.mishra     ssize_t len = 0;
1146720c2b3Smanish.mishra     struct iovec iov = { .iov_base = (char *)buf, .iov_len = buflen };
1156720c2b3Smanish.mishra 
1166720c2b3Smanish.mishra     while (true) {
1176720c2b3Smanish.mishra         len = qio_channel_readv_full(ioc, &iov, 1, NULL, NULL,
1186720c2b3Smanish.mishra                                      QIO_CHANNEL_READ_FLAG_MSG_PEEK, errp);
1196720c2b3Smanish.mishra 
120*4f8cf323SAvihai Horon         if (len < 0 && len != QIO_CHANNEL_ERR_BLOCK) {
121*4f8cf323SAvihai Horon             return -1;
122*4f8cf323SAvihai Horon         }
123*4f8cf323SAvihai Horon 
124*4f8cf323SAvihai Horon         if (len == 0) {
125*4f8cf323SAvihai Horon             error_setg(errp, "Failed to peek at channel");
1266720c2b3Smanish.mishra             return -1;
1276720c2b3Smanish.mishra         }
1286720c2b3Smanish.mishra 
1296720c2b3Smanish.mishra         if (len == buflen) {
1306720c2b3Smanish.mishra             break;
1316720c2b3Smanish.mishra         }
1326720c2b3Smanish.mishra 
1336720c2b3Smanish.mishra         /* 1ms sleep. */
1346720c2b3Smanish.mishra         if (qemu_in_coroutine()) {
1356720c2b3Smanish.mishra             qemu_co_sleep_ns(QEMU_CLOCK_REALTIME, 1000000);
1366720c2b3Smanish.mishra         } else {
1376720c2b3Smanish.mishra             g_usleep(1000);
1386720c2b3Smanish.mishra         }
1396720c2b3Smanish.mishra     }
1406720c2b3Smanish.mishra 
1416720c2b3Smanish.mishra     return 0;
1426720c2b3Smanish.mishra }
143