root / migration-fd.c @ cca1af8c
History | View | Annotate | Download (3.1 kB)
1 | 5ac1fad3 | Paolo Bonzini | /*
|
---|---|---|---|
2 | 5ac1fad3 | Paolo Bonzini | * QEMU live migration via generic fd
|
3 | 5ac1fad3 | Paolo Bonzini | *
|
4 | 5ac1fad3 | Paolo Bonzini | * Copyright Red Hat, Inc. 2009
|
5 | 5ac1fad3 | Paolo Bonzini | *
|
6 | 5ac1fad3 | Paolo Bonzini | * Authors:
|
7 | 5ac1fad3 | Paolo Bonzini | * Chris Lalancette <clalance@redhat.com>
|
8 | 5ac1fad3 | Paolo Bonzini | *
|
9 | 5ac1fad3 | Paolo Bonzini | * This work is licensed under the terms of the GNU GPL, version 2. See
|
10 | 5ac1fad3 | Paolo Bonzini | * the COPYING file in the top-level directory.
|
11 | 5ac1fad3 | Paolo Bonzini | *
|
12 | 5ac1fad3 | Paolo Bonzini | */
|
13 | 5ac1fad3 | Paolo Bonzini | |
14 | 5ac1fad3 | Paolo Bonzini | #include "qemu-common.h" |
15 | 5ac1fad3 | Paolo Bonzini | #include "qemu_socket.h" |
16 | 5ac1fad3 | Paolo Bonzini | #include "migration.h" |
17 | 5ac1fad3 | Paolo Bonzini | #include "monitor.h" |
18 | 5ac1fad3 | Paolo Bonzini | #include "qemu-char.h" |
19 | 5ac1fad3 | Paolo Bonzini | #include "sysemu.h" |
20 | 5ac1fad3 | Paolo Bonzini | #include "buffered_file.h" |
21 | 5ac1fad3 | Paolo Bonzini | #include "block.h" |
22 | 5ac1fad3 | Paolo Bonzini | #include "qemu_socket.h" |
23 | 5ac1fad3 | Paolo Bonzini | |
24 | 5ac1fad3 | Paolo Bonzini | //#define DEBUG_MIGRATION_FD
|
25 | 5ac1fad3 | Paolo Bonzini | |
26 | 5ac1fad3 | Paolo Bonzini | #ifdef DEBUG_MIGRATION_FD
|
27 | d0f2c4c6 | malc | #define DPRINTF(fmt, ...) \
|
28 | 5ac1fad3 | Paolo Bonzini | do { printf("migration-fd: " fmt, ## __VA_ARGS__); } while (0) |
29 | 5ac1fad3 | Paolo Bonzini | #else
|
30 | d0f2c4c6 | malc | #define DPRINTF(fmt, ...) \
|
31 | 5ac1fad3 | Paolo Bonzini | do { } while (0) |
32 | 5ac1fad3 | Paolo Bonzini | #endif
|
33 | 5ac1fad3 | Paolo Bonzini | |
34 | 5ac1fad3 | Paolo Bonzini | static int fd_errno(FdMigrationState *s) |
35 | 5ac1fad3 | Paolo Bonzini | { |
36 | 5ac1fad3 | Paolo Bonzini | return errno;
|
37 | 5ac1fad3 | Paolo Bonzini | } |
38 | 5ac1fad3 | Paolo Bonzini | |
39 | 5ac1fad3 | Paolo Bonzini | static int fd_write(FdMigrationState *s, const void * buf, size_t size) |
40 | 5ac1fad3 | Paolo Bonzini | { |
41 | 5ac1fad3 | Paolo Bonzini | return write(s->fd, buf, size);
|
42 | 5ac1fad3 | Paolo Bonzini | } |
43 | 5ac1fad3 | Paolo Bonzini | |
44 | 5ac1fad3 | Paolo Bonzini | static int fd_close(FdMigrationState *s) |
45 | 5ac1fad3 | Paolo Bonzini | { |
46 | d0f2c4c6 | malc | DPRINTF("fd_close\n");
|
47 | 5ac1fad3 | Paolo Bonzini | if (s->fd != -1) { |
48 | 5ac1fad3 | Paolo Bonzini | close(s->fd); |
49 | 5ac1fad3 | Paolo Bonzini | s->fd = -1;
|
50 | 5ac1fad3 | Paolo Bonzini | } |
51 | 5ac1fad3 | Paolo Bonzini | return 0; |
52 | 5ac1fad3 | Paolo Bonzini | } |
53 | 5ac1fad3 | Paolo Bonzini | |
54 | 5ac1fad3 | Paolo Bonzini | MigrationState *fd_start_outgoing_migration(Monitor *mon, |
55 | 5ac1fad3 | Paolo Bonzini | const char *fdname, |
56 | 5ac1fad3 | Paolo Bonzini | int64_t bandwidth_limit, |
57 | c163b5ca | lirans@il.ibm.com | int detach,
|
58 | c163b5ca | lirans@il.ibm.com | int blk,
|
59 | c163b5ca | lirans@il.ibm.com | int inc)
|
60 | 5ac1fad3 | Paolo Bonzini | { |
61 | 5ac1fad3 | Paolo Bonzini | FdMigrationState *s; |
62 | 5ac1fad3 | Paolo Bonzini | |
63 | 5ac1fad3 | Paolo Bonzini | s = qemu_mallocz(sizeof(*s));
|
64 | 5ac1fad3 | Paolo Bonzini | |
65 | 5ac1fad3 | Paolo Bonzini | s->fd = monitor_get_fd(mon, fdname); |
66 | 5ac1fad3 | Paolo Bonzini | if (s->fd == -1) { |
67 | d0f2c4c6 | malc | DPRINTF("fd_migration: invalid file descriptor identifier\n");
|
68 | 5ac1fad3 | Paolo Bonzini | goto err_after_alloc;
|
69 | 5ac1fad3 | Paolo Bonzini | } |
70 | 5ac1fad3 | Paolo Bonzini | |
71 | 5ac1fad3 | Paolo Bonzini | if (fcntl(s->fd, F_SETFL, O_NONBLOCK) == -1) { |
72 | d0f2c4c6 | malc | DPRINTF("Unable to set nonblocking mode on file descriptor\n");
|
73 | 5ac1fad3 | Paolo Bonzini | goto err_after_open;
|
74 | 5ac1fad3 | Paolo Bonzini | } |
75 | 5ac1fad3 | Paolo Bonzini | |
76 | 5ac1fad3 | Paolo Bonzini | s->get_error = fd_errno; |
77 | 5ac1fad3 | Paolo Bonzini | s->write = fd_write; |
78 | 5ac1fad3 | Paolo Bonzini | s->close = fd_close; |
79 | 5ac1fad3 | Paolo Bonzini | s->mig_state.cancel = migrate_fd_cancel; |
80 | 5ac1fad3 | Paolo Bonzini | s->mig_state.get_status = migrate_fd_get_status; |
81 | 5ac1fad3 | Paolo Bonzini | s->mig_state.release = migrate_fd_release; |
82 | 5ac1fad3 | Paolo Bonzini | |
83 | c163b5ca | lirans@il.ibm.com | s->mig_state.blk = blk; |
84 | c163b5ca | lirans@il.ibm.com | s->mig_state.shared = inc; |
85 | f327aa0c | Jan Kiszka | |
86 | 5ac1fad3 | Paolo Bonzini | s->state = MIG_STATE_ACTIVE; |
87 | f327aa0c | Jan Kiszka | s->mon = NULL;
|
88 | 5ac1fad3 | Paolo Bonzini | s->bandwidth_limit = bandwidth_limit; |
89 | 5ac1fad3 | Paolo Bonzini | |
90 | f327aa0c | Jan Kiszka | if (!detach) {
|
91 | f327aa0c | Jan Kiszka | migrate_fd_monitor_suspend(s, mon); |
92 | f327aa0c | Jan Kiszka | } |
93 | 5ac1fad3 | Paolo Bonzini | |
94 | 5ac1fad3 | Paolo Bonzini | migrate_fd_connect(s); |
95 | 5ac1fad3 | Paolo Bonzini | return &s->mig_state;
|
96 | 5ac1fad3 | Paolo Bonzini | |
97 | 5ac1fad3 | Paolo Bonzini | err_after_open:
|
98 | 5ac1fad3 | Paolo Bonzini | close(s->fd); |
99 | 5ac1fad3 | Paolo Bonzini | err_after_alloc:
|
100 | 5ac1fad3 | Paolo Bonzini | qemu_free(s); |
101 | 5ac1fad3 | Paolo Bonzini | return NULL; |
102 | 5ac1fad3 | Paolo Bonzini | } |
103 | 5ac1fad3 | Paolo Bonzini | |
104 | 5ac1fad3 | Paolo Bonzini | static void fd_accept_incoming_migration(void *opaque) |
105 | 5ac1fad3 | Paolo Bonzini | { |
106 | 5ac1fad3 | Paolo Bonzini | QEMUFile *f = opaque; |
107 | 5ac1fad3 | Paolo Bonzini | int ret;
|
108 | 5ac1fad3 | Paolo Bonzini | |
109 | 5ac1fad3 | Paolo Bonzini | ret = qemu_loadvm_state(f); |
110 | 5ac1fad3 | Paolo Bonzini | if (ret < 0) { |
111 | 5ac1fad3 | Paolo Bonzini | fprintf(stderr, "load of migration failed\n");
|
112 | 5ac1fad3 | Paolo Bonzini | goto err;
|
113 | 5ac1fad3 | Paolo Bonzini | } |
114 | 5ac1fad3 | Paolo Bonzini | qemu_announce_self(); |
115 | d0f2c4c6 | malc | DPRINTF("successfully loaded vm state\n");
|
116 | 5ac1fad3 | Paolo Bonzini | /* we've successfully migrated, close the fd */
|
117 | 5ac1fad3 | Paolo Bonzini | qemu_set_fd_handler2(qemu_stdio_fd(f), NULL, NULL, NULL, NULL); |
118 | 5ac1fad3 | Paolo Bonzini | if (autostart)
|
119 | 5ac1fad3 | Paolo Bonzini | vm_start(); |
120 | 5ac1fad3 | Paolo Bonzini | |
121 | 5ac1fad3 | Paolo Bonzini | err:
|
122 | 5ac1fad3 | Paolo Bonzini | qemu_fclose(f); |
123 | 5ac1fad3 | Paolo Bonzini | } |
124 | 5ac1fad3 | Paolo Bonzini | |
125 | 5ac1fad3 | Paolo Bonzini | int fd_start_incoming_migration(const char *infd) |
126 | 5ac1fad3 | Paolo Bonzini | { |
127 | 5ac1fad3 | Paolo Bonzini | int fd;
|
128 | 5ac1fad3 | Paolo Bonzini | QEMUFile *f; |
129 | 5ac1fad3 | Paolo Bonzini | |
130 | d0f2c4c6 | malc | DPRINTF("Attempting to start an incoming migration via fd\n");
|
131 | 5ac1fad3 | Paolo Bonzini | |
132 | 5ac1fad3 | Paolo Bonzini | fd = strtol(infd, NULL, 0); |
133 | 5ac1fad3 | Paolo Bonzini | f = qemu_fdopen(fd, "rb");
|
134 | 5ac1fad3 | Paolo Bonzini | if(f == NULL) { |
135 | d0f2c4c6 | malc | DPRINTF("Unable to apply qemu wrapper to file descriptor\n");
|
136 | 5ac1fad3 | Paolo Bonzini | return -errno;
|
137 | 5ac1fad3 | Paolo Bonzini | } |
138 | 5ac1fad3 | Paolo Bonzini | |
139 | 5ac1fad3 | Paolo Bonzini | qemu_set_fd_handler2(fd, NULL, fd_accept_incoming_migration, NULL, |
140 | 5ac1fad3 | Paolo Bonzini | (void *)(unsigned long)f); |
141 | 5ac1fad3 | Paolo Bonzini | |
142 | 5ac1fad3 | Paolo Bonzini | return 0; |
143 | 5ac1fad3 | Paolo Bonzini | } |