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(); 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 */ 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 */ 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