xref: /qemu/migration/channel.c (revision 9af3d9a931156142199c61518937506bfa5475f1)
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();
36*2b667a8cSFabiano Rosas     MigrationIncomingState *mis = migration_incoming_get_current();
3749ed0d24SFei Li     Error *local_err = NULL;
3854314711SJuan Quintela 
39dd4339c5SJuan Quintela     trace_migration_set_incoming_channel(
40dd4339c5SJuan Quintela         ioc, object_get_typename(OBJECT(ioc)));
41dd4339c5SJuan Quintela 
4285a8578eSPeter Xu     if (migrate_channel_requires_tls_upgrade(ioc)) {
43dd4339c5SJuan Quintela         migration_tls_channel_process_incoming(s, ioc, &local_err);
4449ed0d24SFei Li     } else {
4518711405SPeter Xu         migration_ioc_register_yank(ioc);
4649ed0d24SFei Li         migration_ioc_process_incoming(ioc, &local_err);
4749ed0d24SFei Li     }
4849ed0d24SFei Li 
49dd4339c5SJuan Quintela     if (local_err) {
50dd4339c5SJuan Quintela         error_report_err(local_err);
51*2b667a8cSFabiano Rosas         migrate_set_state(&s->state, s->state, MIGRATION_STATUS_FAILED);
52*2b667a8cSFabiano Rosas         if (mis->exit_on_error) {
53*2b667a8cSFabiano Rosas             exit(EXIT_FAILURE);
54*2b667a8cSFabiano Rosas         }
55dd4339c5SJuan Quintela     }
56dd4339c5SJuan Quintela }
57dd4339c5SJuan Quintela 
58dd4339c5SJuan Quintela 
598e1a1931SJuan Quintela /**
608e1a1931SJuan Quintela  * @migration_channel_connect - Create new outgoing migration channel
618e1a1931SJuan Quintela  *
628e1a1931SJuan Quintela  * @s: Current migration state
638e1a1931SJuan Quintela  * @ioc: Channel to which we are connecting
648e1a1931SJuan Quintela  * @hostname: Where we want to connect
65688a3dcbSDr. David Alan Gilbert  * @error: Error indicating failure to connect, free'd here
668e1a1931SJuan Quintela  */
migration_channel_connect(MigrationState * s,QIOChannel * ioc,const char * hostname,Error * error)67dd4339c5SJuan Quintela void migration_channel_connect(MigrationState *s,
68dd4339c5SJuan Quintela                                QIOChannel *ioc,
69688a3dcbSDr. David Alan Gilbert                                const char *hostname,
70688a3dcbSDr. David Alan Gilbert                                Error *error)
71dd4339c5SJuan Quintela {
72dd4339c5SJuan Quintela     trace_migration_set_outgoing_channel(
73688a3dcbSDr. David Alan Gilbert         ioc, object_get_typename(OBJECT(ioc)), hostname, error);
74dd4339c5SJuan Quintela 
75688a3dcbSDr. David Alan Gilbert     if (!error) {
7685a8578eSPeter Xu         if (migrate_channel_requires_tls_upgrade(ioc)) {
77688a3dcbSDr. David Alan Gilbert             migration_tls_channel_connect(s, ioc, hostname, &error);
788b7bf2baSDr. David Alan Gilbert 
798b7bf2baSDr. David Alan Gilbert             if (!error) {
808b7bf2baSDr. David Alan Gilbert                 /* tls_channel_connect will call back to this
818b7bf2baSDr. David Alan Gilbert                  * function after the TLS handshake,
824bbadfc5SFabiano Rosas                  * so we mustn't call migration_connect until then
838b7bf2baSDr. David Alan Gilbert                  */
848b7bf2baSDr. David Alan Gilbert 
858b7bf2baSDr. David Alan Gilbert                 return;
868b7bf2baSDr. David Alan Gilbert             }
87dd4339c5SJuan Quintela         } else {
8877ef2dc1SDaniel P. Berrangé             QEMUFile *f = qemu_file_new_output(ioc);
89dd4339c5SJuan Quintela 
9018711405SPeter Xu             migration_ioc_register_yank(ioc);
917de2e856SLeonardo Bras 
9262df066fSPeter Xu             qemu_mutex_lock(&s->qemu_file_lock);
93dd4339c5SJuan Quintela             s->to_dst_file = f;
9462df066fSPeter Xu             qemu_mutex_unlock(&s->qemu_file_lock);
95dd4339c5SJuan Quintela         }
96dd4339c5SJuan Quintela     }
974bbadfc5SFabiano Rosas     migration_connect(s, error);
98688a3dcbSDr. David Alan Gilbert     error_free(error);
99688a3dcbSDr. David Alan Gilbert }
1006720c2b3Smanish.mishra 
1016720c2b3Smanish.mishra 
1026720c2b3Smanish.mishra /**
1036720c2b3Smanish.mishra  * @migration_channel_read_peek - Peek at migration channel, without
1046720c2b3Smanish.mishra  *     actually removing it from channel buffer.
1056720c2b3Smanish.mishra  *
1066720c2b3Smanish.mishra  * @ioc: the channel object
1076720c2b3Smanish.mishra  * @buf: the memory region to read data into
1086720c2b3Smanish.mishra  * @buflen: the number of bytes to read in @buf
1096720c2b3Smanish.mishra  * @errp: pointer to a NULL-initialized error object
1106720c2b3Smanish.mishra  *
1116720c2b3Smanish.mishra  * Returns 0 if successful, returns -1 and sets @errp if fails.
1126720c2b3Smanish.mishra  */
migration_channel_read_peek(QIOChannel * ioc,const char * buf,const size_t buflen,Error ** errp)1136720c2b3Smanish.mishra int migration_channel_read_peek(QIOChannel *ioc,
1146720c2b3Smanish.mishra                                 const char *buf,
1156720c2b3Smanish.mishra                                 const size_t buflen,
1166720c2b3Smanish.mishra                                 Error **errp)
1176720c2b3Smanish.mishra {
1186720c2b3Smanish.mishra     ssize_t len = 0;
1196720c2b3Smanish.mishra     struct iovec iov = { .iov_base = (char *)buf, .iov_len = buflen };
1206720c2b3Smanish.mishra 
1216720c2b3Smanish.mishra     while (true) {
1226720c2b3Smanish.mishra         len = qio_channel_readv_full(ioc, &iov, 1, NULL, NULL,
1236720c2b3Smanish.mishra                                      QIO_CHANNEL_READ_FLAG_MSG_PEEK, errp);
1246720c2b3Smanish.mishra 
1254f8cf323SAvihai Horon         if (len < 0 && len != QIO_CHANNEL_ERR_BLOCK) {
1264f8cf323SAvihai Horon             return -1;
1274f8cf323SAvihai Horon         }
1284f8cf323SAvihai Horon 
1294f8cf323SAvihai Horon         if (len == 0) {
1304f8cf323SAvihai Horon             error_setg(errp, "Failed to peek at channel");
1316720c2b3Smanish.mishra             return -1;
1326720c2b3Smanish.mishra         }
1336720c2b3Smanish.mishra 
1346720c2b3Smanish.mishra         if (len == buflen) {
1356720c2b3Smanish.mishra             break;
1366720c2b3Smanish.mishra         }
1376720c2b3Smanish.mishra 
1386720c2b3Smanish.mishra         /* 1ms sleep. */
1396720c2b3Smanish.mishra         if (qemu_in_coroutine()) {
1406720c2b3Smanish.mishra             qemu_co_sleep_ns(QEMU_CLOCK_REALTIME, 1000000);
1416720c2b3Smanish.mishra         } else {
1426720c2b3Smanish.mishra             g_usleep(1000);
1436720c2b3Smanish.mishra         }
1446720c2b3Smanish.mishra     }
1456720c2b3Smanish.mishra 
1466720c2b3Smanish.mishra     return 0;
1476720c2b3Smanish.mishra }
148