| /* | 
 |  * QEMU live migration | 
 |  * | 
 |  * Copyright IBM, Corp. 2008 | 
 |  * Copyright Dell MessageOne 2008 | 
 |  * Copyright Red Hat, Inc. 2015-2016 | 
 |  * | 
 |  * Authors: | 
 |  *  Anthony Liguori   <aliguori@us.ibm.com> | 
 |  *  Charles Duffy     <charles_duffy@messageone.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 "qapi/type-helpers.h" | 
 | #include "qemu/error-report.h" | 
 | #include "channel.h" | 
 | #include "exec.h" | 
 | #include "migration.h" | 
 | #include "io/channel-command.h" | 
 | #include "trace.h" | 
 | #include "qemu/cutils.h" | 
 |  | 
 | #ifdef WIN32 | 
 | const char *exec_get_cmd_path(void) | 
 | { | 
 |     g_autofree char *detected_path = g_new(char, MAX_PATH); | 
 |     if (GetSystemDirectoryA(detected_path, MAX_PATH) == 0) { | 
 |         warn_report("Could not detect cmd.exe path, using default."); | 
 |         return "C:\\Windows\\System32\\cmd.exe"; | 
 |     } | 
 |     pstrcat(detected_path, MAX_PATH, "\\cmd.exe"); | 
 |     return g_steal_pointer(&detected_path); | 
 | } | 
 | #endif | 
 |  | 
 | void exec_start_outgoing_migration(MigrationState *s, strList *command, | 
 |                                    Error **errp) | 
 | { | 
 |     QIOChannel *ioc = NULL; | 
 |     g_auto(GStrv) argv = strv_from_str_list(command); | 
 |     const char * const *args = (const char * const *) argv; | 
 |     g_autofree char *new_command = g_strjoinv(" ", (char **)argv); | 
 |  | 
 |     trace_migration_exec_outgoing(new_command); | 
 |     ioc = QIO_CHANNEL(qio_channel_command_new_spawn(args, O_RDWR, errp)); | 
 |     if (!ioc) { | 
 |         return; | 
 |     } | 
 |  | 
 |     qio_channel_set_name(ioc, "migration-exec-outgoing"); | 
 |     migration_channel_connect(s, ioc, NULL, NULL); | 
 |     object_unref(OBJECT(ioc)); | 
 | } | 
 |  | 
 | static gboolean exec_accept_incoming_migration(QIOChannel *ioc, | 
 |                                                GIOCondition condition, | 
 |                                                gpointer opaque) | 
 | { | 
 |     migration_channel_process_incoming(ioc); | 
 |     object_unref(OBJECT(ioc)); | 
 |     return G_SOURCE_REMOVE; | 
 | } | 
 |  | 
 | void exec_start_incoming_migration(strList *command, Error **errp) | 
 | { | 
 |     QIOChannel *ioc; | 
 |     g_auto(GStrv) argv = strv_from_str_list(command); | 
 |     const char * const *args = (const char * const *) argv; | 
 |     g_autofree char *new_command = g_strjoinv(" ", (char **)argv); | 
 |  | 
 |     trace_migration_exec_incoming(new_command); | 
 |     ioc = QIO_CHANNEL(qio_channel_command_new_spawn(args, O_RDWR, errp)); | 
 |     if (!ioc) { | 
 |         return; | 
 |     } | 
 |  | 
 |     qio_channel_set_name(ioc, "migration-exec-incoming"); | 
 |     qio_channel_add_watch_full(ioc, G_IO_IN, | 
 |                                exec_accept_incoming_migration, | 
 |                                NULL, NULL, | 
 |                                g_main_context_get_thread_default()); | 
 | } |