2009-08-18 15:56:25 +02:00
|
|
|
/*
|
|
|
|
* QEMU live migration via generic fd
|
|
|
|
*
|
2016-04-27 12:05:05 +02:00
|
|
|
* Copyright Red Hat, Inc. 2009-2016
|
2009-08-18 15:56:25 +02:00
|
|
|
*
|
|
|
|
* Authors:
|
|
|
|
* Chris Lalancette <clalance@redhat.com>
|
2016-04-27 12:05:05 +02:00
|
|
|
* Daniel P. Berrange <berrange@redhat.com>
|
2009-08-18 15:56:25 +02:00
|
|
|
*
|
|
|
|
* This work is licensed under the terms of the GNU GPL, version 2. See
|
|
|
|
* the COPYING file in the top-level directory.
|
|
|
|
*
|
2012-01-13 17:44:23 +01:00
|
|
|
* Contributions after 2012-01-13 are licensed under the terms of the
|
|
|
|
* GNU GPL, version 2 or (at your option) any later version.
|
2009-08-18 15:56:25 +02:00
|
|
|
*/
|
|
|
|
|
2016-01-26 19:16:54 +01:00
|
|
|
#include "qemu/osdep.h"
|
2024-02-29 16:30:16 +01:00
|
|
|
#include "qapi/error.h"
|
2017-04-17 17:07:04 +02:00
|
|
|
#include "channel.h"
|
2017-04-05 15:58:29 +02:00
|
|
|
#include "fd.h"
|
2024-03-13 22:28:23 +01:00
|
|
|
#include "file.h"
|
2018-05-23 11:14:11 +02:00
|
|
|
#include "migration.h"
|
2012-12-17 18:19:49 +01:00
|
|
|
#include "monitor/monitor.h"
|
2024-02-29 16:30:16 +01:00
|
|
|
#include "io/channel-file.h"
|
2024-03-15 04:20:38 +01:00
|
|
|
#include "io/channel-socket.h"
|
2016-04-27 12:05:05 +02:00
|
|
|
#include "io/channel-util.h"
|
2024-02-29 16:30:16 +01:00
|
|
|
#include "options.h"
|
2016-04-27 12:05:05 +02:00
|
|
|
#include "trace.h"
|
2009-08-18 15:56:25 +02:00
|
|
|
|
2015-01-08 12:11:31 +01:00
|
|
|
|
2024-02-29 16:30:16 +01:00
|
|
|
static struct FdOutgoingArgs {
|
|
|
|
int fd;
|
|
|
|
} outgoing_args;
|
|
|
|
|
|
|
|
int fd_args_get_fd(void)
|
|
|
|
{
|
|
|
|
return outgoing_args.fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
void fd_cleanup_outgoing_migration(void)
|
|
|
|
{
|
|
|
|
if (outgoing_args.fd > 0) {
|
|
|
|
close(outgoing_args.fd);
|
|
|
|
outgoing_args.fd = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-02 10:02:46 +02:00
|
|
|
void fd_start_outgoing_migration(MigrationState *s, const char *fdname, Error **errp)
|
2009-08-18 15:56:25 +02:00
|
|
|
{
|
2016-04-27 12:05:05 +02:00
|
|
|
QIOChannel *ioc;
|
2020-10-05 17:58:44 +02:00
|
|
|
int fd = monitor_get_fd(monitor_cur(), fdname, errp);
|
2024-03-15 04:20:39 +01:00
|
|
|
int newfd;
|
2024-02-29 16:30:16 +01:00
|
|
|
|
2013-02-22 17:36:41 +01:00
|
|
|
if (fd == -1) {
|
2012-10-02 10:02:46 +02:00
|
|
|
return;
|
2009-08-18 15:56:25 +02:00
|
|
|
}
|
2015-01-08 12:11:31 +01:00
|
|
|
|
2016-04-27 12:05:05 +02:00
|
|
|
trace_migration_fd_outgoing(fd);
|
|
|
|
ioc = qio_channel_new_fd(fd, errp);
|
|
|
|
if (!ioc) {
|
|
|
|
close(fd);
|
|
|
|
return;
|
2015-01-08 12:11:31 +01:00
|
|
|
}
|
2009-08-18 15:56:25 +02:00
|
|
|
|
2024-03-15 04:20:39 +01:00
|
|
|
/*
|
|
|
|
* This is dup()ed just to avoid referencing an fd that might
|
|
|
|
* be already closed by the iochannel.
|
|
|
|
*/
|
|
|
|
newfd = dup(fd);
|
|
|
|
if (newfd == -1) {
|
|
|
|
error_setg_errno(errp, errno, "Could not dup FD %d", fd);
|
|
|
|
object_unref(ioc);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
outgoing_args.fd = newfd;
|
2024-02-29 16:30:16 +01:00
|
|
|
|
2023-06-01 11:34:52 +02:00
|
|
|
qio_channel_set_name(ioc, "migration-fd-outgoing");
|
2017-12-15 18:16:55 +01:00
|
|
|
migration_channel_connect(s, ioc, NULL, NULL);
|
2016-04-27 12:05:05 +02:00
|
|
|
object_unref(OBJECT(ioc));
|
2009-08-18 15:56:25 +02:00
|
|
|
}
|
|
|
|
|
2016-04-27 12:05:05 +02:00
|
|
|
static gboolean fd_accept_incoming_migration(QIOChannel *ioc,
|
|
|
|
GIOCondition condition,
|
|
|
|
gpointer opaque)
|
2009-08-18 15:56:25 +02:00
|
|
|
{
|
2017-04-17 17:15:02 +02:00
|
|
|
migration_channel_process_incoming(ioc);
|
2016-04-27 12:05:05 +02:00
|
|
|
object_unref(OBJECT(ioc));
|
2017-07-24 12:51:59 +02:00
|
|
|
return G_SOURCE_REMOVE;
|
2009-08-18 15:56:25 +02:00
|
|
|
}
|
|
|
|
|
2019-05-28 11:16:32 +02:00
|
|
|
void fd_start_incoming_migration(const char *fdname, Error **errp)
|
2009-08-18 15:56:25 +02:00
|
|
|
{
|
2016-04-27 12:05:05 +02:00
|
|
|
QIOChannel *ioc;
|
2020-10-05 17:58:44 +02:00
|
|
|
int fd = monitor_fd_param(monitor_cur(), fdname, errp);
|
2019-05-28 11:16:32 +02:00
|
|
|
if (fd == -1) {
|
|
|
|
return;
|
|
|
|
}
|
2009-08-18 15:56:25 +02:00
|
|
|
|
2016-04-27 12:05:05 +02:00
|
|
|
trace_migration_fd_incoming(fd);
|
|
|
|
|
|
|
|
ioc = qio_channel_new_fd(fd, errp);
|
|
|
|
if (!ioc) {
|
|
|
|
close(fd);
|
2012-10-02 18:21:18 +02:00
|
|
|
return;
|
2009-08-18 15:56:25 +02:00
|
|
|
}
|
|
|
|
|
2024-02-29 16:30:16 +01:00
|
|
|
if (migrate_multifd()) {
|
2024-03-15 04:20:38 +01:00
|
|
|
if (fd_is_socket(fd)) {
|
|
|
|
error_setg(errp,
|
|
|
|
"Multifd migration to a socket FD is not supported");
|
|
|
|
object_unref(ioc);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-03-13 22:28:23 +01:00
|
|
|
file_create_incoming_channels(ioc, errp);
|
|
|
|
} else {
|
|
|
|
qio_channel_set_name(ioc, "migration-fd-incoming");
|
|
|
|
qio_channel_add_watch_full(ioc, G_IO_IN,
|
|
|
|
fd_accept_incoming_migration,
|
|
|
|
NULL, NULL,
|
|
|
|
g_main_context_get_thread_default());
|
2024-02-29 16:30:16 +01:00
|
|
|
}
|
2009-08-18 15:56:25 +02:00
|
|
|
}
|