blob: e14552ec01ba85283d2da6af3ed2faeeebdb68c6 [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 *
14 */
15
16#include "qemu-common.h"
17#include "qemu_socket.h"
18#include "migration.h"
19#include "qemu-char.h"
aliguori065e2812008-11-11 16:46:33 +000020#include "buffered_file.h"
21#include "block.h"
Blue Swirl0ffbba32010-06-04 20:01:07 +000022#include <sys/types.h>
23#include <sys/wait.h>
aliguori065e2812008-11-11 16:46:33 +000024
25//#define DEBUG_MIGRATION_EXEC
26
27#ifdef DEBUG_MIGRATION_EXEC
malcd0f2c4c2010-02-07 02:03:50 +030028#define DPRINTF(fmt, ...) \
aliguori065e2812008-11-11 16:46:33 +000029 do { printf("migration-exec: " fmt, ## __VA_ARGS__); } while (0)
30#else
malcd0f2c4c2010-02-07 02:03:50 +030031#define DPRINTF(fmt, ...) \
aliguori065e2812008-11-11 16:46:33 +000032 do { } while (0)
33#endif
34
Juan Quintela22f00a42010-05-11 15:56:35 +020035static int file_errno(MigrationState *s)
aliguori065e2812008-11-11 16:46:33 +000036{
37 return errno;
38}
39
Juan Quintela22f00a42010-05-11 15:56:35 +020040static int file_write(MigrationState *s, const void * buf, size_t size)
aliguori065e2812008-11-11 16:46:33 +000041{
42 return write(s->fd, buf, size);
43}
44
Juan Quintela22f00a42010-05-11 15:56:35 +020045static int exec_close(MigrationState *s)
aliguori065e2812008-11-11 16:46:33 +000046{
Anthony Liguori41ef56e2010-06-02 14:55:25 -050047 int ret = 0;
malcd0f2c4c2010-02-07 02:03:50 +030048 DPRINTF("exec_close\n");
aliguori065e2812008-11-11 16:46:33 +000049 if (s->opaque) {
Anthony Liguori41ef56e2010-06-02 14:55:25 -050050 ret = qemu_fclose(s->opaque);
aliguori065e2812008-11-11 16:46:33 +000051 s->opaque = NULL;
52 s->fd = -1;
Eduardo Habkoste375fe32011-11-10 10:41:46 -020053 if (ret >= 0 && !(WIFEXITED(ret) && WEXITSTATUS(ret) == 0)) {
54 /* close succeeded, but non-zero exit code: */
55 ret = -EIO; /* fake errno value */
Anthony Liguori41ef56e2010-06-02 14:55:25 -050056 }
aliguori065e2812008-11-11 16:46:33 +000057 }
Anthony Liguori41ef56e2010-06-02 14:55:25 -050058 return ret;
aliguori065e2812008-11-11 16:46:33 +000059}
60
Juan Quintela07af4452010-05-11 22:27:45 +020061int exec_start_outgoing_migration(MigrationState *s, const char *command)
aliguori065e2812008-11-11 16:46:33 +000062{
aliguori065e2812008-11-11 16:46:33 +000063 FILE *f;
64
aliguori065e2812008-11-11 16:46:33 +000065 f = popen(command, "w");
66 if (f == NULL) {
malcd0f2c4c2010-02-07 02:03:50 +030067 DPRINTF("Unable to popen exec target\n");
Juan Quintela07af4452010-05-11 22:27:45 +020068 goto err_after_popen;
aliguori065e2812008-11-11 16:46:33 +000069 }
70
71 s->fd = fileno(f);
72 if (s->fd == -1) {
malcd0f2c4c2010-02-07 02:03:50 +030073 DPRINTF("Unable to retrieve file descriptor for popen'd handle\n");
aliguori065e2812008-11-11 16:46:33 +000074 goto err_after_open;
75 }
76
Chris Lalancette90750002009-08-05 17:07:35 +020077 socket_set_nonblock(s->fd);
aliguori065e2812008-11-11 16:46:33 +000078
79 s->opaque = qemu_popen(f, "w");
80
aliguori8ad9fa52008-11-12 22:29:11 +000081 s->close = exec_close;
aliguori065e2812008-11-11 16:46:33 +000082 s->get_error = file_errno;
83 s->write = file_write;
aliguori065e2812008-11-11 16:46:33 +000084
85 migrate_fd_connect(s);
Juan Quintela07af4452010-05-11 22:27:45 +020086 return 0;
aliguori065e2812008-11-11 16:46:33 +000087
88err_after_open:
89 pclose(f);
Juan Quintela07af4452010-05-11 22:27:45 +020090err_after_popen:
91 return -1;
aliguori065e2812008-11-11 16:46:33 +000092}
93
Chris Lalancette8a43b1e2009-05-25 16:38:23 +020094static void exec_accept_incoming_migration(void *opaque)
aliguori065e2812008-11-11 16:46:33 +000095{
Chris Lalancette8a43b1e2009-05-25 16:38:23 +020096 QEMUFile *f = opaque;
aliguori065e2812008-11-11 16:46:33 +000097
Juan Quintela511c0232010-06-09 14:10:55 +020098 process_incoming_migration(f);
Juan Quintelacfaf6d32010-03-10 00:10:35 +010099 qemu_set_fd_handler2(qemu_stdio_fd(f), NULL, NULL, NULL, NULL);
aliguori065e2812008-11-11 16:46:33 +0000100 qemu_fclose(f);
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200101}
102
103int exec_start_incoming_migration(const char *command)
104{
105 QEMUFile *f;
106
malcd0f2c4c2010-02-07 02:03:50 +0300107 DPRINTF("Attempting to start an incoming migration\n");
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200108 f = qemu_popen_cmd(command, "r");
109 if(f == NULL) {
malcd0f2c4c2010-02-07 02:03:50 +0300110 DPRINTF("Unable to apply qemu wrapper to popen file\n");
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200111 return -errno;
112 }
113
Paolo Bonzini7f79dd22009-08-12 14:17:35 +0200114 qemu_set_fd_handler2(qemu_stdio_fd(f), NULL,
Juan Quintela1c39e2a2010-03-11 17:55:38 +0100115 exec_accept_incoming_migration, NULL, f);
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200116
117 return 0;
aliguori065e2812008-11-11 16:46:33 +0000118}