Valgrind Client Request interface for Rust programs
[crates.io]: https://crates.io/crates/crabgrind [libs.rs]: https://lib.rs/crates/crabgrind [documentation]: https://docs.rs/crabgrind [license]: https://github.com/2dav/crabgrind/blob/main/LICENSE [][crates.io] [][libs.rs] [][documentation] [][license]crabgrind
wraps various Valgrind macros in C functions, compiles and links against
the resulting binary, and exposes an unsafe interface to allow Rust programs running under Valgrind to
interact with the tools and environment.
crabgrind
imports macros from Valgrind's header files, so they must be accessible to build the project.
If you installed Valgrind using your OS-specific package manager, the header files will be placed at the paths
according to your OS's conventions, and most likely cc
, the build
tool crabgrind
uses, will find them.
If you have installed Vallgrind manually or having any issues, you can set DEP_VALGRIND
environment variable to the appropriate path, if
one is specified, its value will be directly passed to cc::Build::include
.
env DEP_VALGRIND=/path/to/valgrind cargo build
Add the following to your Cargo.toml
file:
toml
[dependencies]
crabgrind = "^0.1"
Next, use some of the Valgrind's API ```rust use crabgrind as cg;
fn main() { if matches!(cg::run_mode(), cg::RunMode::Native) { println!("run me under Valgrind"); } else { cg::println!("Hey, Valgrind!"); } } ``` And run your application under Valgrind, either with handy cargo-valgrind
cargo valgrind run
or manually
cargo build
valgrind ./target/debug/appname
Valgrind provides VALGRIND_PRINTF_BACKTRACE
macro to print the message with the stack-trace attached,
crabgrind::print_stacktrace
is it's crabbed wrapper.
```rust
use crabgrind as cg;
fn printtrace(){ let mode = cg::runmode(); cg::print_stacktrace!("current mode: {mode:?}"); }
print_trace(); ```
One way to do this would be to turn off stats collection at stratup with the
--collect-atstart=no
callgrind command-line attribute, and enable/disable it from the code with callgrind::toggle_collect
```rust use crabgrind as cg;
// ... some expensive initialization
cg::callgrind::togglecollect(); // code of interest cg::callgrind::togglecollect();
// ... some deinitialization ```
We can run on the real CPU instead of the virtual one using valgrind::non_simd_call
,
refer to valgrind.h
for details on limitations and various ways to crash.
```rust use crabgrind as cg;
let mut state = 0; cg::valgrind::nonsimdcall(|tid| { // uncomment following line to see "the 'impossible' happened" // println!("tid: {tid}"); state = tid; });
println!("tid: {state}"); ```
We'll use Massif
tool and the monitor command
interface to run the corresponding Massif command.
```rust
use crabgrind as cg;
let heap = String::from("alloca");
if cg::monitorcommand("snapshot mem.snapshot").isok(){ println!("snapshot is saved to \"mem.snapshot\""); } ```
```rust use crabgrind as cg;
fn factorial1(num: u128) -> u128 { match num { 0 => 1, 1 => 1, _ => factorial(num - 1) * num, } }
fn factorial2(num: u128) -> u128 { (1..=num).product() }
cg::zero_stats();
let a = factorial1(20); cg::dump_stats("factorial1");
let b = factorial2(20); cg::dump_stats("factorial2");
asserteq!(a,b); cg::dumpstats(None); ```
from Valgrind docs
The code added to your binary has negligible performance impact: on x86, amd64, ppc32, ppc64 and ARM, the overhead is 6 simple integer instructions and is probably undetectable except in tight loops.
... the code does nothing when not run on Valgrind, so you are not forced to run your program under Valgrind just because you use the macros in this file.
however,
- wrapping each macros in a function implies function call overhead regardless of the run mode
- functions that returns std::result::Result
involve branching
- functions that takes strings as a parameters internally converts them to std::ffi::CString
If you wish to compile out all (crab)Valgrind from the binary, you can wrap crabgrind
calls with
the feature-gate.
No
Tests must be run under Valgrind, as of now cargo-valgrind
fits nicely, it allows to compile and run tests under Valgrind in one command
cargo valgrind test
crabgrind
is distributed under the same license terms as the Valgrind
that is GPL version 2.