| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * QEMU live migration | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright IBM, Corp. 2008 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Authors: | 
					
						
							|  |  |  |  *  Anthony Liguori   <aliguori@us.ibm.com> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This work is licensed under the terms of the GNU GPL, version 2.  See | 
					
						
							|  |  |  |  * the COPYING file in the top-level directory. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "qemu-common.h"
 | 
					
						
							|  |  |  | #include "qemu_socket.h"
 | 
					
						
							|  |  |  | #include "migration.h"
 | 
					
						
							|  |  |  | #include "qemu-char.h"
 | 
					
						
							|  |  |  | #include "sysemu.h"
 | 
					
						
							|  |  |  | #include "buffered_file.h"
 | 
					
						
							|  |  |  | #include "block.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //#define DEBUG_MIGRATION_TCP
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef DEBUG_MIGRATION_TCP
 | 
					
						
							|  |  |  | #define dprintf(fmt, ...) \
 | 
					
						
							|  |  |  |     do { printf("migration-tcp: " fmt, ## __VA_ARGS__); } while (0) | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #define dprintf(fmt, ...) \
 | 
					
						
							|  |  |  |     do { } while (0) | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-11 16:46:33 +00:00
										 |  |  | static int socket_errno(FdMigrationState *s) | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2008-11-12 22:29:11 +00:00
										 |  |  |     return socket_error(); | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-11 16:46:33 +00:00
										 |  |  | static int socket_write(FdMigrationState *s, const void * buf, size_t size) | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2008-11-11 16:46:33 +00:00
										 |  |  |     return send(s->fd, buf, size, 0); | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-11 16:46:33 +00:00
										 |  |  | static int tcp_close(FdMigrationState *s) | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2008-11-11 16:46:33 +00:00
										 |  |  |     dprintf("tcp_close\n"); | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  |     if (s->fd != -1) { | 
					
						
							| 
									
										
										
										
											2008-10-24 22:10:31 +00:00
										 |  |  |         close(s->fd); | 
					
						
							|  |  |  |         s->fd = -1; | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void tcp_wait_for_connect(void *opaque) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     FdMigrationState *s = opaque; | 
					
						
							|  |  |  |     int val, ret; | 
					
						
							| 
									
										
										
										
											2008-10-25 11:25:48 +00:00
										 |  |  |     socklen_t valsize = sizeof(val); | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     dprintf("connect completed\n"); | 
					
						
							|  |  |  |     do { | 
					
						
							|  |  |  |         ret = getsockopt(s->fd, SOL_SOCKET, SO_ERROR, &val, &valsize); | 
					
						
							| 
									
										
										
										
											2008-11-11 16:46:33 +00:00
										 |  |  |     } while (ret == -1 && (s->get_error(s)) == EINTR); | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (ret < 0) { | 
					
						
							| 
									
										
										
										
											2008-11-11 16:46:33 +00:00
										 |  |  |         migrate_fd_error(s); | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     qemu_set_fd_handler2(s->fd, NULL, NULL, NULL, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (val == 0) | 
					
						
							| 
									
										
										
										
											2008-11-11 16:46:33 +00:00
										 |  |  |         migrate_fd_connect(s); | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  |     else { | 
					
						
							|  |  |  |         dprintf("error connecting %d\n", val); | 
					
						
							| 
									
										
										
										
											2008-11-11 16:46:33 +00:00
										 |  |  |         migrate_fd_error(s); | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | MigrationState *tcp_start_outgoing_migration(const char *host_port, | 
					
						
							| 
									
										
										
										
											2008-10-24 22:10:31 +00:00
										 |  |  |                                              int64_t bandwidth_limit, | 
					
						
							| 
									
										
										
										
											2009-03-05 23:01:42 +00:00
										 |  |  |                                              int detach) | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     struct sockaddr_in addr; | 
					
						
							|  |  |  |     FdMigrationState *s; | 
					
						
							|  |  |  |     int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (parse_host_port(&addr, host_port) < 0) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s = qemu_mallocz(sizeof(*s)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-11 16:46:33 +00:00
										 |  |  |     s->get_error = socket_errno; | 
					
						
							|  |  |  |     s->write = socket_write; | 
					
						
							|  |  |  |     s->close = tcp_close; | 
					
						
							|  |  |  |     s->mig_state.cancel = migrate_fd_cancel; | 
					
						
							|  |  |  |     s->mig_state.get_status = migrate_fd_get_status; | 
					
						
							|  |  |  |     s->mig_state.release = migrate_fd_release; | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     s->state = MIG_STATE_ACTIVE; | 
					
						
							| 
									
										
										
										
											2009-03-05 23:01:42 +00:00
										 |  |  |     s->mon_resume = NULL; | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  |     s->bandwidth_limit = bandwidth_limit; | 
					
						
							|  |  |  |     s->fd = socket(PF_INET, SOCK_STREAM, 0); | 
					
						
							|  |  |  |     if (s->fd == -1) { | 
					
						
							|  |  |  |         qemu_free(s); | 
					
						
							| 
									
										
										
										
											2008-10-24 22:10:31 +00:00
										 |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-24 14:11:41 +00:00
										 |  |  |     socket_set_nonblock(s->fd); | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-05 23:01:42 +00:00
										 |  |  |     if (!detach) | 
					
						
							|  |  |  |         migrate_fd_monitor_suspend(s); | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     do { | 
					
						
							|  |  |  |         ret = connect(s->fd, (struct sockaddr *)&addr, sizeof(addr)); | 
					
						
							|  |  |  |         if (ret == -1) | 
					
						
							| 
									
										
										
										
											2008-11-11 16:46:33 +00:00
										 |  |  |             ret = -(s->get_error(s)); | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-24 21:55:17 +00:00
										 |  |  |         if (ret == -EINPROGRESS || ret == -EWOULDBLOCK) | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  |             qemu_set_fd_handler2(s->fd, NULL, NULL, tcp_wait_for_connect, s); | 
					
						
							|  |  |  |     } while (ret == -EINTR); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-24 21:55:17 +00:00
										 |  |  |     if (ret < 0 && ret != -EINPROGRESS && ret != -EWOULDBLOCK) { | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  |         dprintf("connect failed\n"); | 
					
						
							|  |  |  |         close(s->fd); | 
					
						
							|  |  |  |         qemu_free(s); | 
					
						
							| 
									
										
										
										
											2008-10-24 22:08:22 +00:00
										 |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  |     } else if (ret >= 0) | 
					
						
							| 
									
										
										
										
											2008-11-11 16:46:33 +00:00
										 |  |  |         migrate_fd_connect(s); | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return &s->mig_state; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void tcp_accept_incoming_migration(void *opaque) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct sockaddr_in addr; | 
					
						
							|  |  |  |     socklen_t addrlen = sizeof(addr); | 
					
						
							|  |  |  |     int s = (unsigned long)opaque; | 
					
						
							|  |  |  |     QEMUFile *f; | 
					
						
							|  |  |  |     int c, ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     do { | 
					
						
							|  |  |  |         c = accept(s, (struct sockaddr *)&addr, &addrlen); | 
					
						
							| 
									
										
										
										
											2008-10-24 21:55:17 +00:00
										 |  |  |     } while (c == -1 && socket_error() == EINTR); | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     dprintf("accepted migration\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (c == -1) { | 
					
						
							|  |  |  |         fprintf(stderr, "could not accept migration connection\n"); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-24 21:55:17 +00:00
										 |  |  |     f = qemu_fopen_socket(c); | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  |     if (f == NULL) { | 
					
						
							|  |  |  |         fprintf(stderr, "could not qemu_fopen socket\n"); | 
					
						
							|  |  |  |         goto out; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vm_stop(0); /* just in case */ | 
					
						
							|  |  |  |     ret = qemu_loadvm_state(f); | 
					
						
							|  |  |  |     if (ret < 0) { | 
					
						
							|  |  |  |         fprintf(stderr, "load of migration failed\n"); | 
					
						
							|  |  |  |         goto out_fopen; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     qemu_announce_self(); | 
					
						
							|  |  |  |     dprintf("successfully loaded vm state\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* we've successfully migrated, close the server socket */ | 
					
						
							|  |  |  |     qemu_set_fd_handler2(s, NULL, NULL, NULL, NULL); | 
					
						
							|  |  |  |     close(s); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     vm_start(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | out_fopen: | 
					
						
							|  |  |  |     qemu_fclose(f); | 
					
						
							|  |  |  | out: | 
					
						
							|  |  |  |     close(c); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int tcp_start_incoming_migration(const char *host_port) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     struct sockaddr_in addr; | 
					
						
							|  |  |  |     int val; | 
					
						
							|  |  |  |     int s; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (parse_host_port(&addr, host_port) < 0) { | 
					
						
							|  |  |  |         fprintf(stderr, "invalid host/port combination: %s\n", host_port); | 
					
						
							|  |  |  |         return -EINVAL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s = socket(PF_INET, SOCK_STREAM, 0); | 
					
						
							|  |  |  |     if (s == -1) | 
					
						
							| 
									
										
										
										
											2008-10-24 21:55:17 +00:00
										 |  |  |         return -socket_error(); | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     val = 1; | 
					
						
							|  |  |  |     setsockopt(s, SOL_SOCKET, SO_REUSEADDR, (const char *)&val, sizeof(val)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (bind(s, (struct sockaddr *)&addr, sizeof(addr)) == -1) | 
					
						
							|  |  |  |         goto err; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (listen(s, 1) == -1) | 
					
						
							|  |  |  |         goto err; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     qemu_set_fd_handler2(s, NULL, tcp_accept_incoming_migration, NULL, | 
					
						
							|  |  |  |                          (void *)(unsigned long)s); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | err: | 
					
						
							|  |  |  |     close(s); | 
					
						
							| 
									
										
										
										
											2008-10-24 21:55:17 +00:00
										 |  |  |     return -socket_error(); | 
					
						
							| 
									
										
										
										
											2008-10-13 03:14:31 +00:00
										 |  |  | } |