blob: 6c97db973c93579421ff29cade1623cd09cc8cff [file] [log] [blame]
aliguori065e2812008-11-11 16:46:33 +00001/*
2 * QEMU live migration
3 *
4 * Copyright IBM, Corp. 2008
5 * Copyright Dell MessageOne 2008
6 *
7 * Authors:
8 * Anthony Liguori <aliguori@us.ibm.com>
9 * Charles Duffy <charles_duffy@messageone.com>
10 *
11 * This work is licensed under the terms of the GNU GPL, version 2. See
12 * the COPYING file in the top-level directory.
13 *
Paolo Bonzini6b620ca2012-01-13 17:44:23 +010014 * Contributions after 2012-01-13 are licensed under the terms of the
15 * GNU GPL, version 2 or (at your option) any later version.
aliguori065e2812008-11-11 16:46:33 +000016 */
17
18#include "qemu-common.h"
19#include "qemu_socket.h"
20#include "migration.h"
21#include "qemu-char.h"
aliguori065e2812008-11-11 16:46:33 +000022#include "buffered_file.h"
23#include "block.h"
Blue Swirl0ffbba32010-06-04 20:01:07 +000024#include <sys/types.h>
25#include <sys/wait.h>
aliguori065e2812008-11-11 16:46:33 +000026
27//#define DEBUG_MIGRATION_EXEC
28
29#ifdef DEBUG_MIGRATION_EXEC
malcd0f2c4c2010-02-07 02:03:50 +030030#define DPRINTF(fmt, ...) \
aliguori065e2812008-11-11 16:46:33 +000031 do { printf("migration-exec: " fmt, ## __VA_ARGS__); } while (0)
32#else
malcd0f2c4c2010-02-07 02:03:50 +030033#define DPRINTF(fmt, ...) \
aliguori065e2812008-11-11 16:46:33 +000034 do { } while (0)
35#endif
36
Juan Quintela22f00a42010-05-11 15:56:35 +020037static int file_errno(MigrationState *s)
aliguori065e2812008-11-11 16:46:33 +000038{
39 return errno;
40}
41
Juan Quintela22f00a42010-05-11 15:56:35 +020042static int file_write(MigrationState *s, const void * buf, size_t size)
aliguori065e2812008-11-11 16:46:33 +000043{
44 return write(s->fd, buf, size);
45}
46
Juan Quintela22f00a42010-05-11 15:56:35 +020047static int exec_close(MigrationState *s)
aliguori065e2812008-11-11 16:46:33 +000048{
Anthony Liguori41ef56e2010-06-02 14:55:25 -050049 int ret = 0;
malcd0f2c4c2010-02-07 02:03:50 +030050 DPRINTF("exec_close\n");
aliguori065e2812008-11-11 16:46:33 +000051 if (s->opaque) {
Anthony Liguori41ef56e2010-06-02 14:55:25 -050052 ret = qemu_fclose(s->opaque);
aliguori065e2812008-11-11 16:46:33 +000053 s->opaque = NULL;
54 s->fd = -1;
Eduardo Habkoste375fe32011-11-10 10:41:46 -020055 if (ret >= 0 && !(WIFEXITED(ret) && WEXITSTATUS(ret) == 0)) {
56 /* close succeeded, but non-zero exit code: */
57 ret = -EIO; /* fake errno value */
Anthony Liguori41ef56e2010-06-02 14:55:25 -050058 }
aliguori065e2812008-11-11 16:46:33 +000059 }
Anthony Liguori41ef56e2010-06-02 14:55:25 -050060 return ret;
aliguori065e2812008-11-11 16:46:33 +000061}
62
Juan Quintela07af4452010-05-11 22:27:45 +020063int exec_start_outgoing_migration(MigrationState *s, const char *command)
aliguori065e2812008-11-11 16:46:33 +000064{
aliguori065e2812008-11-11 16:46:33 +000065 FILE *f;
66
aliguori065e2812008-11-11 16:46:33 +000067 f = popen(command, "w");
68 if (f == NULL) {
malcd0f2c4c2010-02-07 02:03:50 +030069 DPRINTF("Unable to popen exec target\n");
Juan Quintela07af4452010-05-11 22:27:45 +020070 goto err_after_popen;
aliguori065e2812008-11-11 16:46:33 +000071 }
72
73 s->fd = fileno(f);
74 if (s->fd == -1) {
malcd0f2c4c2010-02-07 02:03:50 +030075 DPRINTF("Unable to retrieve file descriptor for popen'd handle\n");
aliguori065e2812008-11-11 16:46:33 +000076 goto err_after_open;
77 }
78
Chris Lalancette90750002009-08-05 17:07:35 +020079 socket_set_nonblock(s->fd);
aliguori065e2812008-11-11 16:46:33 +000080
81 s->opaque = qemu_popen(f, "w");
82
aliguori8ad9fa52008-11-12 22:29:11 +000083 s->close = exec_close;
aliguori065e2812008-11-11 16:46:33 +000084 s->get_error = file_errno;
85 s->write = file_write;
aliguori065e2812008-11-11 16:46:33 +000086
87 migrate_fd_connect(s);
Juan Quintela07af4452010-05-11 22:27:45 +020088 return 0;
aliguori065e2812008-11-11 16:46:33 +000089
90err_after_open:
91 pclose(f);
Juan Quintela07af4452010-05-11 22:27:45 +020092err_after_popen:
93 return -1;
aliguori065e2812008-11-11 16:46:33 +000094}
95
Chris Lalancette8a43b1e2009-05-25 16:38:23 +020096static void exec_accept_incoming_migration(void *opaque)
aliguori065e2812008-11-11 16:46:33 +000097{
Chris Lalancette8a43b1e2009-05-25 16:38:23 +020098 QEMUFile *f = opaque;
aliguori065e2812008-11-11 16:46:33 +000099
Juan Quintela511c0232010-06-09 14:10:55 +0200100 process_incoming_migration(f);
Juan Quintelacfaf6d32010-03-10 00:10:35 +0100101 qemu_set_fd_handler2(qemu_stdio_fd(f), NULL, NULL, NULL, NULL);
aliguori065e2812008-11-11 16:46:33 +0000102 qemu_fclose(f);
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200103}
104
105int exec_start_incoming_migration(const char *command)
106{
107 QEMUFile *f;
108
malcd0f2c4c2010-02-07 02:03:50 +0300109 DPRINTF("Attempting to start an incoming migration\n");
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200110 f = qemu_popen_cmd(command, "r");
111 if(f == NULL) {
malcd0f2c4c2010-02-07 02:03:50 +0300112 DPRINTF("Unable to apply qemu wrapper to popen file\n");
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200113 return -errno;
114 }
115
Paolo Bonzini7f79dd22009-08-12 14:17:35 +0200116 qemu_set_fd_handler2(qemu_stdio_fd(f), NULL,
Juan Quintela1c39e2a2010-03-11 17:55:38 +0100117 exec_accept_incoming_migration, NULL, f);
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200118
119 return 0;
aliguori065e2812008-11-11 16:46:33 +0000120}