tests: introduce a framework for testing migration performance
This introduces a moderately general purpose framework for
testing performance of migration.
The initial guest workload is provided by the included 'stress'
program, which is configured to spawn one thread per guest CPU
and run a maximally memory intensive workload. It will loop
over GB of memory, xor'ing each byte with data from a 4k array
of random bytes. This ensures heavy read and write load across
all of guest memory to stress the migration performance. While
running the 'stress' program will record how long it takes to
xor each GB of memory and print this data for later reporting.
The test engine will spawn a pair of QEMU processes, either on
the same host, or with the target on a remote host via ssh,
using the host kernel and a custom initrd built with 'stress'
as the /init binary. Kernel command line args are set to ensure
a fast kernel boot time (< 1 second) between launching QEMU and
the stress program starting execution.
None the less, the test engine will initially wait N seconds for
the guest workload to stablize, before starting the migration
operation. When migration is running, the engine will use pause,
post-copy, autoconverge, xbzrle compression and multithread
compression features, as well as downtime & bandwidth tuning
to encourage completion. If migration completes, the test engine
will wait N seconds again for the guest workooad to stablize on
the target host. If migration does not complete after a preset
number of iterations, it will be aborted.
While the QEMU process is running on the source host, the test
engine will sample the host CPU usage of QEMU as a whole, and
each vCPU thread. While migration is running, it will record
all the stats reported by 'query-migration'. Finally, it will
capture the output of the stress program running in the guest.
All the data produced from a single test execution is recorded
in a structured JSON file. A separate program is then able to
create interactive charts using the "plotly" python + javascript
libraries, showing the characteristics of the migration.
The data output provides visualization of the effect on guest
vCPU workloads from the migration process, the corresponding
vCPU utilization on the host, and the overall CPU hit from
QEMU on the host. This is correlated from statistics from the
migration process, such as downtime, vCPU throttling and iteration
number.
While the tests can be run individually with arbitrary parameters,
there is also a facility for producing batch reports for a number
of pre-defined scenarios / comparisons, in order to be able to
get standardized results across different hardware configurations
(eg TCP vs RDMA, or comparing different VCPU counts / memory
sizes, etc).
To use this, first you must build the initrd image
$ make tests/migration/initrd-stress.img
To run a a one-shot test with all default parameters
$ ./tests/migration/guestperf.py > result.json
This has many command line args for varying its behaviour.
For example, to increase the RAM size and CPU count and
bind it to specific host NUMA nodes
$ ./tests/migration/guestperf.py \
--mem 4 --cpus 2 \
--src-mem-bind 0 --src-cpu-bind 0,1 \
--dst-mem-bind 1 --dst-cpu-bind 2,3 \
> result.json
Using mem + cpu binding is strongly recommended on NUMA
machines, otherwise the guest performance results will
vary wildly between runs of the test due to lucky/unlucky
NUMA placement, making sensible data analysis impossible.
To make it run across separate hosts:
$ ./tests/migration/guestperf.py \
--dst-host somehostname > result.json
To request that post-copy is enabled, with switchover
after 5 iterations
$ ./tests/migration/guestperf.py \
--post-copy --post-copy-iters 5 > result.json
Once a result.json file is created, a graph of the data
can be generated, showing guest workload performance per
thread and the migration iteration points:
$ ./tests/migration/guestperf-plot.py --output result.html \
--migration-iters --split-guest-cpu result.json
To further include host vCPU utilization and overall QEMU
utilization
$ ./tests/migration/guestperf-plot.py --output result.html \
--migration-iters --split-guest-cpu \
--qemu-cpu --vcpu-cpu result.json
NB, the 'guestperf-plot.py' command requires that you have
the plotly python library installed. eg you must do
$ pip install --user plotly
Viewing the result.html file requires that you have the
plotly.min.js file in the same directory as the HTML
output. This js file is installed as part of the plotly
python library, so can be found in
$HOME/.local/lib/python2.7/site-packages/plotly/offline/plotly.min.js
The guestperf-plot.py program can accept multiple json files
to plot, enabling results from different configurations to
be compared.
Finally, to run the entire standardized set of comparisons
$ ./tests/migration/guestperf-batch.py \
--dst-host somehost \
--mem 4 --cpus 2 \
--src-mem-bind 0 --src-cpu-bind 0,1 \
--dst-mem-bind 1 --dst-cpu-bind 2,3
--output tcp-somehost-4gb-2cpu
will store JSON files from all scenarios in the directory
named tcp-somehost-4gb-2cpu
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
Message-Id: <1469020993-29426-7-git-send-email-berrange@redhat.com>
Signed-off-by: Amit Shah <amit.shah@redhat.com>
2016-07-20 15:23:13 +02:00
|
|
|
#
|
|
|
|
# Migration test graph plotting
|
|
|
|
#
|
|
|
|
# Copyright (c) 2016 Red Hat, Inc.
|
|
|
|
#
|
|
|
|
# This library is free software; you can redistribute it and/or
|
|
|
|
# modify it under the terms of the GNU Lesser General Public
|
|
|
|
# License as published by the Free Software Foundation; either
|
2020-11-10 19:42:21 +01:00
|
|
|
# version 2.1 of the License, or (at your option) any later version.
|
tests: introduce a framework for testing migration performance
This introduces a moderately general purpose framework for
testing performance of migration.
The initial guest workload is provided by the included 'stress'
program, which is configured to spawn one thread per guest CPU
and run a maximally memory intensive workload. It will loop
over GB of memory, xor'ing each byte with data from a 4k array
of random bytes. This ensures heavy read and write load across
all of guest memory to stress the migration performance. While
running the 'stress' program will record how long it takes to
xor each GB of memory and print this data for later reporting.
The test engine will spawn a pair of QEMU processes, either on
the same host, or with the target on a remote host via ssh,
using the host kernel and a custom initrd built with 'stress'
as the /init binary. Kernel command line args are set to ensure
a fast kernel boot time (< 1 second) between launching QEMU and
the stress program starting execution.
None the less, the test engine will initially wait N seconds for
the guest workload to stablize, before starting the migration
operation. When migration is running, the engine will use pause,
post-copy, autoconverge, xbzrle compression and multithread
compression features, as well as downtime & bandwidth tuning
to encourage completion. If migration completes, the test engine
will wait N seconds again for the guest workooad to stablize on
the target host. If migration does not complete after a preset
number of iterations, it will be aborted.
While the QEMU process is running on the source host, the test
engine will sample the host CPU usage of QEMU as a whole, and
each vCPU thread. While migration is running, it will record
all the stats reported by 'query-migration'. Finally, it will
capture the output of the stress program running in the guest.
All the data produced from a single test execution is recorded
in a structured JSON file. A separate program is then able to
create interactive charts using the "plotly" python + javascript
libraries, showing the characteristics of the migration.
The data output provides visualization of the effect on guest
vCPU workloads from the migration process, the corresponding
vCPU utilization on the host, and the overall CPU hit from
QEMU on the host. This is correlated from statistics from the
migration process, such as downtime, vCPU throttling and iteration
number.
While the tests can be run individually with arbitrary parameters,
there is also a facility for producing batch reports for a number
of pre-defined scenarios / comparisons, in order to be able to
get standardized results across different hardware configurations
(eg TCP vs RDMA, or comparing different VCPU counts / memory
sizes, etc).
To use this, first you must build the initrd image
$ make tests/migration/initrd-stress.img
To run a a one-shot test with all default parameters
$ ./tests/migration/guestperf.py > result.json
This has many command line args for varying its behaviour.
For example, to increase the RAM size and CPU count and
bind it to specific host NUMA nodes
$ ./tests/migration/guestperf.py \
--mem 4 --cpus 2 \
--src-mem-bind 0 --src-cpu-bind 0,1 \
--dst-mem-bind 1 --dst-cpu-bind 2,3 \
> result.json
Using mem + cpu binding is strongly recommended on NUMA
machines, otherwise the guest performance results will
vary wildly between runs of the test due to lucky/unlucky
NUMA placement, making sensible data analysis impossible.
To make it run across separate hosts:
$ ./tests/migration/guestperf.py \
--dst-host somehostname > result.json
To request that post-copy is enabled, with switchover
after 5 iterations
$ ./tests/migration/guestperf.py \
--post-copy --post-copy-iters 5 > result.json
Once a result.json file is created, a graph of the data
can be generated, showing guest workload performance per
thread and the migration iteration points:
$ ./tests/migration/guestperf-plot.py --output result.html \
--migration-iters --split-guest-cpu result.json
To further include host vCPU utilization and overall QEMU
utilization
$ ./tests/migration/guestperf-plot.py --output result.html \
--migration-iters --split-guest-cpu \
--qemu-cpu --vcpu-cpu result.json
NB, the 'guestperf-plot.py' command requires that you have
the plotly python library installed. eg you must do
$ pip install --user plotly
Viewing the result.html file requires that you have the
plotly.min.js file in the same directory as the HTML
output. This js file is installed as part of the plotly
python library, so can be found in
$HOME/.local/lib/python2.7/site-packages/plotly/offline/plotly.min.js
The guestperf-plot.py program can accept multiple json files
to plot, enabling results from different configurations to
be compared.
Finally, to run the entire standardized set of comparisons
$ ./tests/migration/guestperf-batch.py \
--dst-host somehost \
--mem 4 --cpus 2 \
--src-mem-bind 0 --src-cpu-bind 0,1 \
--dst-mem-bind 1 --dst-cpu-bind 2,3
--output tcp-somehost-4gb-2cpu
will store JSON files from all scenarios in the directory
named tcp-somehost-4gb-2cpu
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
Message-Id: <1469020993-29426-7-git-send-email-berrange@redhat.com>
Signed-off-by: Amit Shah <amit.shah@redhat.com>
2016-07-20 15:23:13 +02:00
|
|
|
#
|
|
|
|
# This library is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
# Lesser General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU Lesser General Public
|
|
|
|
# License along with this library; if not, see <http://www.gnu.org/licenses/>.
|
|
|
|
#
|
|
|
|
|
|
|
|
import sys
|
|
|
|
|
|
|
|
|
|
|
|
class Plot(object):
|
|
|
|
|
|
|
|
# Generated using
|
|
|
|
# http://tools.medialab.sciences-po.fr/iwanthue/
|
|
|
|
COLORS = ["#CD54D0",
|
|
|
|
"#79D94C",
|
|
|
|
"#7470CD",
|
|
|
|
"#D2D251",
|
|
|
|
"#863D79",
|
|
|
|
"#76DDA6",
|
|
|
|
"#D4467B",
|
|
|
|
"#61923D",
|
|
|
|
"#CB9CCA",
|
|
|
|
"#D98F36",
|
|
|
|
"#8CC8DA",
|
|
|
|
"#CE4831",
|
|
|
|
"#5E7693",
|
|
|
|
"#9B803F",
|
|
|
|
"#412F4C",
|
|
|
|
"#CECBA6",
|
|
|
|
"#6D3229",
|
|
|
|
"#598B73",
|
|
|
|
"#C8827C",
|
|
|
|
"#394427"]
|
|
|
|
|
|
|
|
def __init__(self,
|
|
|
|
reports,
|
|
|
|
migration_iters,
|
|
|
|
total_guest_cpu,
|
|
|
|
split_guest_cpu,
|
|
|
|
qemu_cpu,
|
|
|
|
vcpu_cpu):
|
|
|
|
|
|
|
|
self._reports = reports
|
|
|
|
self._migration_iters = migration_iters
|
|
|
|
self._total_guest_cpu = total_guest_cpu
|
|
|
|
self._split_guest_cpu = split_guest_cpu
|
|
|
|
self._qemu_cpu = qemu_cpu
|
|
|
|
self._vcpu_cpu = vcpu_cpu
|
|
|
|
self._color_idx = 0
|
|
|
|
|
|
|
|
def _next_color(self):
|
|
|
|
color = self.COLORS[self._color_idx]
|
|
|
|
self._color_idx += 1
|
|
|
|
if self._color_idx >= len(self.COLORS):
|
|
|
|
self._color_idx = 0
|
|
|
|
return color
|
|
|
|
|
|
|
|
def _get_progress_label(self, progress):
|
|
|
|
if progress:
|
|
|
|
return "\n\n" + "\n".join(
|
|
|
|
["Status: %s" % progress._status,
|
|
|
|
"Iteration: %d" % progress._ram._iterations,
|
|
|
|
"Throttle: %02d%%" % progress._throttle_pcent,
|
|
|
|
"Dirty rate: %dMB/s" % (progress._ram._dirty_rate_pps * 4 / 1024.0)])
|
|
|
|
else:
|
|
|
|
return "\n\n" + "\n".join(
|
|
|
|
["Status: %s" % "none",
|
|
|
|
"Iteration: %d" % 0])
|
|
|
|
|
|
|
|
def _find_start_time(self, report):
|
|
|
|
startqemu = report._qemu_timings._records[0]._timestamp
|
|
|
|
startguest = report._guest_timings._records[0]._timestamp
|
|
|
|
if startqemu < startguest:
|
|
|
|
return startqemu
|
|
|
|
else:
|
|
|
|
return stasrtguest
|
|
|
|
|
|
|
|
def _get_guest_max_value(self, report):
|
|
|
|
maxvalue = 0
|
|
|
|
for record in report._guest_timings._records:
|
|
|
|
if record._value > maxvalue:
|
|
|
|
maxvalue = record._value
|
|
|
|
return maxvalue
|
|
|
|
|
|
|
|
def _get_qemu_max_value(self, report):
|
|
|
|
maxvalue = 0
|
|
|
|
oldvalue = None
|
|
|
|
oldtime = None
|
|
|
|
for record in report._qemu_timings._records:
|
|
|
|
if oldvalue is not None:
|
|
|
|
cpudelta = (record._value - oldvalue) / 1000.0
|
|
|
|
timedelta = record._timestamp - oldtime
|
|
|
|
if timedelta == 0:
|
|
|
|
continue
|
|
|
|
util = cpudelta / timedelta * 100.0
|
|
|
|
else:
|
|
|
|
util = 0
|
|
|
|
oldvalue = record._value
|
|
|
|
oldtime = record._timestamp
|
|
|
|
|
|
|
|
if util > maxvalue:
|
|
|
|
maxvalue = util
|
|
|
|
return maxvalue
|
|
|
|
|
|
|
|
def _get_total_guest_cpu_graph(self, report, starttime):
|
|
|
|
xaxis = []
|
|
|
|
yaxis = []
|
|
|
|
labels = []
|
|
|
|
progress_idx = -1
|
|
|
|
for record in report._guest_timings._records:
|
|
|
|
while ((progress_idx + 1) < len(report._progress_history) and
|
|
|
|
report._progress_history[progress_idx + 1]._now < record._timestamp):
|
|
|
|
progress_idx = progress_idx + 1
|
|
|
|
|
|
|
|
if progress_idx >= 0:
|
|
|
|
progress = report._progress_history[progress_idx]
|
|
|
|
else:
|
|
|
|
progress = None
|
|
|
|
|
|
|
|
xaxis.append(record._timestamp - starttime)
|
|
|
|
yaxis.append(record._value)
|
|
|
|
labels.append(self._get_progress_label(progress))
|
|
|
|
|
|
|
|
from plotly import graph_objs as go
|
|
|
|
return go.Scatter(x=xaxis,
|
|
|
|
y=yaxis,
|
|
|
|
name="Guest PIDs: %s" % report._scenario._name,
|
|
|
|
mode='lines',
|
|
|
|
line={
|
|
|
|
"dash": "solid",
|
|
|
|
"color": self._next_color(),
|
|
|
|
"shape": "linear",
|
|
|
|
"width": 1
|
|
|
|
},
|
|
|
|
text=labels)
|
|
|
|
|
|
|
|
def _get_split_guest_cpu_graphs(self, report, starttime):
|
|
|
|
threads = {}
|
|
|
|
for record in report._guest_timings._records:
|
|
|
|
if record._tid in threads:
|
|
|
|
continue
|
|
|
|
threads[record._tid] = {
|
|
|
|
"xaxis": [],
|
|
|
|
"yaxis": [],
|
|
|
|
"labels": [],
|
|
|
|
}
|
|
|
|
|
|
|
|
progress_idx = -1
|
|
|
|
for record in report._guest_timings._records:
|
|
|
|
while ((progress_idx + 1) < len(report._progress_history) and
|
|
|
|
report._progress_history[progress_idx + 1]._now < record._timestamp):
|
|
|
|
progress_idx = progress_idx + 1
|
|
|
|
|
|
|
|
if progress_idx >= 0:
|
|
|
|
progress = report._progress_history[progress_idx]
|
|
|
|
else:
|
|
|
|
progress = None
|
|
|
|
|
|
|
|
threads[record._tid]["xaxis"].append(record._timestamp - starttime)
|
|
|
|
threads[record._tid]["yaxis"].append(record._value)
|
|
|
|
threads[record._tid]["labels"].append(self._get_progress_label(progress))
|
|
|
|
|
|
|
|
|
|
|
|
graphs = []
|
|
|
|
from plotly import graph_objs as go
|
|
|
|
for tid in threads.keys():
|
|
|
|
graphs.append(
|
|
|
|
go.Scatter(x=threads[tid]["xaxis"],
|
|
|
|
y=threads[tid]["yaxis"],
|
|
|
|
name="PID %s: %s" % (tid, report._scenario._name),
|
|
|
|
mode="lines",
|
|
|
|
line={
|
|
|
|
"dash": "solid",
|
|
|
|
"color": self._next_color(),
|
|
|
|
"shape": "linear",
|
|
|
|
"width": 1
|
|
|
|
},
|
|
|
|
text=threads[tid]["labels"]))
|
|
|
|
return graphs
|
|
|
|
|
|
|
|
def _get_migration_iters_graph(self, report, starttime):
|
|
|
|
xaxis = []
|
|
|
|
yaxis = []
|
|
|
|
labels = []
|
|
|
|
for progress in report._progress_history:
|
|
|
|
xaxis.append(progress._now - starttime)
|
|
|
|
yaxis.append(0)
|
|
|
|
labels.append(self._get_progress_label(progress))
|
|
|
|
|
|
|
|
from plotly import graph_objs as go
|
|
|
|
return go.Scatter(x=xaxis,
|
|
|
|
y=yaxis,
|
|
|
|
text=labels,
|
|
|
|
name="Migration iterations",
|
|
|
|
mode="markers",
|
|
|
|
marker={
|
|
|
|
"color": self._next_color(),
|
|
|
|
"symbol": "star",
|
|
|
|
"size": 5
|
|
|
|
})
|
|
|
|
|
|
|
|
def _get_qemu_cpu_graph(self, report, starttime):
|
|
|
|
xaxis = []
|
|
|
|
yaxis = []
|
|
|
|
labels = []
|
|
|
|
progress_idx = -1
|
|
|
|
|
|
|
|
first = report._qemu_timings._records[0]
|
|
|
|
abstimestamps = [first._timestamp]
|
|
|
|
absvalues = [first._value]
|
|
|
|
|
|
|
|
for record in report._qemu_timings._records[1:]:
|
|
|
|
while ((progress_idx + 1) < len(report._progress_history) and
|
|
|
|
report._progress_history[progress_idx + 1]._now < record._timestamp):
|
|
|
|
progress_idx = progress_idx + 1
|
|
|
|
|
|
|
|
if progress_idx >= 0:
|
|
|
|
progress = report._progress_history[progress_idx]
|
|
|
|
else:
|
|
|
|
progress = None
|
|
|
|
|
|
|
|
oldvalue = absvalues[-1]
|
|
|
|
oldtime = abstimestamps[-1]
|
|
|
|
|
|
|
|
cpudelta = (record._value - oldvalue) / 1000.0
|
|
|
|
timedelta = record._timestamp - oldtime
|
|
|
|
if timedelta == 0:
|
|
|
|
continue
|
|
|
|
util = cpudelta / timedelta * 100.0
|
|
|
|
|
|
|
|
abstimestamps.append(record._timestamp)
|
|
|
|
absvalues.append(record._value)
|
|
|
|
|
|
|
|
xaxis.append(record._timestamp - starttime)
|
|
|
|
yaxis.append(util)
|
|
|
|
labels.append(self._get_progress_label(progress))
|
|
|
|
|
|
|
|
from plotly import graph_objs as go
|
|
|
|
return go.Scatter(x=xaxis,
|
|
|
|
y=yaxis,
|
|
|
|
yaxis="y2",
|
|
|
|
name="QEMU: %s" % report._scenario._name,
|
|
|
|
mode='lines',
|
|
|
|
line={
|
|
|
|
"dash": "solid",
|
|
|
|
"color": self._next_color(),
|
|
|
|
"shape": "linear",
|
|
|
|
"width": 1
|
|
|
|
},
|
|
|
|
text=labels)
|
|
|
|
|
|
|
|
def _get_vcpu_cpu_graphs(self, report, starttime):
|
|
|
|
threads = {}
|
|
|
|
for record in report._vcpu_timings._records:
|
|
|
|
if record._tid in threads:
|
|
|
|
continue
|
|
|
|
threads[record._tid] = {
|
|
|
|
"xaxis": [],
|
|
|
|
"yaxis": [],
|
|
|
|
"labels": [],
|
|
|
|
"absvalue": [record._value],
|
|
|
|
"abstime": [record._timestamp],
|
|
|
|
}
|
|
|
|
|
|
|
|
progress_idx = -1
|
|
|
|
for record in report._vcpu_timings._records:
|
|
|
|
while ((progress_idx + 1) < len(report._progress_history) and
|
|
|
|
report._progress_history[progress_idx + 1]._now < record._timestamp):
|
|
|
|
progress_idx = progress_idx + 1
|
|
|
|
|
|
|
|
if progress_idx >= 0:
|
|
|
|
progress = report._progress_history[progress_idx]
|
|
|
|
else:
|
|
|
|
progress = None
|
|
|
|
|
|
|
|
oldvalue = threads[record._tid]["absvalue"][-1]
|
|
|
|
oldtime = threads[record._tid]["abstime"][-1]
|
|
|
|
|
|
|
|
cpudelta = (record._value - oldvalue) / 1000.0
|
|
|
|
timedelta = record._timestamp - oldtime
|
|
|
|
if timedelta == 0:
|
|
|
|
continue
|
|
|
|
util = cpudelta / timedelta * 100.0
|
|
|
|
if util > 100:
|
|
|
|
util = 100
|
|
|
|
|
|
|
|
threads[record._tid]["absvalue"].append(record._value)
|
|
|
|
threads[record._tid]["abstime"].append(record._timestamp)
|
|
|
|
|
|
|
|
threads[record._tid]["xaxis"].append(record._timestamp - starttime)
|
|
|
|
threads[record._tid]["yaxis"].append(util)
|
|
|
|
threads[record._tid]["labels"].append(self._get_progress_label(progress))
|
|
|
|
|
|
|
|
|
|
|
|
graphs = []
|
|
|
|
from plotly import graph_objs as go
|
|
|
|
for tid in threads.keys():
|
|
|
|
graphs.append(
|
|
|
|
go.Scatter(x=threads[tid]["xaxis"],
|
|
|
|
y=threads[tid]["yaxis"],
|
|
|
|
yaxis="y2",
|
|
|
|
name="VCPU %s: %s" % (tid, report._scenario._name),
|
|
|
|
mode="lines",
|
|
|
|
line={
|
|
|
|
"dash": "solid",
|
|
|
|
"color": self._next_color(),
|
|
|
|
"shape": "linear",
|
|
|
|
"width": 1
|
|
|
|
},
|
|
|
|
text=threads[tid]["labels"]))
|
|
|
|
return graphs
|
|
|
|
|
|
|
|
def _generate_chart_report(self, report):
|
|
|
|
graphs = []
|
|
|
|
starttime = self._find_start_time(report)
|
|
|
|
if self._total_guest_cpu:
|
|
|
|
graphs.append(self._get_total_guest_cpu_graph(report, starttime))
|
|
|
|
if self._split_guest_cpu:
|
|
|
|
graphs.extend(self._get_split_guest_cpu_graphs(report, starttime))
|
|
|
|
if self._qemu_cpu:
|
|
|
|
graphs.append(self._get_qemu_cpu_graph(report, starttime))
|
|
|
|
if self._vcpu_cpu:
|
|
|
|
graphs.extend(self._get_vcpu_cpu_graphs(report, starttime))
|
|
|
|
if self._migration_iters:
|
|
|
|
graphs.append(self._get_migration_iters_graph(report, starttime))
|
|
|
|
return graphs
|
|
|
|
|
|
|
|
def _generate_annotation(self, starttime, progress):
|
|
|
|
return {
|
|
|
|
"text": progress._status,
|
|
|
|
"x": progress._now - starttime,
|
|
|
|
"y": 10,
|
|
|
|
}
|
|
|
|
|
|
|
|
def _generate_annotations(self, report):
|
|
|
|
starttime = self._find_start_time(report)
|
|
|
|
annotations = {}
|
|
|
|
started = False
|
|
|
|
for progress in report._progress_history:
|
|
|
|
if progress._status == "setup":
|
|
|
|
continue
|
|
|
|
if progress._status not in annotations:
|
|
|
|
annotations[progress._status] = self._generate_annotation(starttime, progress)
|
|
|
|
|
|
|
|
return annotations.values()
|
|
|
|
|
|
|
|
def _generate_chart(self):
|
|
|
|
from plotly.offline import plot
|
|
|
|
from plotly import graph_objs as go
|
|
|
|
|
|
|
|
graphs = []
|
|
|
|
yaxismax = 0
|
|
|
|
yaxismax2 = 0
|
|
|
|
for report in self._reports:
|
|
|
|
graphs.extend(self._generate_chart_report(report))
|
|
|
|
|
|
|
|
maxvalue = self._get_guest_max_value(report)
|
|
|
|
if maxvalue > yaxismax:
|
|
|
|
yaxismax = maxvalue
|
|
|
|
|
|
|
|
maxvalue = self._get_qemu_max_value(report)
|
|
|
|
if maxvalue > yaxismax2:
|
|
|
|
yaxismax2 = maxvalue
|
|
|
|
|
|
|
|
yaxismax += 100
|
|
|
|
if not self._qemu_cpu:
|
|
|
|
yaxismax2 = 110
|
|
|
|
yaxismax2 += 10
|
|
|
|
|
|
|
|
annotations = []
|
|
|
|
if self._migration_iters:
|
|
|
|
for report in self._reports:
|
|
|
|
annotations.extend(self._generate_annotations(report))
|
|
|
|
|
|
|
|
layout = go.Layout(title="Migration comparison",
|
|
|
|
xaxis={
|
|
|
|
"title": "Wallclock time (secs)",
|
|
|
|
"showgrid": False,
|
|
|
|
},
|
|
|
|
yaxis={
|
|
|
|
"title": "Memory update speed (ms/GB)",
|
|
|
|
"showgrid": False,
|
|
|
|
"range": [0, yaxismax],
|
|
|
|
},
|
|
|
|
yaxis2={
|
|
|
|
"title": "Hostutilization (%)",
|
|
|
|
"overlaying": "y",
|
|
|
|
"side": "right",
|
|
|
|
"range": [0, yaxismax2],
|
|
|
|
"showgrid": False,
|
|
|
|
},
|
|
|
|
annotations=annotations)
|
|
|
|
|
|
|
|
figure = go.Figure(data=graphs, layout=layout)
|
|
|
|
|
|
|
|
return plot(figure,
|
|
|
|
show_link=False,
|
|
|
|
include_plotlyjs=False,
|
|
|
|
output_type="div")
|
|
|
|
|
|
|
|
|
|
|
|
def _generate_report(self):
|
|
|
|
pieces = []
|
|
|
|
for report in self._reports:
|
|
|
|
pieces.append("""
|
|
|
|
<h3>Report %s</h3>
|
|
|
|
<table>
|
|
|
|
""" % report._scenario._name)
|
|
|
|
|
|
|
|
pieces.append("""
|
|
|
|
<tr class="subhead">
|
|
|
|
<th colspan="2">Test config</th>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Emulator:</th>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Kernel:</th>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Ramdisk:</th>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Transport:</th>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Host:</th>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>
|
|
|
|
""" % (report._binary, report._kernel,
|
|
|
|
report._initrd, report._transport, report._dst_host))
|
|
|
|
|
|
|
|
hardware = report._hardware
|
|
|
|
pieces.append("""
|
|
|
|
<tr class="subhead">
|
|
|
|
<th colspan="2">Hardware config</th>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>CPUs:</th>
|
|
|
|
<td>%d</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>RAM:</th>
|
|
|
|
<td>%d GB</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Source CPU bind:</th>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Source RAM bind:</th>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Dest CPU bind:</th>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Dest RAM bind:</th>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Preallocate RAM:</th>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Locked RAM:</th>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Huge pages:</th>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>
|
|
|
|
""" % (hardware._cpus, hardware._mem,
|
|
|
|
",".join(hardware._src_cpu_bind),
|
|
|
|
",".join(hardware._src_mem_bind),
|
|
|
|
",".join(hardware._dst_cpu_bind),
|
|
|
|
",".join(hardware._dst_mem_bind),
|
|
|
|
"yes" if hardware._prealloc_pages else "no",
|
|
|
|
"yes" if hardware._locked_pages else "no",
|
|
|
|
"yes" if hardware._huge_pages else "no"))
|
|
|
|
|
|
|
|
scenario = report._scenario
|
|
|
|
pieces.append("""
|
|
|
|
<tr class="subhead">
|
|
|
|
<th colspan="2">Scenario config</th>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Max downtime:</th>
|
|
|
|
<td>%d milli-sec</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Max bandwidth:</th>
|
|
|
|
<td>%d MB/sec</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Max iters:</th>
|
|
|
|
<td>%d</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Max time:</th>
|
|
|
|
<td>%d secs</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Pause:</th>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Pause iters:</th>
|
|
|
|
<td>%d</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Post-copy:</th>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Post-copy iters:</th>
|
|
|
|
<td>%d</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Auto-converge:</th>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>Auto-converge iters:</th>
|
|
|
|
<td>%d</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>MT compression:</th>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>MT compression threads:</th>
|
|
|
|
<td>%d</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>XBZRLE compression:</th>
|
|
|
|
<td>%s</td>
|
|
|
|
</tr>
|
|
|
|
<tr>
|
|
|
|
<th>XBZRLE compression cache:</th>
|
|
|
|
<td>%d%% of RAM</td>
|
|
|
|
</tr>
|
|
|
|
""" % (scenario._downtime, scenario._bandwidth,
|
|
|
|
scenario._max_iters, scenario._max_time,
|
|
|
|
"yes" if scenario._pause else "no", scenario._pause_iters,
|
|
|
|
"yes" if scenario._post_copy else "no", scenario._post_copy_iters,
|
|
|
|
"yes" if scenario._auto_converge else "no", scenario._auto_converge_step,
|
|
|
|
"yes" if scenario._compression_mt else "no", scenario._compression_mt_threads,
|
|
|
|
"yes" if scenario._compression_xbzrle else "no", scenario._compression_xbzrle_cache))
|
|
|
|
|
|
|
|
pieces.append("""
|
|
|
|
</table>
|
|
|
|
""")
|
|
|
|
|
|
|
|
return "\n".join(pieces)
|
|
|
|
|
|
|
|
def _generate_style(self):
|
|
|
|
return """
|
|
|
|
#report table tr th {
|
|
|
|
text-align: right;
|
|
|
|
}
|
|
|
|
#report table tr td {
|
|
|
|
text-align: left;
|
|
|
|
}
|
|
|
|
#report table tr.subhead th {
|
|
|
|
background: rgb(192, 192, 192);
|
|
|
|
text-align: center;
|
|
|
|
}
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
def generate_html(self, fh):
|
2018-06-08 14:29:43 +02:00
|
|
|
print("""<html>
|
tests: introduce a framework for testing migration performance
This introduces a moderately general purpose framework for
testing performance of migration.
The initial guest workload is provided by the included 'stress'
program, which is configured to spawn one thread per guest CPU
and run a maximally memory intensive workload. It will loop
over GB of memory, xor'ing each byte with data from a 4k array
of random bytes. This ensures heavy read and write load across
all of guest memory to stress the migration performance. While
running the 'stress' program will record how long it takes to
xor each GB of memory and print this data for later reporting.
The test engine will spawn a pair of QEMU processes, either on
the same host, or with the target on a remote host via ssh,
using the host kernel and a custom initrd built with 'stress'
as the /init binary. Kernel command line args are set to ensure
a fast kernel boot time (< 1 second) between launching QEMU and
the stress program starting execution.
None the less, the test engine will initially wait N seconds for
the guest workload to stablize, before starting the migration
operation. When migration is running, the engine will use pause,
post-copy, autoconverge, xbzrle compression and multithread
compression features, as well as downtime & bandwidth tuning
to encourage completion. If migration completes, the test engine
will wait N seconds again for the guest workooad to stablize on
the target host. If migration does not complete after a preset
number of iterations, it will be aborted.
While the QEMU process is running on the source host, the test
engine will sample the host CPU usage of QEMU as a whole, and
each vCPU thread. While migration is running, it will record
all the stats reported by 'query-migration'. Finally, it will
capture the output of the stress program running in the guest.
All the data produced from a single test execution is recorded
in a structured JSON file. A separate program is then able to
create interactive charts using the "plotly" python + javascript
libraries, showing the characteristics of the migration.
The data output provides visualization of the effect on guest
vCPU workloads from the migration process, the corresponding
vCPU utilization on the host, and the overall CPU hit from
QEMU on the host. This is correlated from statistics from the
migration process, such as downtime, vCPU throttling and iteration
number.
While the tests can be run individually with arbitrary parameters,
there is also a facility for producing batch reports for a number
of pre-defined scenarios / comparisons, in order to be able to
get standardized results across different hardware configurations
(eg TCP vs RDMA, or comparing different VCPU counts / memory
sizes, etc).
To use this, first you must build the initrd image
$ make tests/migration/initrd-stress.img
To run a a one-shot test with all default parameters
$ ./tests/migration/guestperf.py > result.json
This has many command line args for varying its behaviour.
For example, to increase the RAM size and CPU count and
bind it to specific host NUMA nodes
$ ./tests/migration/guestperf.py \
--mem 4 --cpus 2 \
--src-mem-bind 0 --src-cpu-bind 0,1 \
--dst-mem-bind 1 --dst-cpu-bind 2,3 \
> result.json
Using mem + cpu binding is strongly recommended on NUMA
machines, otherwise the guest performance results will
vary wildly between runs of the test due to lucky/unlucky
NUMA placement, making sensible data analysis impossible.
To make it run across separate hosts:
$ ./tests/migration/guestperf.py \
--dst-host somehostname > result.json
To request that post-copy is enabled, with switchover
after 5 iterations
$ ./tests/migration/guestperf.py \
--post-copy --post-copy-iters 5 > result.json
Once a result.json file is created, a graph of the data
can be generated, showing guest workload performance per
thread and the migration iteration points:
$ ./tests/migration/guestperf-plot.py --output result.html \
--migration-iters --split-guest-cpu result.json
To further include host vCPU utilization and overall QEMU
utilization
$ ./tests/migration/guestperf-plot.py --output result.html \
--migration-iters --split-guest-cpu \
--qemu-cpu --vcpu-cpu result.json
NB, the 'guestperf-plot.py' command requires that you have
the plotly python library installed. eg you must do
$ pip install --user plotly
Viewing the result.html file requires that you have the
plotly.min.js file in the same directory as the HTML
output. This js file is installed as part of the plotly
python library, so can be found in
$HOME/.local/lib/python2.7/site-packages/plotly/offline/plotly.min.js
The guestperf-plot.py program can accept multiple json files
to plot, enabling results from different configurations to
be compared.
Finally, to run the entire standardized set of comparisons
$ ./tests/migration/guestperf-batch.py \
--dst-host somehost \
--mem 4 --cpus 2 \
--src-mem-bind 0 --src-cpu-bind 0,1 \
--dst-mem-bind 1 --dst-cpu-bind 2,3
--output tcp-somehost-4gb-2cpu
will store JSON files from all scenarios in the directory
named tcp-somehost-4gb-2cpu
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
Message-Id: <1469020993-29426-7-git-send-email-berrange@redhat.com>
Signed-off-by: Amit Shah <amit.shah@redhat.com>
2016-07-20 15:23:13 +02:00
|
|
|
<head>
|
|
|
|
<script type="text/javascript" src="plotly.min.js">
|
|
|
|
</script>
|
|
|
|
<style type="text/css">
|
|
|
|
%s
|
|
|
|
</style>
|
|
|
|
<title>Migration report</title>
|
|
|
|
</head>
|
|
|
|
<body>
|
|
|
|
<h1>Migration report</h1>
|
|
|
|
<h2>Chart summary</h2>
|
|
|
|
<div id="chart">
|
2018-06-08 14:29:43 +02:00
|
|
|
""" % self._generate_style(), file=fh)
|
|
|
|
print(self._generate_chart(), file=fh)
|
|
|
|
print("""
|
tests: introduce a framework for testing migration performance
This introduces a moderately general purpose framework for
testing performance of migration.
The initial guest workload is provided by the included 'stress'
program, which is configured to spawn one thread per guest CPU
and run a maximally memory intensive workload. It will loop
over GB of memory, xor'ing each byte with data from a 4k array
of random bytes. This ensures heavy read and write load across
all of guest memory to stress the migration performance. While
running the 'stress' program will record how long it takes to
xor each GB of memory and print this data for later reporting.
The test engine will spawn a pair of QEMU processes, either on
the same host, or with the target on a remote host via ssh,
using the host kernel and a custom initrd built with 'stress'
as the /init binary. Kernel command line args are set to ensure
a fast kernel boot time (< 1 second) between launching QEMU and
the stress program starting execution.
None the less, the test engine will initially wait N seconds for
the guest workload to stablize, before starting the migration
operation. When migration is running, the engine will use pause,
post-copy, autoconverge, xbzrle compression and multithread
compression features, as well as downtime & bandwidth tuning
to encourage completion. If migration completes, the test engine
will wait N seconds again for the guest workooad to stablize on
the target host. If migration does not complete after a preset
number of iterations, it will be aborted.
While the QEMU process is running on the source host, the test
engine will sample the host CPU usage of QEMU as a whole, and
each vCPU thread. While migration is running, it will record
all the stats reported by 'query-migration'. Finally, it will
capture the output of the stress program running in the guest.
All the data produced from a single test execution is recorded
in a structured JSON file. A separate program is then able to
create interactive charts using the "plotly" python + javascript
libraries, showing the characteristics of the migration.
The data output provides visualization of the effect on guest
vCPU workloads from the migration process, the corresponding
vCPU utilization on the host, and the overall CPU hit from
QEMU on the host. This is correlated from statistics from the
migration process, such as downtime, vCPU throttling and iteration
number.
While the tests can be run individually with arbitrary parameters,
there is also a facility for producing batch reports for a number
of pre-defined scenarios / comparisons, in order to be able to
get standardized results across different hardware configurations
(eg TCP vs RDMA, or comparing different VCPU counts / memory
sizes, etc).
To use this, first you must build the initrd image
$ make tests/migration/initrd-stress.img
To run a a one-shot test with all default parameters
$ ./tests/migration/guestperf.py > result.json
This has many command line args for varying its behaviour.
For example, to increase the RAM size and CPU count and
bind it to specific host NUMA nodes
$ ./tests/migration/guestperf.py \
--mem 4 --cpus 2 \
--src-mem-bind 0 --src-cpu-bind 0,1 \
--dst-mem-bind 1 --dst-cpu-bind 2,3 \
> result.json
Using mem + cpu binding is strongly recommended on NUMA
machines, otherwise the guest performance results will
vary wildly between runs of the test due to lucky/unlucky
NUMA placement, making sensible data analysis impossible.
To make it run across separate hosts:
$ ./tests/migration/guestperf.py \
--dst-host somehostname > result.json
To request that post-copy is enabled, with switchover
after 5 iterations
$ ./tests/migration/guestperf.py \
--post-copy --post-copy-iters 5 > result.json
Once a result.json file is created, a graph of the data
can be generated, showing guest workload performance per
thread and the migration iteration points:
$ ./tests/migration/guestperf-plot.py --output result.html \
--migration-iters --split-guest-cpu result.json
To further include host vCPU utilization and overall QEMU
utilization
$ ./tests/migration/guestperf-plot.py --output result.html \
--migration-iters --split-guest-cpu \
--qemu-cpu --vcpu-cpu result.json
NB, the 'guestperf-plot.py' command requires that you have
the plotly python library installed. eg you must do
$ pip install --user plotly
Viewing the result.html file requires that you have the
plotly.min.js file in the same directory as the HTML
output. This js file is installed as part of the plotly
python library, so can be found in
$HOME/.local/lib/python2.7/site-packages/plotly/offline/plotly.min.js
The guestperf-plot.py program can accept multiple json files
to plot, enabling results from different configurations to
be compared.
Finally, to run the entire standardized set of comparisons
$ ./tests/migration/guestperf-batch.py \
--dst-host somehost \
--mem 4 --cpus 2 \
--src-mem-bind 0 --src-cpu-bind 0,1 \
--dst-mem-bind 1 --dst-cpu-bind 2,3
--output tcp-somehost-4gb-2cpu
will store JSON files from all scenarios in the directory
named tcp-somehost-4gb-2cpu
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
Message-Id: <1469020993-29426-7-git-send-email-berrange@redhat.com>
Signed-off-by: Amit Shah <amit.shah@redhat.com>
2016-07-20 15:23:13 +02:00
|
|
|
</div>
|
|
|
|
<h2>Report details</h2>
|
|
|
|
<div id="report">
|
2018-06-08 14:29:43 +02:00
|
|
|
""", file=fh)
|
|
|
|
print(self._generate_report(), file=fh)
|
|
|
|
print("""
|
tests: introduce a framework for testing migration performance
This introduces a moderately general purpose framework for
testing performance of migration.
The initial guest workload is provided by the included 'stress'
program, which is configured to spawn one thread per guest CPU
and run a maximally memory intensive workload. It will loop
over GB of memory, xor'ing each byte with data from a 4k array
of random bytes. This ensures heavy read and write load across
all of guest memory to stress the migration performance. While
running the 'stress' program will record how long it takes to
xor each GB of memory and print this data for later reporting.
The test engine will spawn a pair of QEMU processes, either on
the same host, or with the target on a remote host via ssh,
using the host kernel and a custom initrd built with 'stress'
as the /init binary. Kernel command line args are set to ensure
a fast kernel boot time (< 1 second) between launching QEMU and
the stress program starting execution.
None the less, the test engine will initially wait N seconds for
the guest workload to stablize, before starting the migration
operation. When migration is running, the engine will use pause,
post-copy, autoconverge, xbzrle compression and multithread
compression features, as well as downtime & bandwidth tuning
to encourage completion. If migration completes, the test engine
will wait N seconds again for the guest workooad to stablize on
the target host. If migration does not complete after a preset
number of iterations, it will be aborted.
While the QEMU process is running on the source host, the test
engine will sample the host CPU usage of QEMU as a whole, and
each vCPU thread. While migration is running, it will record
all the stats reported by 'query-migration'. Finally, it will
capture the output of the stress program running in the guest.
All the data produced from a single test execution is recorded
in a structured JSON file. A separate program is then able to
create interactive charts using the "plotly" python + javascript
libraries, showing the characteristics of the migration.
The data output provides visualization of the effect on guest
vCPU workloads from the migration process, the corresponding
vCPU utilization on the host, and the overall CPU hit from
QEMU on the host. This is correlated from statistics from the
migration process, such as downtime, vCPU throttling and iteration
number.
While the tests can be run individually with arbitrary parameters,
there is also a facility for producing batch reports for a number
of pre-defined scenarios / comparisons, in order to be able to
get standardized results across different hardware configurations
(eg TCP vs RDMA, or comparing different VCPU counts / memory
sizes, etc).
To use this, first you must build the initrd image
$ make tests/migration/initrd-stress.img
To run a a one-shot test with all default parameters
$ ./tests/migration/guestperf.py > result.json
This has many command line args for varying its behaviour.
For example, to increase the RAM size and CPU count and
bind it to specific host NUMA nodes
$ ./tests/migration/guestperf.py \
--mem 4 --cpus 2 \
--src-mem-bind 0 --src-cpu-bind 0,1 \
--dst-mem-bind 1 --dst-cpu-bind 2,3 \
> result.json
Using mem + cpu binding is strongly recommended on NUMA
machines, otherwise the guest performance results will
vary wildly between runs of the test due to lucky/unlucky
NUMA placement, making sensible data analysis impossible.
To make it run across separate hosts:
$ ./tests/migration/guestperf.py \
--dst-host somehostname > result.json
To request that post-copy is enabled, with switchover
after 5 iterations
$ ./tests/migration/guestperf.py \
--post-copy --post-copy-iters 5 > result.json
Once a result.json file is created, a graph of the data
can be generated, showing guest workload performance per
thread and the migration iteration points:
$ ./tests/migration/guestperf-plot.py --output result.html \
--migration-iters --split-guest-cpu result.json
To further include host vCPU utilization and overall QEMU
utilization
$ ./tests/migration/guestperf-plot.py --output result.html \
--migration-iters --split-guest-cpu \
--qemu-cpu --vcpu-cpu result.json
NB, the 'guestperf-plot.py' command requires that you have
the plotly python library installed. eg you must do
$ pip install --user plotly
Viewing the result.html file requires that you have the
plotly.min.js file in the same directory as the HTML
output. This js file is installed as part of the plotly
python library, so can be found in
$HOME/.local/lib/python2.7/site-packages/plotly/offline/plotly.min.js
The guestperf-plot.py program can accept multiple json files
to plot, enabling results from different configurations to
be compared.
Finally, to run the entire standardized set of comparisons
$ ./tests/migration/guestperf-batch.py \
--dst-host somehost \
--mem 4 --cpus 2 \
--src-mem-bind 0 --src-cpu-bind 0,1 \
--dst-mem-bind 1 --dst-cpu-bind 2,3
--output tcp-somehost-4gb-2cpu
will store JSON files from all scenarios in the directory
named tcp-somehost-4gb-2cpu
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
Message-Id: <1469020993-29426-7-git-send-email-berrange@redhat.com>
Signed-off-by: Amit Shah <amit.shah@redhat.com>
2016-07-20 15:23:13 +02:00
|
|
|
</div>
|
|
|
|
</body>
|
|
|
|
</html>
|
2018-06-08 14:29:43 +02:00
|
|
|
""", file=fh)
|
tests: introduce a framework for testing migration performance
This introduces a moderately general purpose framework for
testing performance of migration.
The initial guest workload is provided by the included 'stress'
program, which is configured to spawn one thread per guest CPU
and run a maximally memory intensive workload. It will loop
over GB of memory, xor'ing each byte with data from a 4k array
of random bytes. This ensures heavy read and write load across
all of guest memory to stress the migration performance. While
running the 'stress' program will record how long it takes to
xor each GB of memory and print this data for later reporting.
The test engine will spawn a pair of QEMU processes, either on
the same host, or with the target on a remote host via ssh,
using the host kernel and a custom initrd built with 'stress'
as the /init binary. Kernel command line args are set to ensure
a fast kernel boot time (< 1 second) between launching QEMU and
the stress program starting execution.
None the less, the test engine will initially wait N seconds for
the guest workload to stablize, before starting the migration
operation. When migration is running, the engine will use pause,
post-copy, autoconverge, xbzrle compression and multithread
compression features, as well as downtime & bandwidth tuning
to encourage completion. If migration completes, the test engine
will wait N seconds again for the guest workooad to stablize on
the target host. If migration does not complete after a preset
number of iterations, it will be aborted.
While the QEMU process is running on the source host, the test
engine will sample the host CPU usage of QEMU as a whole, and
each vCPU thread. While migration is running, it will record
all the stats reported by 'query-migration'. Finally, it will
capture the output of the stress program running in the guest.
All the data produced from a single test execution is recorded
in a structured JSON file. A separate program is then able to
create interactive charts using the "plotly" python + javascript
libraries, showing the characteristics of the migration.
The data output provides visualization of the effect on guest
vCPU workloads from the migration process, the corresponding
vCPU utilization on the host, and the overall CPU hit from
QEMU on the host. This is correlated from statistics from the
migration process, such as downtime, vCPU throttling and iteration
number.
While the tests can be run individually with arbitrary parameters,
there is also a facility for producing batch reports for a number
of pre-defined scenarios / comparisons, in order to be able to
get standardized results across different hardware configurations
(eg TCP vs RDMA, or comparing different VCPU counts / memory
sizes, etc).
To use this, first you must build the initrd image
$ make tests/migration/initrd-stress.img
To run a a one-shot test with all default parameters
$ ./tests/migration/guestperf.py > result.json
This has many command line args for varying its behaviour.
For example, to increase the RAM size and CPU count and
bind it to specific host NUMA nodes
$ ./tests/migration/guestperf.py \
--mem 4 --cpus 2 \
--src-mem-bind 0 --src-cpu-bind 0,1 \
--dst-mem-bind 1 --dst-cpu-bind 2,3 \
> result.json
Using mem + cpu binding is strongly recommended on NUMA
machines, otherwise the guest performance results will
vary wildly between runs of the test due to lucky/unlucky
NUMA placement, making sensible data analysis impossible.
To make it run across separate hosts:
$ ./tests/migration/guestperf.py \
--dst-host somehostname > result.json
To request that post-copy is enabled, with switchover
after 5 iterations
$ ./tests/migration/guestperf.py \
--post-copy --post-copy-iters 5 > result.json
Once a result.json file is created, a graph of the data
can be generated, showing guest workload performance per
thread and the migration iteration points:
$ ./tests/migration/guestperf-plot.py --output result.html \
--migration-iters --split-guest-cpu result.json
To further include host vCPU utilization and overall QEMU
utilization
$ ./tests/migration/guestperf-plot.py --output result.html \
--migration-iters --split-guest-cpu \
--qemu-cpu --vcpu-cpu result.json
NB, the 'guestperf-plot.py' command requires that you have
the plotly python library installed. eg you must do
$ pip install --user plotly
Viewing the result.html file requires that you have the
plotly.min.js file in the same directory as the HTML
output. This js file is installed as part of the plotly
python library, so can be found in
$HOME/.local/lib/python2.7/site-packages/plotly/offline/plotly.min.js
The guestperf-plot.py program can accept multiple json files
to plot, enabling results from different configurations to
be compared.
Finally, to run the entire standardized set of comparisons
$ ./tests/migration/guestperf-batch.py \
--dst-host somehost \
--mem 4 --cpus 2 \
--src-mem-bind 0 --src-cpu-bind 0,1 \
--dst-mem-bind 1 --dst-cpu-bind 2,3
--output tcp-somehost-4gb-2cpu
will store JSON files from all scenarios in the directory
named tcp-somehost-4gb-2cpu
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
Message-Id: <1469020993-29426-7-git-send-email-berrange@redhat.com>
Signed-off-by: Amit Shah <amit.shah@redhat.com>
2016-07-20 15:23:13 +02:00
|
|
|
|
|
|
|
def generate(self, filename):
|
|
|
|
if filename is None:
|
|
|
|
self.generate_html(sys.stdout)
|
|
|
|
else:
|
|
|
|
with open(filename, "w") as fh:
|
|
|
|
self.generate_html(fh)
|