llvm-project/libc/utils/benchmarks
Anthony Steinhauser 504d8d9d8a [libc] Fixing the build command for benchmarks.
Building libc without clang fails with:
CMake Error at /home/asteinhauser/llvm-project/libc/CMakeLists.txt:49 (message):
'clang' and 'clang-tools-extra' are required in LLVM_ENABLE_PROJECTS to
lint llvm-libc. The linting step performs important checks to help prevent
the introduction of subtle bugs, but it may increase build times.

Reviewers: sivachandra

Reviewed By: sivachandra

Differential Revision: https://reviews.llvm.org/D80495
2020-05-28 11:19:20 -07:00
..
CMakeLists.txt [libc] Change target name for testing benchmark utils infrastructure. 2020-05-06 01:22:15 -07:00
JSON.cpp [libc] Fix benchmarks build. 2020-04-28 14:30:48 -07:00
JSON.h [libc][NFC] Make all top of file comments consistent. 2020-04-08 10:18:37 -07:00
JSONTest.cpp [libc][NFC] Make all top of file comments consistent. 2020-04-08 10:18:37 -07:00
LibcBenchmark.cpp Add StringRef include to libc benchmark. 2020-04-28 13:12:26 -07:00
LibcBenchmark.h [libc][NFC] Make all top of file comments consistent. 2020-04-08 10:18:37 -07:00
LibcBenchmarkTest.cpp [libc][NFC] Make all top of file comments consistent. 2020-04-08 10:18:37 -07:00
LibcMemoryBenchmark.cpp [libc][NFC] Make all top of file comments consistent. 2020-04-08 10:18:37 -07:00
LibcMemoryBenchmark.h [libc][NFC] Make all top of file comments consistent. 2020-04-08 10:18:37 -07:00
LibcMemoryBenchmarkMain.cpp [libc] Fix benchmarks build. 2020-04-28 14:30:48 -07:00
LibcMemoryBenchmarkMain.h [libc][NFC] Make all top of file comments consistent. 2020-04-08 10:18:37 -07:00
LibcMemoryBenchmarkTest.cpp [libc] Fix warnings on release build. 2020-05-07 11:56:11 -07:00
Memcmp.cpp [libc][NFC] Make all top of file comments consistent. 2020-04-08 10:18:37 -07:00
Memcpy.cpp [libc][NFC] Make all top of file comments consistent. 2020-04-08 10:18:37 -07:00
Memset.cpp [libc][NFC] Make all top of file comments consistent. 2020-04-08 10:18:37 -07:00
RATIONALE.md
README.md [libc] Fixing the build command for benchmarks. 2020-05-28 11:19:20 -07:00
configuration_big.json
configuration_small.json
render.py3

README.md

Libc mem* benchmarks

This framework has been designed to evaluate and compare relative performance of memory function implementations on a particular host.

It will also be use to track implementations performances over time.

Quick start

Setup

Python 2 being deprecated it is advised to used Python 3.

Then make sure to have matplotlib, scipy and numpy setup correctly:

apt-get install python3-pip
pip3 install matplotlib scipy numpy

You may need python3-gtk or similar package for displaying benchmark results.

To get good reproducibility it is important to make sure that the system runs in performance mode. This is achieved by running:

cpupower frequency-set --governor performance

Run and display memcpy benchmark

The following commands will run the benchmark and display a 95 percentile confidence interval curve of time per copied bytes. It also features host informations and benchmarking configuration.

cd llvm-project
cmake -B/tmp/build -Sllvm -DLLVM_ENABLE_PROJECTS='clang;clang-tools-extra;libc' -DCMAKE_BUILD_TYPE=Release
make -C /tmp/build -j display-libc-memcpy-benchmark-small

The display target will attempt to open a window on the machine where you're running the benchmark. If this may not work for you then you may want render or run instead as detailed below.

Benchmarking targets

The benchmarking process occurs in two steps:

  1. Benchmark the functions and produce a json file
  2. Display (or renders) the json file

Targets are of the form <action>-libc-<function>-benchmark-<configuration>

  • action is one of :
    • run, runs the benchmark and writes the json file
    • display, displays the graph on screen
    • render, renders the graph on disk as a png file
  • function is one of : memcpy, memcmp, memset
  • configuration is one of : small, big

Benchmarking regimes

Using a profiler to observe size distributions for calls into libc functions, it was found most operations act on a small number of bytes.

Function % of calls with size ≤ 128 % of calls with size ≤ 1024
memcpy 96% 99%
memset 91% 99.9%
memcmp1 99.5% ~100%

Benchmarking configurations come in two flavors:

  • small
    • Exercises sizes up to 1KiB, representative of normal usage
    • The data is kept in the L1 cache to prevent measuring the memory subsystem
  • big
    • Exercises sizes up to 32MiB to test large operations
    • Caching effects can show up here which prevents comparing different hosts

1 - The size refers to the size of the buffers to compare and not the number of bytes until the first difference.

Superposing curves

It is possible to merge several json files into a single graph. This is useful to compare implementations.

In the following example we superpose the curves for memcpy, memset and memcmp:

> make -C /tmp/build run-libc-memcpy-benchmark-small run-libc-memcmp-benchmark-small run-libc-memset-benchmark-small
> python libc/utils/benchmarks/render.py3 /tmp/last-libc-memcpy-benchmark-small.json /tmp/last-libc-memcmp-benchmark-small.json /tmp/last-libc-memset-benchmark-small.json

Useful render.py3 flags

  • To save the produced graph --output=/tmp/benchmark_curve.png.
  • To prevent the graph from appearing on the screen --headless.

Under the hood

To learn more about the design decisions behind the benchmarking framework, have a look at the RATIONALE.md file.