blob: b7b1055e881d877eeb70aa9ae72677e4c3a42da3 [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;
Anthony Liguori41ef56e2010-06-02 14:55:25 -050053 if (ret != -1 &&
54 WIFEXITED(ret)
55 && WEXITSTATUS(ret) == 0) {
56 ret = 0;
57 } else {
58 ret = -1;
59 }
aliguori065e2812008-11-11 16:46:33 +000060 }
Anthony Liguori41ef56e2010-06-02 14:55:25 -050061 return ret;
aliguori065e2812008-11-11 16:46:33 +000062}
63
Juan Quintela07af4452010-05-11 22:27:45 +020064int exec_start_outgoing_migration(MigrationState *s, const char *command)
aliguori065e2812008-11-11 16:46:33 +000065{
aliguori065e2812008-11-11 16:46:33 +000066 FILE *f;
67
aliguori065e2812008-11-11 16:46:33 +000068 f = popen(command, "w");
69 if (f == NULL) {
malcd0f2c4c2010-02-07 02:03:50 +030070 DPRINTF("Unable to popen exec target\n");
Juan Quintela07af4452010-05-11 22:27:45 +020071 goto err_after_popen;
aliguori065e2812008-11-11 16:46:33 +000072 }
73
74 s->fd = fileno(f);
75 if (s->fd == -1) {
malcd0f2c4c2010-02-07 02:03:50 +030076 DPRINTF("Unable to retrieve file descriptor for popen'd handle\n");
aliguori065e2812008-11-11 16:46:33 +000077 goto err_after_open;
78 }
79
Chris Lalancette90750002009-08-05 17:07:35 +020080 socket_set_nonblock(s->fd);
aliguori065e2812008-11-11 16:46:33 +000081
82 s->opaque = qemu_popen(f, "w");
83
aliguori8ad9fa52008-11-12 22:29:11 +000084 s->close = exec_close;
aliguori065e2812008-11-11 16:46:33 +000085 s->get_error = file_errno;
86 s->write = file_write;
aliguori065e2812008-11-11 16:46:33 +000087
88 migrate_fd_connect(s);
Juan Quintela07af4452010-05-11 22:27:45 +020089 return 0;
aliguori065e2812008-11-11 16:46:33 +000090
91err_after_open:
92 pclose(f);
Juan Quintela07af4452010-05-11 22:27:45 +020093err_after_popen:
94 return -1;
aliguori065e2812008-11-11 16:46:33 +000095}
96
Chris Lalancette8a43b1e2009-05-25 16:38:23 +020097static void exec_accept_incoming_migration(void *opaque)
aliguori065e2812008-11-11 16:46:33 +000098{
Chris Lalancette8a43b1e2009-05-25 16:38:23 +020099 QEMUFile *f = opaque;
aliguori065e2812008-11-11 16:46:33 +0000100
Juan Quintela511c0232010-06-09 14:10:55 +0200101 process_incoming_migration(f);
Juan Quintelacfaf6d32010-03-10 00:10:35 +0100102 qemu_set_fd_handler2(qemu_stdio_fd(f), NULL, NULL, NULL, NULL);
aliguori065e2812008-11-11 16:46:33 +0000103 qemu_fclose(f);
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200104}
105
106int exec_start_incoming_migration(const char *command)
107{
108 QEMUFile *f;
109
malcd0f2c4c2010-02-07 02:03:50 +0300110 DPRINTF("Attempting to start an incoming migration\n");
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200111 f = qemu_popen_cmd(command, "r");
112 if(f == NULL) {
malcd0f2c4c2010-02-07 02:03:50 +0300113 DPRINTF("Unable to apply qemu wrapper to popen file\n");
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200114 return -errno;
115 }
116
Paolo Bonzini7f79dd22009-08-12 14:17:35 +0200117 qemu_set_fd_handler2(qemu_stdio_fd(f), NULL,
Juan Quintela1c39e2a2010-03-11 17:55:38 +0100118 exec_accept_incoming_migration, NULL, f);
Chris Lalancette8a43b1e2009-05-25 16:38:23 +0200119
120 return 0;
aliguori065e2812008-11-11 16:46:33 +0000121}