blob: 54358271c5b5b5793b2989f4e2cbbe92609b83f4 [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"
20#include "sysemu.h"
aliguori065e2812008-11-11 16:46:33 +000021#include "buffered_file.h"
22#include "block.h"
23
24//#define DEBUG_MIGRATION_EXEC
25
26#ifdef DEBUG_MIGRATION_EXEC
malcd0f2c4c2010-02-07 02:03:50 +030027#define DPRINTF(fmt, ...) \
aliguori065e2812008-11-11 16:46:33 +000028 do { printf("migration-exec: " fmt, ## __VA_ARGS__); } while (0)
29#else
malcd0f2c4c2010-02-07 02:03:50 +030030#define DPRINTF(fmt, ...) \
aliguori065e2812008-11-11 16:46:33 +000031 do { } while (0)
32#endif
33
34static int file_errno(FdMigrationState *s)
35{
36 return errno;
37}
38
39static int file_write(FdMigrationState *s, const void * buf, size_t size)
40{
41 return write(s->fd, buf, size);
42}
43
44static int exec_close(FdMigrationState *s)
45{
malcd0f2c4c2010-02-07 02:03:50 +030046 DPRINTF("exec_close\n");
aliguori065e2812008-11-11 16:46:33 +000047 if (s->opaque) {
48 qemu_fclose(s->opaque);
49 s->opaque = NULL;
50 s->fd = -1;
51 }
52 return 0;
53}
54
Jan Kiszkaf327aa02009-11-30 18:21:21 +010055MigrationState *exec_start_outgoing_migration(Monitor *mon,
56 const char *command,
lirans@il.ibm.comc163b5c2009-11-02 15:40:58 +020057 int64_t bandwidth_limit,
58 int detach,
59 int blk,
60 int inc)
aliguori065e2812008-11-11 16:46:33 +000061{
62 FdMigrationState *s;
63 FILE *f;
64
65 s = qemu_mallocz(sizeof(*s));
aliguori065e2812008-11-11 16:46:33 +000066
67 f = popen(command, "w");
68 if (f == NULL) {
malcd0f2c4c2010-02-07 02:03:50 +030069 DPRINTF("Unable to popen exec target\n");
aliguori065e2812008-11-11 16:46:33 +000070 goto err_after_alloc;
71 }
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;
86 s->mig_state.cancel = migrate_fd_cancel;
87 s->mig_state.get_status = migrate_fd_get_status;
88 s->mig_state.release = migrate_fd_release;
89
lirans@il.ibm.comc163b5c2009-11-02 15:40:58 +020090 s->mig_state.blk = blk;
91 s->mig_state.shared = inc;
Jan Kiszkaf327aa02009-11-30 18:21:21 +010092
aliguori065e2812008-11-11 16:46:33 +000093 s->state = MIG_STATE_ACTIVE;
Jan Kiszkaf327aa02009-11-30 18:21:21 +010094 s->mon = NULL;
aliguori065e2812008-11-11 16:46:33 +000095 s->bandwidth_limit = bandwidth_limit;
96
Jan Kiszkaf327aa02009-11-30 18:21:21 +010097 if (!detach) {
98 migrate_fd_monitor_suspend(s, mon);
99 }
aliguori065e2812008-11-11 16:46:33 +0000100
101 migrate_fd_connect(s);
102 return &s->mig_state;
103
104err_after_open:
105 pclose(f);
106err_after_alloc:
107 qemu_free(s);
aliguori065e2812008-11-11 16:46:33 +0000108 return NULL;
109}
110
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200111static void exec_accept_incoming_migration(void *opaque)
aliguori065e2812008-11-11 16:46:33 +0000112{
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200113 QEMUFile *f = opaque;
aliguori065e2812008-11-11 16:46:33 +0000114 int ret;
aliguori065e2812008-11-11 16:46:33 +0000115
aliguori065e2812008-11-11 16:46:33 +0000116 ret = qemu_loadvm_state(f);
117 if (ret < 0) {
118 fprintf(stderr, "load of migration failed\n");
119 goto err;
120 }
121 qemu_announce_self();
malcd0f2c4c2010-02-07 02:03:50 +0300122 DPRINTF("successfully loaded vm state\n");
Juan Quintelacfaf6d32010-03-10 00:10:35 +0100123
Paolo Bonzinid399f672009-07-27 23:17:51 +0200124 if (autostart)
125 vm_start();
aliguori065e2812008-11-11 16:46:33 +0000126
127err:
Juan Quintelacfaf6d32010-03-10 00:10:35 +0100128 qemu_set_fd_handler2(qemu_stdio_fd(f), NULL, NULL, NULL, NULL);
aliguori065e2812008-11-11 16:46:33 +0000129 qemu_fclose(f);
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200130}
131
132int exec_start_incoming_migration(const char *command)
133{
134 QEMUFile *f;
135
malcd0f2c4c2010-02-07 02:03:50 +0300136 DPRINTF("Attempting to start an incoming migration\n");
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200137 f = qemu_popen_cmd(command, "r");
138 if(f == NULL) {
malcd0f2c4c2010-02-07 02:03:50 +0300139 DPRINTF("Unable to apply qemu wrapper to popen file\n");
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200140 return -errno;
141 }
142
Paolo Bonzini7f79dd22009-08-12 14:17:35 +0200143 qemu_set_fd_handler2(qemu_stdio_fd(f), NULL,
Juan Quintela1c39e2a2010-03-11 17:55:38 +0100144 exec_accept_incoming_migration, NULL, f);
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200145
146 return 0;
aliguori065e2812008-11-11 16:46:33 +0000147}