Commuter is an automated scalability testing tool that hunts down unnecessary sharing in your code. Unlike traditional scalability testing, Commuter doesn't require handcrafted workloads or even multicore hardware. Starting with a high-level software interface model written in Python -- not some esoteric specification language -- Commuter automatically generates thousands of test cases for your interface and tests your implementation to root out the exact causes and locations of unnecessary, scalability-limiting sharing.
Commuter is currently a research prototype. It has been used to successfully analyze an 18 call model of the POSIX file system and virtual memory APIs and test the scalability of these calls in both Linux and our research OS kernel, sv6.
Check out Commuter's home page.
Quick start
To simplify getting up and running, Commuter includes a "setup" script
that can download and build all of Commuter's unusual dependencies
with default configurations, as well as both sv6 and Mtrace-enabled
Linux. You can run ./setup all
, then jump straight to "Running the
POSIX model" below. You can run ./setup
with different arguments to
install just some of the components.
Dependencies
Commuter depends on several other software packages, which themselves depend on other packages. You can either use Commuter's setup script to install these, or install them yourself.
-
Z3 - Z3 is available from https://github.com/Z3Prover/z3 . As of this writing you'll need a nightly build, since Commuter depends on several bug fixes that are not in the stable 4.3.0 release. You can download pre-built nightlies for various OSs from Z3's home page, or clone their unstable branch.
You may either install Z3 system-wide, or set the environment variables
LD_LIBRARY_PATH
andPYTHONPATH
to the local installation of Z3 before running Commuter.The results in the SOSP '13 paper were generated with Z3 commit
a60b53bfd
from the unstable branch, but it's likely that any nightly build will work. -
mtrace - Mtrace is our modified version of QEMU that Commuter uses to run tests and detect sharing in OS kernels. Mtrace supports many other types of analyses, too, so it lives in a separate repository at
git clone https://github.com/aclements/mtrace.git
See mtrace's
README.md
for build instructions.
Operating systems
Running an operating system in Commuter requires a small amount of special support (specifically, hypercalls to mtrace). Commuter has been used with the following operating systems:
-
Linux - The one and only. Our mtrace-enabled version of Linux can be found at
git clone https://github.com/aclements/linux-mtrace.git
See the
README.md
in the mtrace repository for build instructions and more information on using mtrace with Linux.Note that the we use an unconventional Linux setup. To keep the virtual machine minimal, we run it with no disk, completely from an initramfs. However, the initramfs does not contain a typical Linux user-space; to keep the sv6 and Linux environments as similar as possible, the initramfs contains the sv6 user-space, compiled for Linux. This is probably not the best setup in the long run, but it was expedient.
-
sv6 - The scalable POSIX-like operating system we built with the help of Commuter. sv6 includes the RadixVM virtual memory system and the ScaleFS file system, published in EuroSys '13 and SOSP '13, respectively. sv6 can be found at
git clone https://github.com/aclements/sv6.git
Building dependencies
The setup
script will download and build all of Commuter's
dependencies and both operating systems. However, if you do this by
hand,
-
Make sure mtrace is checked out in
ext/mtrace
and that you've compiled the mtrace version of QEMU as well as the tools inext/mtrace/mtrace-tools
. -
Make sure sv6 is checked out in
ext/sv6
. You don't need to compile it, since the below instructions compile it with special options. -
Make sure linux-mtrace is checked out in
ext/linux-mtrace
and compiled.
You can of course use a directory other than ext
, but the directions
below assume dependencies are checked out in ext
.
Commuter components
Commuter consists of several separate components designed to make it easy to run (or re-run) just parts of the Commuter process.
-
spec.py
is the main entry point of Commuter,spec.py
runs a model and its test generator and optionally produces commutativity conditions, test case code, and a machine-readable "model.out" summary of all explored paths and tests.spec.py
can also optionally analyze interface idempotence. -
par-spec.py
is a parallel driver forspec.py
. Takes all the same arguments asspec.py
. -
split-testgen.py
splits test case code output into several files to enable parallel compilation. -
par-mtrace.py
is a driver for running an OS kernel on the generated tests under mtrace to produce a memory access log. -
par-mscan.py
is a driver for processing memory access logs to produce access conflict reports.
Models
Commuter comes with several models. The main POSIX file system and
virtual memory model is models.fs
. There are also several toy
models under models/
.
Tools
There are several tools for analyzing output files under tools/
.
-
profile
generates statistics on a "model.out" file (and optionally its delta from another "model.out" file). -
idem
slices and dices idempotence data in "model.out" files in various ways. -
make-heatmaps
generates heat map figures from mscan output in either SVG or TikZ format.
Viewer
There is a web-based interactive data visualizer in viewer/
. See
its README.
Running the POSIX model
Generate test cases
# Run model; go get lunch
./par-spec.py models.fs -t testgen.c -m model.out --max-tests-per-path 500
# Split testgen.c for parallel build
./split-testgen.py -d ext/sv6/libutil < testgen.c
Neither Linux nor sv6 support testing reboot
, so if your goal is
only to examine cache line sharing, you can speed up spec.py
by
passing -f '!reboot'
.
Furthermore, sv6 does not have an on-disk file system, so sync
and
fsync
are no-ops in sv6. To disable test generation for these as
well, pass -f '!reboot,!sync,!fsync'
to spec.py
.
Check cache line sharing on sv6 (serial version)
cd ext/sv6
make HW=mtrace RUN='fstest -t; halt' mtrace.out
../mtrace/mtrace-tools/mscan --check-testcases --kernel o.mtrace/kernel.elf > mscan-sv6.out
Check cache line sharing on sv6 (parallel version)
cd ext/sv6
../../par-mtrace.py -m xv6
../../par-mscan.py --kernel o.mtrace/kernel.elf > mscan-sv6.out
Check cache line sharing on Linux with ramfs (serial version)
cd ext/sv6
# Build the Linux initramfs for mtrace
make HW=linuxmtrace
# Run Linux kernel with tracing
make HW=linuxmtrace KERN=../linux-mtrace/arch/x86_64/boot/bzImage RUN='fstest -t;halt' mtrace.out
# Check sharing
../mtrace/mtrace-tools/mscan --kernel ../linux-mtrace/vmlinux --check-testcases > mscan-linux.out
Check cache line sharing on Linux with ramfs (parallel version)
cd ext/sv6
../../par-mtrace.py -m linux
../../par-mscan.py --kernel ../linux-mtrace/vmlinux > mscan-linux.out
Check cache line sharing on Linux with disk-based FS
This requires an mtrace kernel compiled with CONFIG_DEVTMPFS=y
,
CONFIG_BLK_DEV_RAM=y
, and support for the desired file system.
cd ext/sv6
../../par-mtrace.py -m linux --fs-type ext4
../../par-mscan.py --kernel ../linux-mtrace/vmlinux > mscan-linux.out
Now you can examine mscan-sv6.out
or mscan-linux.out
for
unnecessary sharing, or use the tools in tools/
to generate
summaries, or fire up the interactive viewer in
viewer/
.