2016-10-27 14:42:53 +08:00
|
|
|
/*
|
|
|
|
* COarse-grain LOck-stepping Virtual Machines for Non-stop Service (COLO)
|
|
|
|
* (a.k.a. Fault Tolerance or Continuous Replication)
|
|
|
|
*
|
|
|
|
* Copyright (c) 2016 HUAWEI TECHNOLOGIES CO., LTD.
|
|
|
|
* Copyright (c) 2016 FUJITSU LIMITED
|
|
|
|
* Copyright (c) 2016 Intel Corporation
|
|
|
|
*
|
|
|
|
* This work is licensed under the terms of the GNU GPL, version 2 or
|
|
|
|
* later. See the COPYING file in the top-level directory.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "qemu/osdep.h"
|
2017-04-25 02:07:27 +08:00
|
|
|
#include "migration.h"
|
2017-04-06 18:03:13 +08:00
|
|
|
#include "migration/colo.h"
|
2017-04-18 01:02:59 +08:00
|
|
|
#include "migration/vmstate.h"
|
2016-10-27 14:42:53 +08:00
|
|
|
#include "trace.h"
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
bool colo_requested;
|
|
|
|
} COLOInfo;
|
|
|
|
|
|
|
|
static COLOInfo colo_info;
|
|
|
|
|
COLO: Add 'x-colo-lost-heartbeat' command to trigger failover
We leave users to choose whatever heartbeat solution they want,
if the heartbeat is lost, or other errors they detect, they can use
experimental command 'x_colo_lost_heartbeat' to tell COLO to do failover,
COLO will do operations accordingly.
For example, if the command is sent to the Primary side,
the Primary side will exit COLO mode, does cleanup work,
and then, PVM will take over the service work. If sent to the Secondary side,
the Secondary side will run failover work, then takes over PVM's service work.
Cc: Luiz Capitulino <lcapitulino@redhat.com>
Cc: Eric Blake <eblake@redhat.com>
Cc: Markus Armbruster <armbru@redhat.com>
Signed-off-by: zhanghailiang <zhang.zhanghailiang@huawei.com>
Signed-off-by: Li Zhijian <lizhijian@cn.fujitsu.com>
Reviewed-by: Dr. David Alan Gilbert <dgilbert@redhat.com>
Reviewed-by: Amit Shah <amit.shah@redhat.com>
Signed-off-by: Amit Shah <amit@amitshah.net>
2016-10-27 14:43:03 +08:00
|
|
|
COLOMode get_colo_mode(void)
|
|
|
|
{
|
|
|
|
if (migration_in_colo_state()) {
|
|
|
|
return COLO_MODE_PRIMARY;
|
|
|
|
} else if (migration_incoming_in_colo_state()) {
|
|
|
|
return COLO_MODE_SECONDARY;
|
|
|
|
} else {
|
|
|
|
return COLO_MODE_UNKNOWN;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-25 19:29:12 +08:00
|
|
|
static int colo_info_pre_save(void *opaque)
|
2016-10-27 14:42:53 +08:00
|
|
|
{
|
|
|
|
COLOInfo *s = opaque;
|
|
|
|
|
|
|
|
s->colo_requested = migrate_colo_enabled();
|
2017-09-25 19:29:12 +08:00
|
|
|
|
|
|
|
return 0;
|
2016-10-27 14:42:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool colo_info_need(void *opaque)
|
|
|
|
{
|
|
|
|
return migrate_colo_enabled();
|
|
|
|
}
|
|
|
|
|
|
|
|
static const VMStateDescription colo_state = {
|
|
|
|
.name = "COLOState",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.pre_save = colo_info_pre_save,
|
|
|
|
.needed = colo_info_need,
|
|
|
|
.fields = (VMStateField[]) {
|
|
|
|
VMSTATE_BOOL(colo_requested, COLOInfo),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
void colo_info_init(void)
|
|
|
|
{
|
|
|
|
vmstate_register(NULL, 0, &colo_state, &colo_info);
|
|
|
|
}
|
2016-10-27 14:42:55 +08:00
|
|
|
|
|
|
|
bool migration_incoming_enable_colo(void)
|
|
|
|
{
|
|
|
|
return colo_info.colo_requested;
|
|
|
|
}
|
|
|
|
|
|
|
|
void migration_incoming_exit_colo(void)
|
|
|
|
{
|
|
|
|
colo_info.colo_requested = false;
|
|
|
|
}
|