Files
poco/Benchmark/README.md
Aleksandar Fabijanic 1850dc16aa Benchmark and FastLogger (#5081)
* fix(SharedLibrary): Missing DLLs not reported #5069

* fix(CMake): not producing proper binary names #5070

* fix(SharedLibrary): disable shared lib tests in static build #5069

* fix(misc): add pdjson links to gitignore, remove unused var in SharedLibrary, harden TaskManagerTest

* fic(ci): separate oracle and sqlserver odbc (out of disk space) (#5075)

* fic(ci): separate oracle and sqlserver odbc (out of disk space)

* use oracle odbc driver

* use oracle free

* ad db user

* postpone adding user after build

* remove default tablespace (does not exist)

* reinstate all ci jobs

* add postgresl odb tests to ci

* remove spurious syminks

* fix gitignore (pdjson)

* Remove VS projects #5076

* chore: revert leftover ODB IP address

* fix(CodeQL): float comparison alerts

* fix: compile errors

* chore: upgrade asan to macos-14 (tryout)

* fix: .gitignore symlinks; XML Makefile wrong pattern

* Optimize PatternFormatter and Timezone performance #5078

PatternFormatter:
- Cache node name (Environment::nodeName()) to avoid repeated syscalls
- Add extractBasename() for efficient %O format specifier
- Add string reserve(128) to reduce reallocations during formatting

Timezone:
- Cache UTC offset to avoid repeated syscalls (8x speedup for %L patterns)
- Auto-detect TZ environment variable changes to invalidate cache
- Add reloadCache() method for explicit cache refresh

Tests:
- Add TimezoneTest::testUtcOffsetCaching()
- Add PatternFormatterTest::testExtractBasename()

* fix: use Path::separatorin extractBasename #5078

* Add Benchmark #5080

* enh(Logging): move constructors for Message and Logger #5078

* chore(AsyncNotificationCenter): eliminate MSVC warnings

* enh(build): c++20 support #5084

* feat(CppUnit): print class name
execute all named tests (not only the first one)
accept test name with class (eg. testrunner LoggerTest::testLogger) #5083

* feat(Benchmark): Add Logger/FastLogger comparison benchmarks and Windows support

- Add LoggerBench.cpp with AsyncChannel vs FastLogger benchmarks
- Add compare.sh (Linux/macOS) and compare.ps1 (Windows) scripts
- Add LOGGER_BENCHMARK.md with cross-platform benchmark results
- Update README.md with Windows build instructions (Ninja, CMAKE_PREFIX_PATH)
- Add error message when -- options are used on Windows (should use /)
- Update CMakeLists.txt and Makefile to include LoggerBench #5080

* feat(FastLogger): #5078
FastLogger provides a Poco-compatible wrapper around the Quill logging
library, offering significant performance improvements over AsyncChannel
through lock-free SPSC queues and backend thread processing.

Key features:
- Drop-in replacement for Poco::Logger with FastLogger::get()
- Support for all standard Poco channels (Console, File, Rotating, etc.)
- XML/properties configuration via FastLoggerConfigurator
- Thread affinity for backend worker on Linux and Windows
- Log file rotation with size and time-based policies

Performance (CPU time - calling thread latency):
- Linux: 31-70x faster than AsyncChannel
- Windows: 23-87x faster than AsyncChannel
- macOS: Limited improvement due to lack of thread affinity support

New files:
- Foundation/include/Poco/FastLogger.h
- Foundation/src/FastLogger.cpp
- Util/include/Poco/Util/FastLoggerConfigurator.h
- Util/src/FastLoggerConfigurator.cpp
- dependencies/quill/ (header-only Quill 7.5.0 library)

* fix(cmake): disable FastLogger on emscripten (not supported) #5078

* feat(FastLogger): add cpuAfinity config parameter #5087

* fix(FastLogger): Fix lock-order-inversion in FastLogger (TSAN) #5078

* fix(cmake): build not stripping release binaries #5085

* fix(PCRE): fails to compile with clang/c++20 #5131

* feat(AsyncChannel): add CPU affinity property #5087

* feat(SpinlockMutex): make it adaptive #5132

* feat(AsyncChannel): add CPU affinity property #5087

* chore: remove leftover file commited by mistake

* feat(build): allow FastLogger to be fully disabled at build time #5078

Build system changes:
- Add POCO_NO_FASTLOGGER compile definition in CMake when ENABLE_FASTLOGGER=OFF
  to prevent Config.h from auto-enabling FastLogger
- Add ifdef guards around FastLogger tests in LoggingTestSuite.cpp
- Exclude FastLoggerTest.cpp and FastLoggerChannelsTest.cpp from CMake build
  when FastLogger is disabled
- Add POCO_NO_FASTLOGGER support to Make build system for Foundation and Util
- Add CI jobs to verify builds work without FastLogger (CMake and Make)

Code changes:
- Add LoggingConfigurator::configure() convenience method for quick logging setup

* fix(ci): testrunner args

* chore(progen): remove leftover script #5076

* fix(test): give ANC a bit more time to process

* fix(ci): set env before test run

* chore(doc): quill license

* feat(Channel): add log(Message&&) #5133

* fix(ci): set env before test run

* fix(TestRunner): don't search children #5083

* feat: lock-free queues #5134

* feat(Benchmark): various comparisons

* chore: cleanup benchmark
2025-12-22 21:06:43 +01:00

4.1 KiB

Poco Benchmark

Performance benchmarks for Poco components using Google Benchmark.

Dependencies

  • Poco Foundation
  • Poco Util
  • Google Benchmark library

Installing Google Benchmark

macOS (Homebrew):

brew install google-benchmark

Debian/Ubuntu:

apt install libbenchmark-dev

Fedora/RHEL:

dnf install google-benchmark-devel

Windows (vcpkg):

vcpkg install benchmark:x64-windows

From source (Linux/macOS):

git clone https://github.com/google/benchmark.git
cd benchmark
cmake -DBENCHMARK_DOWNLOAD_DEPENDENCIES=on -DCMAKE_BUILD_TYPE=Release -S . -B build
cmake --build build --config Release
sudo cmake --install build

From source (Windows):

git clone https://github.com/google/benchmark.git
cd benchmark
cmake -DBENCHMARK_DOWNLOAD_DEPENDENCIES=on -DCMAKE_BUILD_TYPE=Release -DCMAKE_INSTALL_PREFIX=C:\local -S . -B build
cmake --build build --config Release
cmake --install build --config Release

Building

Using Make (Linux/macOS):

cd poco
make -C Benchmark

Using CMake (Linux/macOS):

cmake -B build -DENABLE_BENCHMARK=ON
cmake --build build --target Benchmark

Using CMake (Windows):

cmake -B build -G Ninja -DENABLE_BENCHMARK=ON -DCMAKE_PREFIX_PATH=C:\local
cmake --build build --target Benchmark

Note: On Windows, use Ninja generator for reliable target builds. The CMAKE_PREFIX_PATH should point to where Google Benchmark was installed.

Usage

./Benchmark/bin/$(uname)/$(arch)/benchmark [OPTIONS]

Options

Option Description
-h, --help Display help information
-l, --list List all available benchmarks
-f, --filter=<regex> Run only benchmarks matching regex
-t, --min-time=<seconds> Minimum time to run each benchmark
-r, --repetitions=<num> Number of times to repeat each benchmark
-a, --aggregates-only Report only aggregates (mean, median, stddev)
--format=<fmt> Output format: console, json, or csv
-o, --output=<file> Write results to file
--output-format=<fmt> Format for output file
--tabular Display counters in tabular format

Examples

List all benchmarks:

./benchmark --list

Run specific benchmarks:

./benchmark --filter="PatternFormatter.*"

Run with repetitions and show statistics:

./benchmark --repetitions=5 --aggregates-only

Export results to JSON:

./benchmark --format=json --output=results.json

Writing Benchmarks

Create a new .cpp file in src/ with your benchmarks:

// src/MyComponentBench.cpp

#include <benchmark/benchmark.h>
#include "Poco/MyComponent.h"

using Poco::MyComponent;

static void BM_MyOperation(benchmark::State& state)
{
    MyComponent component;

    for (auto _ : state)
    {
        // Code to benchmark - runs many iterations
        auto result = component.doSomething();
        benchmark::DoNotOptimize(result);
    }

    // Optional: report throughput
    state.SetBytesProcessed(state.iterations() * sizeof(result));
}
BENCHMARK(BM_MyOperation);

// Parameterized benchmark
static void BM_MyOperationSized(benchmark::State& state)
{
    int size = state.range(0);
    std::vector<int> data(size);

    for (auto _ : state)
    {
        // Benchmark with different sizes
        processData(data);
    }
}
BENCHMARK(BM_MyOperationSized)->Range(8, 8<<10);

Then add the file to CMakeLists.txt and Makefile:

CMakeLists.txt:

set(SRCS
    src/BenchmarkApp.cpp
    src/PatternFormatterBench.cpp
    src/MyComponentBench.cpp  # Add this
)

Makefile:

objects = BenchmarkApp PatternFormatterBench MyComponentBench

Tips

  • Use benchmark::DoNotOptimize() to prevent the compiler from optimizing away results
  • Use benchmark::ClobberMemory() if you need to force memory writes to be visible
  • Keep setup code outside the for (auto _ : state) loop
  • Use state.PauseTiming() / state.ResumeTiming() for expensive setup within the loop
  • Report throughput with state.SetBytesProcessed() or state.SetItemsProcessed()