2008-10-13 05:12:02 +02: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 "migration.h"
|
|
|
|
#include "console.h"
|
|
|
|
|
|
|
|
/* Migration speed throttling */
|
|
|
|
static uint32_t max_throttle = (32 << 20);
|
|
|
|
|
|
|
|
static MigrationState *current_migration;
|
|
|
|
|
|
|
|
void qemu_start_incoming_migration(const char *uri)
|
|
|
|
{
|
2008-10-13 05:14:31 +02:00
|
|
|
const char *p;
|
|
|
|
|
|
|
|
if (strstart(uri, "tcp:", &p))
|
|
|
|
tcp_start_incoming_migration(p);
|
|
|
|
else
|
|
|
|
fprintf(stderr, "unknown migration protocol: %s\n", uri);
|
2008-10-13 05:12:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void do_migrate(int detach, const char *uri)
|
|
|
|
{
|
2008-10-13 05:14:31 +02:00
|
|
|
MigrationState *s = NULL;
|
|
|
|
const char *p;
|
|
|
|
|
|
|
|
if (strstart(uri, "tcp:", &p))
|
2008-10-25 00:10:31 +02:00
|
|
|
s = tcp_start_outgoing_migration(p, max_throttle, detach);
|
2008-10-13 05:14:31 +02:00
|
|
|
else
|
|
|
|
term_printf("unknown migration protocol: %s\n", uri);
|
|
|
|
|
|
|
|
if (s == NULL)
|
2008-10-25 00:10:31 +02:00
|
|
|
term_printf("migration failed\n");
|
2008-10-13 05:14:31 +02:00
|
|
|
else {
|
2008-10-25 00:10:31 +02:00
|
|
|
if (current_migration)
|
|
|
|
current_migration->release(current_migration);
|
2008-10-13 05:14:31 +02:00
|
|
|
|
2008-10-25 00:10:31 +02:00
|
|
|
current_migration = s;
|
2008-10-13 05:14:31 +02:00
|
|
|
}
|
2008-10-13 05:12:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void do_migrate_cancel(void)
|
|
|
|
{
|
|
|
|
MigrationState *s = current_migration;
|
|
|
|
|
|
|
|
if (s)
|
2008-10-25 00:10:31 +02:00
|
|
|
s->cancel(s);
|
2008-10-13 05:12:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void do_migrate_set_speed(const char *value)
|
|
|
|
{
|
|
|
|
double d;
|
|
|
|
char *ptr;
|
|
|
|
|
|
|
|
d = strtod(value, &ptr);
|
|
|
|
switch (*ptr) {
|
|
|
|
case 'G': case 'g':
|
2008-10-25 00:10:31 +02:00
|
|
|
d *= 1024;
|
2008-10-13 05:12:02 +02:00
|
|
|
case 'M': case 'm':
|
2008-10-25 00:10:31 +02:00
|
|
|
d *= 1024;
|
2008-10-13 05:12:02 +02:00
|
|
|
case 'K': case 'k':
|
2008-10-25 00:10:31 +02:00
|
|
|
d *= 1024;
|
2008-10-13 05:12:02 +02:00
|
|
|
default:
|
2008-10-25 00:10:31 +02:00
|
|
|
break;
|
2008-10-13 05:12:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
max_throttle = (uint32_t)d;
|
|
|
|
}
|
|
|
|
|
|
|
|
void do_info_migrate(void)
|
|
|
|
{
|
|
|
|
MigrationState *s = current_migration;
|
|
|
|
|
|
|
|
if (s) {
|
2008-10-25 00:10:31 +02:00
|
|
|
term_printf("Migration status: ");
|
|
|
|
switch (s->get_status(s)) {
|
|
|
|
case MIG_STATE_ACTIVE:
|
|
|
|
term_printf("active\n");
|
|
|
|
break;
|
|
|
|
case MIG_STATE_COMPLETED:
|
|
|
|
term_printf("completed\n");
|
|
|
|
break;
|
|
|
|
case MIG_STATE_ERROR:
|
|
|
|
term_printf("failed\n");
|
|
|
|
break;
|
|
|
|
case MIG_STATE_CANCELLED:
|
|
|
|
term_printf("cancelled\n");
|
|
|
|
break;
|
|
|
|
}
|
2008-10-13 05:12:02 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|