|  | /* | 
|  | * QEMU live migration via generic fd | 
|  | * | 
|  | * Copyright Red Hat, Inc. 2009-2016 | 
|  | * | 
|  | * Authors: | 
|  | *  Chris Lalancette <clalance@redhat.com> | 
|  | *  Daniel P. Berrange <berrange@redhat.com> | 
|  | * | 
|  | * This work is licensed under the terms of the GNU GPL, version 2.  See | 
|  | * the COPYING file in the top-level directory. | 
|  | * | 
|  | * Contributions after 2012-01-13 are licensed under the terms of the | 
|  | * GNU GPL, version 2 or (at your option) any later version. | 
|  | */ | 
|  |  | 
|  | #include "qemu/osdep.h" | 
|  | #include "channel.h" | 
|  | #include "fd.h" | 
|  | #include "file.h" | 
|  | #include "migration.h" | 
|  | #include "monitor/monitor.h" | 
|  | #include "qemu/error-report.h" | 
|  | #include "qemu/sockets.h" | 
|  | #include "io/channel-util.h" | 
|  | #include "trace.h" | 
|  |  | 
|  | static bool fd_is_pipe(int fd) | 
|  | { | 
|  | struct stat statbuf; | 
|  |  | 
|  | if (fstat(fd, &statbuf) == -1) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | return S_ISFIFO(statbuf.st_mode); | 
|  | } | 
|  |  | 
|  | static bool migration_fd_valid(int fd) | 
|  | { | 
|  | if (fd_is_socket(fd)) { | 
|  | return true; | 
|  | } | 
|  |  | 
|  | if (fd_is_pipe(fd)) { | 
|  | return true; | 
|  | } | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | void fd_start_outgoing_migration(MigrationState *s, const char *fdname, Error **errp) | 
|  | { | 
|  | QIOChannel *ioc; | 
|  | int fd = monitor_get_fd(monitor_cur(), fdname, errp); | 
|  | if (fd == -1) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (!migration_fd_valid(fd)) { | 
|  | warn_report("fd: migration to a file is deprecated." | 
|  | " Use file: instead."); | 
|  | } | 
|  |  | 
|  | trace_migration_fd_outgoing(fd); | 
|  | ioc = qio_channel_new_fd(fd, errp); | 
|  | if (!ioc) { | 
|  | close(fd); | 
|  | return; | 
|  | } | 
|  |  | 
|  | qio_channel_set_name(ioc, "migration-fd-outgoing"); | 
|  | migration_channel_connect(s, ioc, NULL, NULL); | 
|  | object_unref(OBJECT(ioc)); | 
|  | } | 
|  |  | 
|  | static gboolean fd_accept_incoming_migration(QIOChannel *ioc, | 
|  | GIOCondition condition, | 
|  | gpointer opaque) | 
|  | { | 
|  | migration_channel_process_incoming(ioc); | 
|  | object_unref(OBJECT(ioc)); | 
|  | return G_SOURCE_REMOVE; | 
|  | } | 
|  |  | 
|  | void fd_start_incoming_migration(const char *fdname, Error **errp) | 
|  | { | 
|  | QIOChannel *ioc; | 
|  | int fd = monitor_fd_param(monitor_cur(), fdname, errp); | 
|  | if (fd == -1) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (!migration_fd_valid(fd)) { | 
|  | warn_report("fd: migration to a file is deprecated." | 
|  | " Use file: instead."); | 
|  | } | 
|  |  | 
|  | trace_migration_fd_incoming(fd); | 
|  |  | 
|  | ioc = qio_channel_new_fd(fd, errp); | 
|  | if (!ioc) { | 
|  | close(fd); | 
|  | return; | 
|  | } | 
|  |  | 
|  | 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()); | 
|  | } |