xref: /qemu/migration/channel.c (revision 6720c2b32725e6ac404f22851a0ecd0a71d0cbe2)
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  */
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  */
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 }
95*6720c2b3Smanish.mishra 
96*6720c2b3Smanish.mishra 
97*6720c2b3Smanish.mishra /**
98*6720c2b3Smanish.mishra  * @migration_channel_read_peek - Peek at migration channel, without
99*6720c2b3Smanish.mishra  *     actually removing it from channel buffer.
100*6720c2b3Smanish.mishra  *
101*6720c2b3Smanish.mishra  * @ioc: the channel object
102*6720c2b3Smanish.mishra  * @buf: the memory region to read data into
103*6720c2b3Smanish.mishra  * @buflen: the number of bytes to read in @buf
104*6720c2b3Smanish.mishra  * @errp: pointer to a NULL-initialized error object
105*6720c2b3Smanish.mishra  *
106*6720c2b3Smanish.mishra  * Returns 0 if successful, returns -1 and sets @errp if fails.
107*6720c2b3Smanish.mishra  */
108*6720c2b3Smanish.mishra int migration_channel_read_peek(QIOChannel *ioc,
109*6720c2b3Smanish.mishra                                 const char *buf,
110*6720c2b3Smanish.mishra                                 const size_t buflen,
111*6720c2b3Smanish.mishra                                 Error **errp)
112*6720c2b3Smanish.mishra {
113*6720c2b3Smanish.mishra     ssize_t len = 0;
114*6720c2b3Smanish.mishra     struct iovec iov = { .iov_base = (char *)buf, .iov_len = buflen };
115*6720c2b3Smanish.mishra 
116*6720c2b3Smanish.mishra     while (true) {
117*6720c2b3Smanish.mishra         len = qio_channel_readv_full(ioc, &iov, 1, NULL, NULL,
118*6720c2b3Smanish.mishra                                      QIO_CHANNEL_READ_FLAG_MSG_PEEK, errp);
119*6720c2b3Smanish.mishra 
120*6720c2b3Smanish.mishra         if (len <= 0 && len != QIO_CHANNEL_ERR_BLOCK) {
121*6720c2b3Smanish.mishra             error_setg(errp,
122*6720c2b3Smanish.mishra                        "Failed to peek at channel");
123*6720c2b3Smanish.mishra             return -1;
124*6720c2b3Smanish.mishra         }
125*6720c2b3Smanish.mishra 
126*6720c2b3Smanish.mishra         if (len == buflen) {
127*6720c2b3Smanish.mishra             break;
128*6720c2b3Smanish.mishra         }
129*6720c2b3Smanish.mishra 
130*6720c2b3Smanish.mishra         /* 1ms sleep. */
131*6720c2b3Smanish.mishra         if (qemu_in_coroutine()) {
132*6720c2b3Smanish.mishra             qemu_co_sleep_ns(QEMU_CLOCK_REALTIME, 1000000);
133*6720c2b3Smanish.mishra         } else {
134*6720c2b3Smanish.mishra             g_usleep(1000);
135*6720c2b3Smanish.mishra         }
136*6720c2b3Smanish.mishra     }
137*6720c2b3Smanish.mishra 
138*6720c2b3Smanish.mishra     return 0;
139*6720c2b3Smanish.mishra }
140