Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add benchmark for user_events enabled check #133

Open
wants to merge 9 commits into
base: main
Choose a base branch
from
7 changes: 6 additions & 1 deletion opentelemetry-user-events-logs/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -23,8 +23,13 @@ opentelemetry-appender-tracing = { workspace = true }
tracing = { version = "0.1", default-features = false, features = ["std"] }
tracing-core = "0.1.31"
tracing-subscriber = { version = "0.3.0", default-features = false, features = ["registry", "std"] }
microbench = "0.5"
criterion = "0.4" # Add the Criterion crate for benchmarking

[features]
spec_unstable_logs_enabled = ["opentelemetry/spec_unstable_logs_enabled", "opentelemetry_sdk/spec_unstable_logs_enabled", "opentelemetry-appender-tracing/spec_unstable_logs_enabled"]
default = ["spec_unstable_logs_enabled"]

[[bench]]
name = "benchmark"
harness = false

110 changes: 110 additions & 0 deletions opentelemetry-user-events-logs/benches/benchmark.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,110 @@
// Run as root user as access to `tracefs` is required, or ensure the current user has the necessary permissions.
// To run benchmarks with root privileges, execute:
// sudo -E /home/<username>/.cargo/bin/cargo bench
// Replace <username> with your actual username.
//
// System Information:
// Processor: AMD EPYC 7763 64-Core Processor
// CPU Cores: 8
// Logical Processors: 16
// Memory: 64 GB

// provider_find_set_single time: [3.9862 ns 3.9898 ns 3.9937 ns]
// provider_find_set_concurrent_0threads time: [3.9937 ns 3.9978 ns 4.0024 ns]
// provider_find_set_concurrent_2threads: time: time: [111.52 ns 114.61 ns 117.38 ns]
// provider_find_set_concurrent_4threads: time: time: [199.18 ns 203.92 ns 208.43 ns]
// provider_find_set_concurrent_8threads time: time: [199.18 ns 203.92 ns 208.43 ns]

use criterion::black_box;
use criterion::{criterion_group, criterion_main, Criterion};
use eventheader_dynamic::{Provider, ProviderOptions};
use std::sync::{
atomic::{AtomicBool, Ordering},
Arc,
};
use std::thread;

/// Benchmark `find_set` with no concurrent threads
fn benchmark_find_set_single(c: &mut Criterion) {
// Setup the Provider
let mut options = ProviderOptions::new();
options = *options.group_name("testprovider");
let mut provider = Provider::new("testprovider", &options);

// Register some dummy events with specific levels and keywords
let keyword = 0x01; // Example keyword
let level = 4; // Example level (Informational)
provider.register_set(eventheader::Level::Informational, keyword);

// Benchmark the `find_set` method with `enabled` check
c.bench_function("provider_find_set_single", |b| {
b.iter(|| {
if let Some(event_set) = provider.find_set(level.into(), keyword) {
black_box(event_set.enabled()); // Check if the tracepoint is being listened to
}
});
});
}

/// Benchmark `find_set` with a parameterized number of concurrent threads
fn benchmark_find_set_concurrent(c: &mut Criterion) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

maybe we can simply re-use the existing stress test? (copy it over to contribute repo?)

let thread_counts = [0, 2, 4, 8]; // Test with 2, 4, and 8 threads

for &thread_count in &thread_counts {
// Setup the Provider
let mut options = ProviderOptions::new();
options = *options.group_name("testprovider");
let mut provider = Provider::new("testprovider", &options);

// Register some dummy events with specific levels and keywords
let keyword = 0x01; // Example keyword
let level = 4; // Example level (Informational)
provider.register_set(eventheader::Level::Informational, keyword);

// Shared Provider and stop flag
let provider = Arc::new(provider);
let stop_flag = Arc::new(AtomicBool::new(false));

// Spawn worker threads
let mut worker_handles = Vec::new();
for _ in 0..thread_count {
let provider_clone = Arc::clone(&provider);
let stop_flag_clone = Arc::clone(&stop_flag);
worker_handles.push(thread::spawn(move || {
while !stop_flag_clone.load(Ordering::Relaxed) {
if let Some(event_set) = provider_clone.find_set(level.into(), keyword) {
black_box(event_set.enabled()); // Check if tracepoint is being listened to
}
}
}));
}

// Dereference the `Arc` once before the benchmark to reduce overhead
let provider_ref: &Provider = &provider;

// Benchmark the `find_set` method with `enabled` check
let benchmark_name = format!("provider_find_set_concurrent_{}threads", thread_count);
c.bench_function(&benchmark_name, |b| {
b.iter(|| {
if let Some(event_set) = provider_ref.find_set(level.into(), keyword) {
black_box(event_set.enabled()); // Check if tracepoint is being listened to
}
});
});

// Signal worker threads to stop
stop_flag.store(true, Ordering::Relaxed);

// Wait for all worker threads to complete
for handle in worker_handles {
let _ = handle.join();
}
}
}

criterion_group!(
benches,
benchmark_find_set_single,
benchmark_find_set_concurrent
);
criterion_main!(benches);
Loading