The tool supports detecting deadlocks caused by incorrect use of mutexes (std::sync::Mutex
) and condition variables (std::sync::Condvar
).
It also supports detecting deadlocks caused by calling join
on a thread that never returns.
It does this by translating the Mid-level Intermediate Representation (MIR) representation of the Rust source code to a Petri net, a mathematical and graphical model. The Petri net is then analyzed by the model checker LoLA to find out if the net can reach a deadlock. This approach is an exhaustive check of all possible program states. It is not just testing a couple of possible executions, it is also not fuzz testing.
For more details about what works and what does not, see Limitations. For more context about this project, see Context.
crates.io
Assuming you already have Rust installed on your system, simply run:
cargo install cargo-check-deadlock
You must then install the model checker as explained in the section Model checker.
To get a local copy for development up and running follow these simple example steps.
Install Rust using rustup
as described on the Rust Website
Install the latest nightly version with the rustc-dev
component running:
rustup toolchain install nightly
Activate the nightly toolchain:
rustup default nightly
Install the components needed to work with the compiler internals
rustup component add rust-src rustc-dev llvm-tools-preview
The project must be compiled with the nightly toolchain to access the private crates of the compiler.
The toolchain file rust-toolchain
in the root folder overrides the currently active toolchain for this project.
See the rustup
documentation for more information: https://rust-lang.github.io/rustup/overrides.html#the-toolchain-file
The settings.json
configures VS Code to instruct the rust-analyzer extension to auto-discover the right toolchain.
This proves extremely useful to get feedback on the types, compiler errors, etc. that appear when working with the private crates of rustc
.
As time goes on and the compiler internals change, the code will inevitably need changes to work again.
The current state of the repository compiles without warnings and with all tests passing with
rustc 1.72.0-nightly (065a1f5df 2023-06-21)
Clone the repo
git clone https://github.com/hlisdero/cargo-check-deadlock.git
Make sure that the sysroot points to a nightly toolchain when running it from the project directory
The output should be something like $HOME/.rustup/toolchains/nightly-<platform>
rustc --print=sysroot
Build the project with cargo
cargo build
Run the tests to check that everything works with cargo
cargo test
Write a valid Rust program that compiles correctly, e.g. rust_program.rs
, then run
cargo check-deadlock <path_to_program>/rust_program.rs
The result is printed on stdout. A file named net.lola
should appear in the CWD.
If you would like to export to other formats or use a custom filename or output folder, use
cargo check-deadlock <path_to_program>/rust_program.rs --dot --pnml --filename=example --output-folder=output/
In this case, files named example.pnml
and example.dot
should appear in the output/
folder.
To obtain the full list of CLI options, use the --help
flag.
Note: For more examples, please refer to the integration tests.
The program supports the verbosity flags defined in the crate clap_verbosity_flag.
For example, running the program with the flag -vvv
prints debug messages that can be useful for pinpointing which line of the MIR representation is not being translated correctly.
cargo check-deadlock <path_to_program>/rust_program.rs -vvv
LoLA model checker supports printing a "witness path" that shows a sequence of transition firings leading to a deadlock.
This is very useful when extending the translator and the Petri net does not match the expected result for a given program.
A convenient script can be found to print the witness path for a .lola
file.
To see the MIR representation of the source code, you can compile the code with the corresponding flag: rustc --emit=mir <path_to_source_code>
To graph a net in .dot
format, install the dot
tool following the instructions on the GraphViz website.
Run dot -Tpng net.dot -o outfile.png
to generate a PNG image from the resulting .dot
file.
Run dot -Tsvg net.dot -o outfile.svg
to generate a SVG image from the resulting .dot
file.
More information and other formats can be found in the documentation.
To see the MIR representation of the source code, you may use the Rust Playground. Simply select the option "MIR" instead of "Run" in the dropdown menu. Remember to select the nightly version too.
To graph a given DOT result, you may use the Graphviz Online tool by dreampuf.
Alternatively, you may use Edotor or Sketchviz.
The model checker LoLA can be downloaded here. It must be compiled from the source code.
An alternative mirror with detailed instructions is available on GitHub: https://github.com/hlisdero/lola
A last option is to copy the precompiled 64-bit executable ./assets/lola
to the $PATH
. A script for this purpose can be found in the repo.
Support for other model checkers and export formats may be added in the future. Adding other backends could be a great way to compare their performance and accuracy. The export formats are implemented in the custom Petri net library used in this project: https://github.com/hlisdero/netcrab
As Rust is a very complex language, supporting all the cases in which a deadlock may arise is impossible to do in practice.
The goal of this project is to demonstrate that an approach using Petri nets is feasible and could detect errors at compile time, therefore enhancing the safety and reliability of Rust code.
The most difficult case to detect at the moment is lost signals. This particular deadlock case arises when a thread calls notify_one
on a condition variable before another thread called wait
.
It is recommended to check out the example programs to see which kinds of programs can be translated and analyzed successfully. Particularly interesting examples are the dining philosophers problem and the producer-consumer problem.
Currently, the programs that the translator can deal with are fairly limited:
struct
s, enum
s, or impl
blocks are supported.Contributions are what makes the open-source community such an amazing place to learn, inspire, and create. Any contributions you make are greatly appreciated.
If you have a suggestion that would make this better, please fork the repo and create a pull request. You can also simply open an issue with the tag "enhancement". Don't forget to give the project a star! Thanks again!
git checkout -b feature/AmazingFeature
)git commit -m 'Add some AmazingFeature'
)git push origin feature/AmazingFeature
)Distributed under the terms of both the MIT license and the Apache License (Version 2.0). See LICENSE-MIT, LICENSE-APACHE for more information.
This project extends and reimplements ideas from the original work by Tom Meyer found in https://github.com/Skasselbard/Granite as part of the master thesis titled "A Petri-Net Semantics for Rust".
The rustc dev guide was a trusty guide in the journey of understanding the compiler. It is a must-read for every interested contributor!
The nightly documentation was also a valuable resource to figure out the details of MIR and how to translate every function.
This project is part of Horacio Lisdero Scaffino's undergraduate thesis titled "Compile-time Deadlock Detection in Rust using Petri Nets" at the School of Engineering of the University of Buenos Aires.
The thesis is publicly available on GitHub: https://github.com/hlisdero/thesis
It provides important background topics and implementation details about this project.