blob: 91c9cf381e1c903cb700be587dc4e3844891bc3f [file] [log] [blame]
aliguori34c9dd82008-10-13 03:14:31 +00001/*
2 * QEMU live migration
3 *
4 * Copyright IBM, Corp. 2008
5 *
6 * Authors:
7 * Anthony Liguori <aliguori@us.ibm.com>
8 *
9 * This work is licensed under the terms of the GNU GPL, version 2. See
10 * the COPYING file in the top-level directory.
11 *
Paolo Bonzini6b620ca2012-01-13 17:44:23 +010012 * Contributions after 2012-01-13 are licensed under the terms of the
13 * GNU GPL, version 2 or (at your option) any later version.
aliguori34c9dd82008-10-13 03:14:31 +000014 */
15
Dr. David Alan Gilbertd99598c2014-03-19 13:34:28 +000016#include <string.h>
17
aliguori34c9dd82008-10-13 03:14:31 +000018#include "qemu-common.h"
Dr. David Alan Gilbertd99598c2014-03-19 13:34:28 +000019#include "qemu/error-report.h"
Paolo Bonzini1de7afc2012-12-17 18:20:00 +010020#include "qemu/sockets.h"
Paolo Bonzinicaf71f82012-12-17 18:19:50 +010021#include "migration/migration.h"
Juan Quintela557ec5a2012-10-03 14:07:31 +020022#include "migration/qemu-file.h"
Paolo Bonzini737e1502012-12-17 18:19:44 +010023#include "block/block.h"
Alex Bligh6a1751b2013-08-21 16:02:47 +010024#include "qemu/main-loop.h"
aliguori34c9dd82008-10-13 03:14:31 +000025
26//#define DEBUG_MIGRATION_TCP
27
aliguori34c9dd82008-10-13 03:14:31 +000028#ifdef DEBUG_MIGRATION_TCP
malcd0f2c4c2010-02-07 02:03:50 +030029#define DPRINTF(fmt, ...) \
aliguori34c9dd82008-10-13 03:14:31 +000030 do { printf("migration-tcp: " fmt, ## __VA_ARGS__); } while (0)
31#else
malcd0f2c4c2010-02-07 02:03:50 +030032#define DPRINTF(fmt, ...) \
aliguori34c9dd82008-10-13 03:14:31 +000033 do { } while (0)
34#endif
35
Corey Minyard51795022014-10-08 07:11:56 -050036static void tcp_wait_for_connect(int fd, Error *err, void *opaque)
aliguori34c9dd82008-10-13 03:14:31 +000037{
Juan Quintela22f00a42010-05-11 15:56:35 +020038 MigrationState *s = opaque;
aliguori34c9dd82008-10-13 03:14:31 +000039
Orit Wasserman233aa5c2012-09-24 13:11:09 +020040 if (fd < 0) {
Corey Minyard51795022014-10-08 07:11:56 -050041 DPRINTF("migrate connect error: %s\n", error_get_pretty(err));
Paolo Bonzinib3523652013-02-22 17:36:47 +010042 s->file = NULL;
aliguori065e2812008-11-11 16:46:33 +000043 migrate_fd_error(s);
Orit Wasserman233aa5c2012-09-24 13:11:09 +020044 } else {
45 DPRINTF("migrate connect success\n");
Paolo Bonzinib3523652013-02-22 17:36:47 +010046 s->file = qemu_fopen_socket(fd, "wb");
aliguori065e2812008-11-11 16:46:33 +000047 migrate_fd_connect(s);
aliguori34c9dd82008-10-13 03:14:31 +000048 }
49}
50
Paolo Bonzinif37afb52012-10-02 10:02:46 +020051void tcp_start_outgoing_migration(MigrationState *s, const char *host_port, Error **errp)
aliguori34c9dd82008-10-13 03:14:31 +000052{
Paolo Bonzinif8bbc122013-02-22 17:36:41 +010053 inet_nonblocking_connect(host_port, tcp_wait_for_connect, s, errp);
aliguori34c9dd82008-10-13 03:14:31 +000054}
55
56static void tcp_accept_incoming_migration(void *opaque)
57{
58 struct sockaddr_in addr;
59 socklen_t addrlen = sizeof(addr);
Stefan Weile0efb992011-02-23 19:09:16 +010060 int s = (intptr_t)opaque;
aliguori34c9dd82008-10-13 03:14:31 +000061 QEMUFile *f;
Dr. David Alan Gilbertd99598c2014-03-19 13:34:28 +000062 int c, err;
aliguori34c9dd82008-10-13 03:14:31 +000063
64 do {
Kevin Wolf40ff6d72009-12-02 12:24:42 +010065 c = qemu_accept(s, (struct sockaddr *)&addr, &addrlen);
Dr. David Alan Gilbertd99598c2014-03-19 13:34:28 +000066 err = socket_error();
67 } while (c < 0 && err == EINTR);
Paolo Bonzinia6ef2902012-08-07 10:49:13 +020068 qemu_set_fd_handler2(s, NULL, NULL, NULL, NULL);
Paolo Bonzini09bac732012-09-27 13:33:08 +020069 closesocket(s);
aliguori34c9dd82008-10-13 03:14:31 +000070
malcd0f2c4c2010-02-07 02:03:50 +030071 DPRINTF("accepted migration\n");
aliguori34c9dd82008-10-13 03:14:31 +000072
Dr. David Alan Gilbertd99598c2014-03-19 13:34:28 +000073 if (c < 0) {
74 error_report("could not accept migration connection (%s)",
75 strerror(err));
76 return;
aliguori34c9dd82008-10-13 03:14:31 +000077 }
78
Paolo Bonzini0cc3f3c2013-02-22 17:36:39 +010079 f = qemu_fopen_socket(c, "rb");
aliguori34c9dd82008-10-13 03:14:31 +000080 if (f == NULL) {
Dr. David Alan Gilbertd99598c2014-03-19 13:34:28 +000081 error_report("could not qemu_fopen socket");
aliguori34c9dd82008-10-13 03:14:31 +000082 goto out;
83 }
84
Juan Quintela511c0232010-06-09 14:10:55 +020085 process_incoming_migration(f);
Paolo Bonziniab52a822012-08-07 10:50:26 +020086 return;
87
aliguori34c9dd82008-10-13 03:14:31 +000088out:
Paolo Bonzini09bac732012-09-27 13:33:08 +020089 closesocket(c);
aliguori34c9dd82008-10-13 03:14:31 +000090}
91
Paolo Bonzini43eaae22012-10-02 18:21:18 +020092void tcp_start_incoming_migration(const char *host_port, Error **errp)
aliguori34c9dd82008-10-13 03:14:31 +000093{
aliguori34c9dd82008-10-13 03:14:31 +000094 int s;
95
Amos Kongd5c5dac2012-05-11 00:28:35 +080096 s = inet_listen(host_port, NULL, 256, SOCK_STREAM, 0, errp);
Amos Kongd5c5dac2012-05-11 00:28:35 +080097 if (s < 0) {
Paolo Bonzini43eaae22012-10-02 18:21:18 +020098 return;
Juan Quintelaee86c612011-02-23 20:44:29 +010099 }
aliguori34c9dd82008-10-13 03:14:31 +0000100
101 qemu_set_fd_handler2(s, NULL, tcp_accept_incoming_migration, NULL,
Stefan Weile0efb992011-02-23 19:09:16 +0100102 (void *)(intptr_t)s);
aliguori34c9dd82008-10-13 03:14:31 +0000103}