Skip to content

Commit

Permalink
Improve syscalls worker
Browse files Browse the repository at this point in the history
Remove unneeded thread spawning (this will produce more load), allow to
run in a tight loop and to specify the syscall.
  • Loading branch information
erthalion committed Jan 7, 2025
1 parent 73912ca commit 0bab235
Show file tree
Hide file tree
Showing 2 changed files with 88 additions and 19 deletions.
29 changes: 28 additions & 1 deletion src/lib.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
use core_affinity::CoreId;
use serde::Deserialize;
use std::fmt::Display;
use syscalls::Sysno;

pub mod worker;

Expand Down Expand Up @@ -42,6 +43,18 @@ fn default_duration() -> u64 {
0
}

fn default_syscalls_arrival_rate() -> f64 {
0.0
}

fn default_syscalls_tight_loop() -> bool {
false
}

fn default_syscalls_syscall_nr() -> u32 {
Sysno::getpid as u32
}

/// Workload specific configuration, contains one enum value for each
/// workload type.
#[derive(Debug, Copy, Clone, Deserialize)]
Expand Down Expand Up @@ -69,7 +82,16 @@ pub enum Workload {
/// How to invoke syscalls
Syscalls {
/// How often to invoke a syscall.
#[serde(default = "default_syscalls_arrival_rate")]
arrival_rate: f64,

/// Run in a tight loop
#[serde(default = "default_syscalls_tight_loop")]
tight_loop: bool,

/// Which syscall to trigger
#[serde(default = "default_syscalls_syscall_nr")]
syscall_nr: u32,
},

/// How to open network connections
Expand Down Expand Up @@ -295,7 +317,12 @@ mod tests {
..
} = config;
assert_eq!(restart_interval, 10);
if let Workload::Syscalls { arrival_rate } = workload {
if let Workload::Syscalls {
arrival_rate,
tight_loop,
syscall_nr,
} = workload
{
assert_eq!(arrival_rate, 10.0);
} else {
panic!("wrong workload type found");
Expand Down
78 changes: 60 additions & 18 deletions src/worker/syscalls.rs
Original file line number Diff line number Diff line change
@@ -1,7 +1,8 @@
use std::time::Instant;
use std::{fmt::Display, thread, time};

use core_affinity::CoreId;
use log::{info, warn};
use log::{info, trace};
use rand::{thread_rng, Rng};
use rand_distr::Exp;
use syscalls::{syscall, Sysno};
Expand All @@ -22,12 +23,21 @@ impl SyscallsWorker {
}
}

fn do_syscall(&self) -> std::io::Result<()> {
match unsafe { syscall!(Sysno::getpid) } {
Ok(_) => Ok(()),
Err(err) => {
warn!("Syscall failed: {}", err);
Ok(())
fn do_syscall(&self, syscall: Sysno) {
#[cfg(debug_assertions)]
{
match unsafe { syscall!(syscall) } {
Ok(_) => (),
Err(err) => {
info!("Syscall failed: {}", err);
}
}
}
#[cfg(not(debug_assertions))]
{
unsafe {
// Some syscalls are expected to fail, ignore the result
let _ = syscall!(syscall);
}
}
}
Expand All @@ -37,29 +47,61 @@ impl Worker for SyscallsWorker {
fn run_payload(&self) -> Result<(), WorkerError> {
info!("{self}");

let Workload::Syscalls { arrival_rate } = self.workload.workload else {
let mut counter = 0;
let mut start = Instant::now();

let Workload::Syscalls {
arrival_rate,
tight_loop,
syscall_nr,
} = self.workload.workload
else {
unreachable!()
};

let exp = Exp::new(arrival_rate).unwrap();
let rng = thread_rng();
let mut rng_iter = rng.sample_iter(exp);

let syscall = Sysno::from(syscall_nr);
info!("Running syscall {syscall}");

loop {
let worker = *self;
thread::spawn(move || {
worker.do_syscall().unwrap();
});

let interval: f64 =
thread_rng().sample(Exp::new(arrival_rate).unwrap());
info!(
if start.elapsed().as_secs() > 10 {
info!(
"CPU {}, {}",
self.config.cpu.id,
counter / start.elapsed().as_secs()
);
start = Instant::now();
counter = 0;
}

counter += 1;
// Do the syscall directly, without spawning a thread (it would
// introduce too much overhead for a quick syscall).
worker.do_syscall(syscall);

// If running in a tight loop, go to the next iteration
if tight_loop {
continue;
}

// Otherwise calculate waiting time
let interval: f64 = rng_iter.next().unwrap();
trace!(
"{}-{}: Interval {}, rounded {}",
self.config.cpu.id,
self.config.process,
interval,
(interval * 1000.0).round() as u64
(interval * 1000000.0).round() as u64
);
thread::sleep(time::Duration::from_millis(
(interval * 1000.0).round() as u64,
thread::sleep(time::Duration::from_nanos(
(interval * 1000000.0).round() as u64,
));
info!("{}-{}: Continue", self.config.cpu.id, self.config.process);
trace!("{}-{}: Continue", self.config.cpu.id, self.config.process);
}
}
}
Expand Down

0 comments on commit 0bab235

Please sign in to comment.