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, 77*4bbadfc5SFabiano Rosas * so we mustn't call migration_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 } 92*4bbadfc5SFabiano Rosas migration_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 */ 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 1204f8cf323SAvihai Horon if (len < 0 && len != QIO_CHANNEL_ERR_BLOCK) { 1214f8cf323SAvihai Horon return -1; 1224f8cf323SAvihai Horon } 1234f8cf323SAvihai Horon 1244f8cf323SAvihai Horon if (len == 0) { 1254f8cf323SAvihai 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