Skip to content

Commit ea5ebcd

Browse files
Initial implementation
0 parents  commit ea5ebcd

File tree

6 files changed

+1446
-0
lines changed

6 files changed

+1446
-0
lines changed

.gitignore

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
/Cargo.lock
2+
/target

Cargo.toml

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
[package]
2+
name = "drop-tracker"
3+
version = "0.1.0"
4+
edition = "2021"
5+
authors = ["Andrea Corbellini <[email protected]>"]
6+
license = "BSD-3-Clause"
7+
8+
description = "Crate to check when a variable gets dropped. Useful for testing wrappers and containers that use unsafe memory management."
9+
repository = "https://github.com/andreacorbellini/rust-drop-tracker"
10+
11+
keywords = ["testing", "drop"]
12+
categories = ["development-tools::testing", "rust-patterns"]
13+
14+
[dependencies]

LICENSE.txt

+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
Copyright 2022 Andrea Corbellini
2+
3+
Redistribution and use in source and binary forms, with or without
4+
modification, are permitted provided that the following conditions are met:
5+
6+
1. Redistributions of source code must retain the above copyright notice, this
7+
list of conditions and the following disclaimer.
8+
9+
2. Redistributions in binary form must reproduce the above copyright notice,
10+
this list of conditions and the following disclaimer in the documentation
11+
and/or other materials provided with the distribution.
12+
13+
3. Neither the name of the copyright holder nor the names of its contributors
14+
may be used to endorse or promote products derived from this software without
15+
specific prior written permission.
16+
17+
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
18+
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
19+
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
20+
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
21+
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22+
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
23+
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
24+
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
25+
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26+
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

README.md

+94
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,94 @@
1+
Rust crate to check if a variable got correctly [dropped]. This crate is mostly useful in unit
2+
tests for code involving [`ManuallyDrop`], [`MaybeUninit`], unsafe memory management,
3+
custom containers, and more.
4+
5+
More specifically, this crate allows you to test if a variable is alive or has been dropped, and
6+
also detects when a variable gets dropped twice. These features can be used to detect bugs in your
7+
custom wrappers or containers that make use of unsafe memory management and cannot be checked at
8+
compile time by the Rust compiler.
9+
10+
[dropped]: https://doc.rust-lang.org/reference/destructors.html
11+
[`ManuallyDrop`]: std::mem::ManuallyDrop
12+
[`MaybeUninit`]: std::mem::MaybeUninit
13+
14+
# Concepts
15+
16+
The main struct of this crate is `DropTracker`. Once you initialize a tracker, you call
17+
`DropTracker::track` on it to get a `DropItem`. Each drop item is identified by a key;
18+
the key can be used at any time to check the state of the item and see if it's alive or
19+
if it has been dropped.
20+
21+
# Examples
22+
23+
This is how you would test that a container like [`Vec`] drops all its items when the container
24+
is dropped:
25+
26+
```
27+
use drop_tracker::DropTracker;
28+
29+
let mut tracker = DropTracker::new();
30+
31+
// Create a new vector and add a bunch of elements to it. The elements in this case are
32+
// identified by integer key (1, 2, 3), but any hashable type would work.
33+
//
34+
// Labels are only used to identify the elements within the tracker, and are not passed
35+
// around. In this example, the integers 1, 2 and 3 are not placed into the vector, but are
36+
// kept into the DropTracker.
37+
let v = vec![tracker.track(1),
38+
tracker.track(2),
39+
tracker.track(3)];
40+
41+
// Assert that all elements in the vector are alive
42+
tracker.all_alive(1..=3)
43+
.expect("expected all elements to be alive");
44+
45+
// Once the vector is dropped, all items should be dropped with it
46+
drop(v);
47+
tracker.all_dropped(1..=3)
48+
.expect("expected all elements to be dropped");
49+
```
50+
51+
This is how you would test a struct that involves [`MaybeUninit`]:
52+
53+
```
54+
# #![allow(dead_code)]
55+
use std::mem::MaybeUninit;
56+
57+
struct MyOption<T> {
58+
set: bool,
59+
data: MaybeUninit<T>,
60+
}
61+
62+
impl<T> MyOption<T> {
63+
fn none() -> Self {
64+
Self { set: false, data: MaybeUninit::uninit() }
65+
}
66+
67+
fn some(x: T) -> Self {
68+
Self { set: true, data: MaybeUninit::new(x) }
69+
}
70+
}
71+
72+
// BUG: MyOption<T> does not implement Drop!
73+
// BUG: The instance inside `data` may be initialized but not be properly destructed!
74+
75+
// BUG: The following code will silently leak memory:
76+
let opt = MyOption::some(String::from("hello"));
77+
drop(opt); // the String does not get deallocated
78+
79+
// DropTracker is able to catch this sort of bugs:
80+
use drop_tracker::DropTracker;
81+
82+
let mut tracker = DropTracker::new();
83+
let opt = MyOption::some(tracker.track("item"));
84+
85+
tracker.state(&"item")
86+
.alive()
87+
.expect("item is expected to be alive"); // works
88+
89+
drop(opt);
90+
91+
tracker.state(&"item")
92+
.dropped()
93+
.expect("item is expected to be dropped"); // panics, meaning that the bug was detected
94+
```

0 commit comments

Comments
 (0)