alloc_counter

Alloc counters

A redesign of the Quick and Dirty Allocation Profiling Tool.

Features

Limitations and known issues

Usage

An AllocCounter<A> wraps an allocator A to individually count the number of calls to alloc, realloc, and dealloc.

```rust use alloc_counter::AllocCounter;

type MyAllocator = std::alloc::System; const MyAllocator: MyAllocator = std::alloc::System;

[global_allocator]

static A: AllocCounter = AllocCounter(MyAllocator); ```

Std-users may prefer to inherit their system's allocator.

```rust use alloc_counter::AllocCounterSystem;

[global_allocator]

static A: AllocCounterSystem = AllocCounterSystem; ```

To count the allocations of an expression, use count_alloc.

```rust

use alloccounter::{countalloc, AllocCounterSystem};

#[global_allocator]

static A: AllocCounterSystem = AllocCounterSystem;

let (counts, v) = countalloc(|| { // no alloc let mut v = Vec::new(); // alloc v.push(0); // realloc v.push(1); // return the vector without deallocating v }); asserteq!(counts, (1, 1, 0)); ```

To deny allocations for an expression use deny_alloc.

```rust,should_panic

use alloccounter::{denyalloc, AllocCounterSystem};

#[global_allocator]

static A: AllocCounterSystem = AllocCounterSystem;

fn foo(b: Box) { // dropping causes a panic deny_alloc(|| drop(b)) } foo(Box::new(0)); ```

Similar to Rust's lints, you can still allow allocation inside a deny block.

```rust

use alloccounter::{allowalloc, deny_alloc, AllocCounterSystem};

#[global_allocator]

static A: AllocCounterSystem = AllocCounterSystem;

fn foo(b: Box) { denyalloc(|| allowalloc(|| drop(b))) } foo(Box::new(0)); ```

Forbidding allocations forces a panic even when allow_alloc is used.

```rust,should_panic

use alloccounter::{allowalloc, forbid_alloc, AllocCounterSystem};

#[global_allocator]

static A: AllocCounterSystem = AllocCounterSystem;

fn foo(b: Box) { // panics because of outer forbid, even though drop happens in an allow block forbidalloc(|| allowalloc(|| drop(b))) } foo(Box::new(0)); ```

For added sugar you may use the #[no_alloc] attribute on functions, including methods with self-binds. #[no_alloc] expands to calling deny_alloc and forcefully moves the parameters into the checked block. #[no_alloc(forbid)] calls forbid_alloc.

```rust,should_panic

#[cfg(not(feature = "macros"))]

panic!("macros feature disabled");

#[cfg(feature = "macros")] {

use alloccounter::{allowalloc, no_alloc, AllocCounterSystem};

#[global_allocator]

static A: AllocCounterSystem = AllocCounterSystem;

[no_alloc(forbid)]

fn foo(b: Box) { allow_alloc(|| drop(b)) } foo(Box::new(0));

}

```

License: MIT OR Apache-2.0