|
@@ -16,6 +16,7 @@
|
|
#include "cpu.h"
|
|
#include "cpu.h"
|
|
#include "exec/ramblock.h"
|
|
#include "exec/ramblock.h"
|
|
#include "qemu/rcu_queue.h"
|
|
#include "qemu/rcu_queue.h"
|
|
|
|
+#include "qemu/main-loop.h"
|
|
#include "qapi/qapi-commands-migration.h"
|
|
#include "qapi/qapi-commands-migration.h"
|
|
#include "ram.h"
|
|
#include "ram.h"
|
|
#include "trace.h"
|
|
#include "trace.h"
|
|
@@ -23,9 +24,19 @@
|
|
#include "monitor/hmp.h"
|
|
#include "monitor/hmp.h"
|
|
#include "monitor/monitor.h"
|
|
#include "monitor/monitor.h"
|
|
#include "qapi/qmp/qdict.h"
|
|
#include "qapi/qmp/qdict.h"
|
|
|
|
+#include "sysemu/kvm.h"
|
|
|
|
+#include "sysemu/runstate.h"
|
|
|
|
+#include "exec/memory.h"
|
|
|
|
+
|
|
|
|
+typedef struct DirtyPageRecord {
|
|
|
|
+ uint64_t start_pages;
|
|
|
|
+ uint64_t end_pages;
|
|
|
|
+} DirtyPageRecord;
|
|
|
|
|
|
static int CalculatingState = DIRTY_RATE_STATUS_UNSTARTED;
|
|
static int CalculatingState = DIRTY_RATE_STATUS_UNSTARTED;
|
|
static struct DirtyRateStat DirtyStat;
|
|
static struct DirtyRateStat DirtyStat;
|
|
|
|
+static DirtyRateMeasureMode dirtyrate_mode =
|
|
|
|
+ DIRTY_RATE_MEASURE_MODE_PAGE_SAMPLING;
|
|
|
|
|
|
static int64_t set_sample_page_period(int64_t msec, int64_t initial_time)
|
|
static int64_t set_sample_page_period(int64_t msec, int64_t initial_time)
|
|
{
|
|
{
|
|
@@ -70,18 +81,37 @@ static int dirtyrate_set_state(int *state, int old_state, int new_state)
|
|
|
|
|
|
static struct DirtyRateInfo *query_dirty_rate_info(void)
|
|
static struct DirtyRateInfo *query_dirty_rate_info(void)
|
|
{
|
|
{
|
|
|
|
+ int i;
|
|
int64_t dirty_rate = DirtyStat.dirty_rate;
|
|
int64_t dirty_rate = DirtyStat.dirty_rate;
|
|
struct DirtyRateInfo *info = g_malloc0(sizeof(DirtyRateInfo));
|
|
struct DirtyRateInfo *info = g_malloc0(sizeof(DirtyRateInfo));
|
|
-
|
|
|
|
- if (qatomic_read(&CalculatingState) == DIRTY_RATE_STATUS_MEASURED) {
|
|
|
|
- info->has_dirty_rate = true;
|
|
|
|
- info->dirty_rate = dirty_rate;
|
|
|
|
- }
|
|
|
|
|
|
+ DirtyRateVcpuList *head = NULL, **tail = &head;
|
|
|
|
|
|
info->status = CalculatingState;
|
|
info->status = CalculatingState;
|
|
info->start_time = DirtyStat.start_time;
|
|
info->start_time = DirtyStat.start_time;
|
|
info->calc_time = DirtyStat.calc_time;
|
|
info->calc_time = DirtyStat.calc_time;
|
|
info->sample_pages = DirtyStat.sample_pages;
|
|
info->sample_pages = DirtyStat.sample_pages;
|
|
|
|
+ info->mode = dirtyrate_mode;
|
|
|
|
+
|
|
|
|
+ if (qatomic_read(&CalculatingState) == DIRTY_RATE_STATUS_MEASURED) {
|
|
|
|
+ info->has_dirty_rate = true;
|
|
|
|
+ info->dirty_rate = dirty_rate;
|
|
|
|
+
|
|
|
|
+ if (dirtyrate_mode == DIRTY_RATE_MEASURE_MODE_DIRTY_RING) {
|
|
|
|
+ /*
|
|
|
|
+ * set sample_pages with 0 to indicate page sampling
|
|
|
|
+ * isn't enabled
|
|
|
|
+ **/
|
|
|
|
+ info->sample_pages = 0;
|
|
|
|
+ info->has_vcpu_dirty_rate = true;
|
|
|
|
+ for (i = 0; i < DirtyStat.dirty_ring.nvcpu; i++) {
|
|
|
|
+ DirtyRateVcpu *rate = g_malloc0(sizeof(DirtyRateVcpu));
|
|
|
|
+ rate->id = DirtyStat.dirty_ring.rates[i].id;
|
|
|
|
+ rate->dirty_rate = DirtyStat.dirty_ring.rates[i].dirty_rate;
|
|
|
|
+ QAPI_LIST_APPEND(tail, rate);
|
|
|
|
+ }
|
|
|
|
+ info->vcpu_dirty_rate = head;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
|
|
trace_query_dirty_rate_info(DirtyRateStatus_str(CalculatingState));
|
|
trace_query_dirty_rate_info(DirtyRateStatus_str(CalculatingState));
|
|
|
|
|
|
@@ -111,6 +141,15 @@ static void init_dirtyrate_stat(int64_t start_time,
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+static void cleanup_dirtyrate_stat(struct DirtyRateConfig config)
|
|
|
|
+{
|
|
|
|
+ /* last calc-dirty-rate qmp use dirty ring mode */
|
|
|
|
+ if (dirtyrate_mode == DIRTY_RATE_MEASURE_MODE_DIRTY_RING) {
|
|
|
|
+ free(DirtyStat.dirty_ring.rates);
|
|
|
|
+ DirtyStat.dirty_ring.rates = NULL;
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
static void update_dirtyrate_stat(struct RamblockDirtyInfo *info)
|
|
static void update_dirtyrate_stat(struct RamblockDirtyInfo *info)
|
|
{
|
|
{
|
|
DirtyStat.page_sampling.total_dirty_samples += info->sample_dirty_count;
|
|
DirtyStat.page_sampling.total_dirty_samples += info->sample_dirty_count;
|
|
@@ -345,7 +384,97 @@ static bool compare_page_hash_info(struct RamblockDirtyInfo *info,
|
|
return true;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
|
|
-static void calculate_dirtyrate(struct DirtyRateConfig config)
|
|
|
|
|
|
+static inline void record_dirtypages(DirtyPageRecord *dirty_pages,
|
|
|
|
+ CPUState *cpu, bool start)
|
|
|
|
+{
|
|
|
|
+ if (start) {
|
|
|
|
+ dirty_pages[cpu->cpu_index].start_pages = cpu->dirty_pages;
|
|
|
|
+ } else {
|
|
|
|
+ dirty_pages[cpu->cpu_index].end_pages = cpu->dirty_pages;
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static void dirtyrate_global_dirty_log_start(void)
|
|
|
|
+{
|
|
|
|
+ qemu_mutex_lock_iothread();
|
|
|
|
+ memory_global_dirty_log_start(GLOBAL_DIRTY_DIRTY_RATE);
|
|
|
|
+ qemu_mutex_unlock_iothread();
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static void dirtyrate_global_dirty_log_stop(void)
|
|
|
|
+{
|
|
|
|
+ qemu_mutex_lock_iothread();
|
|
|
|
+ memory_global_dirty_log_sync();
|
|
|
|
+ memory_global_dirty_log_stop(GLOBAL_DIRTY_DIRTY_RATE);
|
|
|
|
+ qemu_mutex_unlock_iothread();
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static int64_t do_calculate_dirtyrate_vcpu(DirtyPageRecord dirty_pages)
|
|
|
|
+{
|
|
|
|
+ uint64_t memory_size_MB;
|
|
|
|
+ int64_t time_s;
|
|
|
|
+ uint64_t increased_dirty_pages =
|
|
|
|
+ dirty_pages.end_pages - dirty_pages.start_pages;
|
|
|
|
+
|
|
|
|
+ memory_size_MB = (increased_dirty_pages * TARGET_PAGE_SIZE) >> 20;
|
|
|
|
+ time_s = DirtyStat.calc_time;
|
|
|
|
+
|
|
|
|
+ return memory_size_MB / time_s;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static void calculate_dirtyrate_dirty_ring(struct DirtyRateConfig config)
|
|
|
|
+{
|
|
|
|
+ CPUState *cpu;
|
|
|
|
+ int64_t msec = 0;
|
|
|
|
+ int64_t start_time;
|
|
|
|
+ uint64_t dirtyrate = 0;
|
|
|
|
+ uint64_t dirtyrate_sum = 0;
|
|
|
|
+ DirtyPageRecord *dirty_pages;
|
|
|
|
+ int nvcpu = 0;
|
|
|
|
+ int i = 0;
|
|
|
|
+
|
|
|
|
+ CPU_FOREACH(cpu) {
|
|
|
|
+ nvcpu++;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ dirty_pages = malloc(sizeof(*dirty_pages) * nvcpu);
|
|
|
|
+
|
|
|
|
+ DirtyStat.dirty_ring.nvcpu = nvcpu;
|
|
|
|
+ DirtyStat.dirty_ring.rates = malloc(sizeof(DirtyRateVcpu) * nvcpu);
|
|
|
|
+
|
|
|
|
+ dirtyrate_global_dirty_log_start();
|
|
|
|
+
|
|
|
|
+ CPU_FOREACH(cpu) {
|
|
|
|
+ record_dirtypages(dirty_pages, cpu, true);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ start_time = qemu_clock_get_ms(QEMU_CLOCK_REALTIME);
|
|
|
|
+ DirtyStat.start_time = start_time / 1000;
|
|
|
|
+
|
|
|
|
+ msec = config.sample_period_seconds * 1000;
|
|
|
|
+ msec = set_sample_page_period(msec, start_time);
|
|
|
|
+ DirtyStat.calc_time = msec / 1000;
|
|
|
|
+
|
|
|
|
+ dirtyrate_global_dirty_log_stop();
|
|
|
|
+
|
|
|
|
+ CPU_FOREACH(cpu) {
|
|
|
|
+ record_dirtypages(dirty_pages, cpu, false);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ for (i = 0; i < DirtyStat.dirty_ring.nvcpu; i++) {
|
|
|
|
+ dirtyrate = do_calculate_dirtyrate_vcpu(dirty_pages[i]);
|
|
|
|
+ trace_dirtyrate_do_calculate_vcpu(i, dirtyrate);
|
|
|
|
+
|
|
|
|
+ DirtyStat.dirty_ring.rates[i].id = i;
|
|
|
|
+ DirtyStat.dirty_ring.rates[i].dirty_rate = dirtyrate;
|
|
|
|
+ dirtyrate_sum += dirtyrate;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ DirtyStat.dirty_rate = dirtyrate_sum;
|
|
|
|
+ free(dirty_pages);
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static void calculate_dirtyrate_sample_vm(struct DirtyRateConfig config)
|
|
{
|
|
{
|
|
struct RamblockDirtyInfo *block_dinfo = NULL;
|
|
struct RamblockDirtyInfo *block_dinfo = NULL;
|
|
int block_count = 0;
|
|
int block_count = 0;
|
|
@@ -376,6 +505,17 @@ out:
|
|
free_ramblock_dirty_info(block_dinfo, block_count);
|
|
free_ramblock_dirty_info(block_dinfo, block_count);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+static void calculate_dirtyrate(struct DirtyRateConfig config)
|
|
|
|
+{
|
|
|
|
+ if (config.mode == DIRTY_RATE_MEASURE_MODE_DIRTY_RING) {
|
|
|
|
+ calculate_dirtyrate_dirty_ring(config);
|
|
|
|
+ } else {
|
|
|
|
+ calculate_dirtyrate_sample_vm(config);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ trace_dirtyrate_calculate(DirtyStat.dirty_rate);
|
|
|
|
+}
|
|
|
|
+
|
|
void *get_dirtyrate_thread(void *arg)
|
|
void *get_dirtyrate_thread(void *arg)
|
|
{
|
|
{
|
|
struct DirtyRateConfig config = *(struct DirtyRateConfig *)arg;
|
|
struct DirtyRateConfig config = *(struct DirtyRateConfig *)arg;
|
|
@@ -401,8 +541,12 @@ void *get_dirtyrate_thread(void *arg)
|
|
return NULL;
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
|
|
-void qmp_calc_dirty_rate(int64_t calc_time, bool has_sample_pages,
|
|
|
|
- int64_t sample_pages, Error **errp)
|
|
|
|
|
|
+void qmp_calc_dirty_rate(int64_t calc_time,
|
|
|
|
+ bool has_sample_pages,
|
|
|
|
+ int64_t sample_pages,
|
|
|
|
+ bool has_mode,
|
|
|
|
+ DirtyRateMeasureMode mode,
|
|
|
|
+ Error **errp)
|
|
{
|
|
{
|
|
static struct DirtyRateConfig config;
|
|
static struct DirtyRateConfig config;
|
|
QemuThread thread;
|
|
QemuThread thread;
|
|
@@ -424,6 +568,15 @@ void qmp_calc_dirty_rate(int64_t calc_time, bool has_sample_pages,
|
|
return;
|
|
return;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+ if (!has_mode) {
|
|
|
|
+ mode = DIRTY_RATE_MEASURE_MODE_PAGE_SAMPLING;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ if (has_sample_pages && mode == DIRTY_RATE_MEASURE_MODE_DIRTY_RING) {
|
|
|
|
+ error_setg(errp, "either sample-pages or dirty-ring can be specified.");
|
|
|
|
+ return;
|
|
|
|
+ }
|
|
|
|
+
|
|
if (has_sample_pages) {
|
|
if (has_sample_pages) {
|
|
if (!is_sample_pages_valid(sample_pages)) {
|
|
if (!is_sample_pages_valid(sample_pages)) {
|
|
error_setg(errp, "sample-pages is out of range[%d, %d].",
|
|
error_setg(errp, "sample-pages is out of range[%d, %d].",
|
|
@@ -435,6 +588,16 @@ void qmp_calc_dirty_rate(int64_t calc_time, bool has_sample_pages,
|
|
sample_pages = DIRTYRATE_DEFAULT_SAMPLE_PAGES;
|
|
sample_pages = DIRTYRATE_DEFAULT_SAMPLE_PAGES;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+ /*
|
|
|
|
+ * dirty ring mode only works when kvm dirty ring is enabled.
|
|
|
|
+ */
|
|
|
|
+ if ((mode == DIRTY_RATE_MEASURE_MODE_DIRTY_RING) &&
|
|
|
|
+ !kvm_dirty_ring_enabled()) {
|
|
|
|
+ error_setg(errp, "dirty ring is disabled, use sample-pages method "
|
|
|
|
+ "or remeasure later.");
|
|
|
|
+ return;
|
|
|
|
+ }
|
|
|
|
+
|
|
/*
|
|
/*
|
|
* Init calculation state as unstarted.
|
|
* Init calculation state as unstarted.
|
|
*/
|
|
*/
|
|
@@ -447,7 +610,15 @@ void qmp_calc_dirty_rate(int64_t calc_time, bool has_sample_pages,
|
|
|
|
|
|
config.sample_period_seconds = calc_time;
|
|
config.sample_period_seconds = calc_time;
|
|
config.sample_pages_per_gigabytes = sample_pages;
|
|
config.sample_pages_per_gigabytes = sample_pages;
|
|
- config.mode = DIRTY_RATE_MEASURE_MODE_PAGE_SAMPLING;
|
|
|
|
|
|
+ config.mode = mode;
|
|
|
|
+
|
|
|
|
+ cleanup_dirtyrate_stat(config);
|
|
|
|
+
|
|
|
|
+ /*
|
|
|
|
+ * update dirty rate mode so that we can figure out what mode has
|
|
|
|
+ * been used in last calculation
|
|
|
|
+ **/
|
|
|
|
+ dirtyrate_mode = mode;
|
|
|
|
|
|
start_time = qemu_clock_get_ms(QEMU_CLOCK_REALTIME) / 1000;
|
|
start_time = qemu_clock_get_ms(QEMU_CLOCK_REALTIME) / 1000;
|
|
init_dirtyrate_stat(start_time, config);
|
|
init_dirtyrate_stat(start_time, config);
|
|
@@ -473,12 +644,24 @@ void hmp_info_dirty_rate(Monitor *mon, const QDict *qdict)
|
|
info->sample_pages);
|
|
info->sample_pages);
|
|
monitor_printf(mon, "Period: %"PRIi64" (sec)\n",
|
|
monitor_printf(mon, "Period: %"PRIi64" (sec)\n",
|
|
info->calc_time);
|
|
info->calc_time);
|
|
|
|
+ monitor_printf(mon, "Mode: %s\n",
|
|
|
|
+ DirtyRateMeasureMode_str(info->mode));
|
|
monitor_printf(mon, "Dirty rate: ");
|
|
monitor_printf(mon, "Dirty rate: ");
|
|
if (info->has_dirty_rate) {
|
|
if (info->has_dirty_rate) {
|
|
monitor_printf(mon, "%"PRIi64" (MB/s)\n", info->dirty_rate);
|
|
monitor_printf(mon, "%"PRIi64" (MB/s)\n", info->dirty_rate);
|
|
|
|
+ if (info->has_vcpu_dirty_rate) {
|
|
|
|
+ DirtyRateVcpuList *rate, *head = info->vcpu_dirty_rate;
|
|
|
|
+ for (rate = head; rate != NULL; rate = rate->next) {
|
|
|
|
+ monitor_printf(mon, "vcpu[%"PRIi64"], Dirty rate: %"PRIi64
|
|
|
|
+ " (MB/s)\n", rate->value->id,
|
|
|
|
+ rate->value->dirty_rate);
|
|
|
|
+ }
|
|
|
|
+ }
|
|
} else {
|
|
} else {
|
|
monitor_printf(mon, "(not ready)\n");
|
|
monitor_printf(mon, "(not ready)\n");
|
|
}
|
|
}
|
|
|
|
+
|
|
|
|
+ qapi_free_DirtyRateVcpuList(info->vcpu_dirty_rate);
|
|
g_free(info);
|
|
g_free(info);
|
|
}
|
|
}
|
|
|
|
|
|
@@ -487,6 +670,10 @@ void hmp_calc_dirty_rate(Monitor *mon, const QDict *qdict)
|
|
int64_t sec = qdict_get_try_int(qdict, "second", 0);
|
|
int64_t sec = qdict_get_try_int(qdict, "second", 0);
|
|
int64_t sample_pages = qdict_get_try_int(qdict, "sample_pages_per_GB", -1);
|
|
int64_t sample_pages = qdict_get_try_int(qdict, "sample_pages_per_GB", -1);
|
|
bool has_sample_pages = (sample_pages != -1);
|
|
bool has_sample_pages = (sample_pages != -1);
|
|
|
|
+ bool dirty_ring = qdict_get_try_bool(qdict, "dirty_ring", false);
|
|
|
|
+ DirtyRateMeasureMode mode =
|
|
|
|
+ (dirty_ring ? DIRTY_RATE_MEASURE_MODE_DIRTY_RING :
|
|
|
|
+ DIRTY_RATE_MEASURE_MODE_PAGE_SAMPLING);
|
|
Error *err = NULL;
|
|
Error *err = NULL;
|
|
|
|
|
|
if (!sec) {
|
|
if (!sec) {
|
|
@@ -494,7 +681,8 @@ void hmp_calc_dirty_rate(Monitor *mon, const QDict *qdict)
|
|
return;
|
|
return;
|
|
}
|
|
}
|
|
|
|
|
|
- qmp_calc_dirty_rate(sec, has_sample_pages, sample_pages, &err);
|
|
|
|
|
|
+ qmp_calc_dirty_rate(sec, has_sample_pages, sample_pages, true,
|
|
|
|
+ mode, &err);
|
|
if (err) {
|
|
if (err) {
|
|
hmp_handle_error(mon, err);
|
|
hmp_handle_error(mon, err);
|
|
return;
|
|
return;
|