2019-02-23 04:56:29 -06:00
|
|
|
# Miri [![Build Status](https://travis-ci.com/rust-lang/miri.svg?branch=master)](https://travis-ci.com/rust-lang/miri) [![Windows build status](https://ci.appveyor.com/api/projects/status/github/rust-lang/miri?svg=true)](https://ci.appveyor.com/project/rust-lang-libs/miri)
|
2016-02-02 04:47:28 -06:00
|
|
|
|
2016-04-13 19:41:37 -05:00
|
|
|
|
2018-11-13 06:28:09 -06:00
|
|
|
An experimental interpreter for [Rust][rust]'s
|
|
|
|
[mid-level intermediate representation][mir] (MIR). It can run binaries and
|
2019-07-25 12:38:55 -05:00
|
|
|
test suites of cargo projects and detect certain classes of
|
|
|
|
[undefined behavior](https://doc.rust-lang.org/reference/behavior-considered-undefined.html),
|
2018-11-13 06:28:09 -06:00
|
|
|
for example:
|
|
|
|
|
|
|
|
* Out-of-bounds memory accesses and use-after-free
|
|
|
|
* Invalid use of uninitialized data
|
|
|
|
* Violation of intrinsic preconditions (an [`unreachable_unchecked`] being
|
|
|
|
reached, calling [`copy_nonoverlapping`] with overlapping ranges, ...)
|
|
|
|
* Not sufficiently aligned memory accesses and references
|
2019-07-30 04:23:56 -05:00
|
|
|
* Violation of *some* basic type invariants (a `bool` that is not 0 or 1, for example,
|
2018-11-13 06:28:09 -06:00
|
|
|
or an invalid enum discriminant)
|
|
|
|
* WIP: Violations of the rules governing aliasing for reference types
|
|
|
|
|
2019-03-08 03:50:31 -06:00
|
|
|
Miri has already discovered some [real-world bugs](#bugs-found-by-miri). If you
|
|
|
|
found a bug with Miri, we'd appreciate if you tell us and we'll add it to the
|
|
|
|
list!
|
2019-02-15 03:43:27 -06:00
|
|
|
|
2019-07-30 04:23:56 -05:00
|
|
|
Be aware that Miri will not catch all cases of undefined behavior in your
|
|
|
|
program, and cannot run all programs:
|
2019-03-26 04:07:22 -05:00
|
|
|
|
|
|
|
* There are still plenty of open questions around the basic invariants for some
|
2019-05-20 02:32:01 -05:00
|
|
|
types and when these invariants even have to hold. Miri tries to avoid false
|
|
|
|
positives here, so if you program runs fine in Miri right now that is by no
|
2019-07-30 04:23:56 -05:00
|
|
|
means a guarantee that it is UB-free when these questions get answered.
|
|
|
|
|
|
|
|
In particular, Miri does currently not check that integers are initialized
|
|
|
|
or that references point to valid data.
|
2019-03-26 04:07:22 -05:00
|
|
|
* If the program relies on unspecified details of how data is laid out, it will
|
|
|
|
still run fine in Miri -- but might break (including causing UB) on different
|
|
|
|
compiler versions or different platforms.
|
2019-07-30 04:23:56 -05:00
|
|
|
* Program execution is non-deterministic when it depends, for example, on where
|
|
|
|
exactly in memory allocations end up. Miri tests one of many possible
|
|
|
|
executions of your program. If your code is sensitive to allocation base
|
|
|
|
addresses or other non-deterministic data, try running Miri with different
|
|
|
|
values for `-Zmiri-seed` to test different executions.
|
2019-03-26 04:07:22 -05:00
|
|
|
* Miri runs the program as a platform-independent interpreter, so the program
|
|
|
|
has no access to any platform-specific APIs or FFI. A few APIs have been
|
|
|
|
implemented (such as printing to stdout) but most have not: for example, Miri
|
2019-07-14 02:34:08 -05:00
|
|
|
currently does not support concurrency, or SIMD, or networking, or file system
|
|
|
|
access.
|
2019-03-26 04:07:22 -05:00
|
|
|
|
2018-11-13 06:28:09 -06:00
|
|
|
[rust]: https://www.rust-lang.org/
|
|
|
|
[mir]: https://github.com/rust-lang/rfcs/blob/master/text/1211-mir.md
|
|
|
|
[`unreachable_unchecked`]: https://doc.rust-lang.org/stable/std/hint/fn.unreachable_unchecked.html
|
|
|
|
[`copy_nonoverlapping`]: https://doc.rust-lang.org/stable/std/ptr/fn.copy_nonoverlapping.html
|
2016-02-02 04:47:28 -06:00
|
|
|
|
2018-11-16 03:01:54 -06:00
|
|
|
|
2019-09-14 06:40:37 -05:00
|
|
|
## Using Miri
|
2018-08-14 02:35:00 -05:00
|
|
|
|
2019-03-25 16:47:31 -05:00
|
|
|
Install Miri via `rustup`:
|
2018-11-25 15:17:48 -06:00
|
|
|
|
|
|
|
```sh
|
2019-03-25 16:47:31 -05:00
|
|
|
rustup component add miri
|
2018-11-25 15:17:48 -06:00
|
|
|
```
|
|
|
|
|
2019-06-13 01:43:35 -05:00
|
|
|
If `rustup` says the `miri` component is unavailable, that's because not all
|
|
|
|
nightly releases come with all tools. Check out
|
|
|
|
[this website](https://rust-lang.github.io/rustup-components-history) to
|
|
|
|
determine a nightly version that comes with Miri and install that, e.g. using
|
|
|
|
`rustup install nightly-2019-03-28`.
|
2019-03-31 09:03:03 -05:00
|
|
|
|
2018-12-19 04:16:19 -06:00
|
|
|
Now you can run your project in Miri:
|
2018-11-25 11:27:24 -06:00
|
|
|
|
|
|
|
1. Run `cargo clean` to eliminate any cached dependencies. Miri needs your
|
|
|
|
dependencies to be compiled the right way, that would not happen if they have
|
|
|
|
previously already been compiled.
|
2019-03-25 16:47:31 -05:00
|
|
|
2. To run all tests in your project through Miri, use `cargo miri test`.
|
|
|
|
3. If you have a binary project, you can run it through Miri using `cargo miri run`.
|
2018-10-15 13:45:55 -05:00
|
|
|
|
2019-03-08 03:50:31 -06:00
|
|
|
The first time you run Miri, it will perform some extra setup and install some
|
2019-07-30 10:04:57 -05:00
|
|
|
dependencies. It will ask you for confirmation before installing anything.
|
2019-03-08 03:50:31 -06:00
|
|
|
|
2019-02-09 05:44:56 -06:00
|
|
|
You can pass arguments to Miri after the first `--`, and pass arguments to the
|
|
|
|
interpreted program or test suite after the second `--`. For example, `cargo
|
2019-03-25 16:47:31 -05:00
|
|
|
miri run -- -Zmiri-disable-validation` runs the program without validation of
|
|
|
|
basic type invariants and references. `cargo miri test -- -- -Zunstable-options
|
|
|
|
--exclude-should-panic` skips `#[should_panic]` tests, which is a good idea
|
|
|
|
because Miri does not support unwinding or catching panics.
|
2019-02-09 05:44:56 -06:00
|
|
|
|
2019-02-07 06:07:47 -06:00
|
|
|
When running code via `cargo miri`, the `miri` config flag is set. You can
|
2018-12-19 04:16:19 -06:00
|
|
|
use this to exclude test cases that will fail under Miri because they do things
|
|
|
|
Miri does not support:
|
|
|
|
|
|
|
|
```rust
|
2019-02-07 06:07:47 -06:00
|
|
|
#[cfg(not(miri))]
|
2018-12-19 04:16:19 -06:00
|
|
|
#[test]
|
|
|
|
fn does_not_work_on_miri() {
|
|
|
|
let x = 0u8;
|
|
|
|
assert!(&x as *const _ as usize % 4 < 4);
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
2019-06-25 03:05:34 -05:00
|
|
|
### Running Miri on CI
|
|
|
|
|
|
|
|
To run Miri on CI, make sure that you handle the case where the latest nightly
|
|
|
|
does not ship the Miri component because it currently does not build. For
|
|
|
|
example, you can use the following snippet to always test with the latest
|
|
|
|
nightly that *does* come with Miri:
|
|
|
|
|
|
|
|
```sh
|
|
|
|
MIRI_NIGHTLY=nightly-$(curl -s https://rust-lang.github.io/rustup-components-history/x86_64-unknown-linux-gnu/miri)
|
|
|
|
echo "Installing latest nightly with Miri: $MIRI_NIGHTLY"
|
|
|
|
rustup default "$MIRI_NIGHTLY"
|
|
|
|
|
|
|
|
rustup component add miri
|
|
|
|
cargo miri setup
|
|
|
|
|
|
|
|
cargo miri test -- -- -Zunstable-options --exclude-should-panic
|
|
|
|
```
|
|
|
|
|
2019-07-30 10:04:57 -05:00
|
|
|
We use `cargo miri setup` to avoid getting interactive questions about the extra
|
|
|
|
setup needed for Miri.
|
|
|
|
|
2018-10-20 11:31:15 -05:00
|
|
|
### Common Problems
|
|
|
|
|
2019-03-31 09:03:03 -05:00
|
|
|
When using the above instructions, you may encounter a number of confusing compiler
|
2018-10-20 11:31:15 -05:00
|
|
|
errors.
|
|
|
|
|
|
|
|
#### "found possibly newer version of crate `std` which `<dependency>` depends on"
|
|
|
|
|
2018-11-25 11:27:24 -06:00
|
|
|
Your build directory may contain artifacts from an earlier build that have/have
|
|
|
|
not been built for Miri. Run `cargo clean` before switching from non-Miri to
|
|
|
|
Miri builds and vice-versa.
|
2018-10-20 11:31:15 -05:00
|
|
|
|
|
|
|
#### "found crate `std` compiled by an incompatible version of rustc"
|
|
|
|
|
2018-10-23 10:21:19 -05:00
|
|
|
You may be running `cargo miri` with a different compiler version than the one
|
2018-11-25 11:27:24 -06:00
|
|
|
used to build the custom libstd that Miri uses, and Miri failed to detect that.
|
2018-11-27 04:46:09 -06:00
|
|
|
Try deleting `~/.cache/miri`.
|
2018-08-14 02:35:00 -05:00
|
|
|
|
2019-04-20 11:36:43 -05:00
|
|
|
#### "no mir for `std::rt::lang_start_internal`"
|
|
|
|
|
|
|
|
This means the sysroot you are using was not compiled with Miri in mind. This
|
|
|
|
should never happen when you use `cargo miri` because that takes care of setting
|
|
|
|
up the sysroot. If you are using `miri` (the Miri driver) directly, see
|
|
|
|
[below][testing-miri] for how to set up the sysroot.
|
|
|
|
|
2019-07-24 07:18:42 -05:00
|
|
|
|
|
|
|
## Miri `-Z` flags and environment variables
|
|
|
|
[miri-flags]: #miri--z-flags-and-environment-variables
|
|
|
|
|
|
|
|
Several `-Z` flags are relevant for Miri:
|
|
|
|
|
|
|
|
* `-Zmiri-seed=<hex>` is a custom `-Z` flag added by Miri. It configures the
|
|
|
|
seed of the RNG that Miri uses to resolve non-determinism. This RNG is used
|
|
|
|
to pick base addresses for allocations, and when the interpreted program
|
|
|
|
requests system entropy. The default seed is 0.
|
|
|
|
**NOTE**: This entropy is not good enough for cryptographic use! Do not
|
|
|
|
generate secret keys in Miri or perform other kinds of cryptographic
|
|
|
|
operations that rely on proper random numbers.
|
|
|
|
* `-Zmiri-disable-validation` disables enforcing the validity invariant, which
|
|
|
|
is enforced by default. This is mostly useful for debugging; it means Miri
|
|
|
|
will miss bugs in your program. However, this can also help to make Miri run
|
|
|
|
faster.
|
2019-08-23 16:29:11 -05:00
|
|
|
* `-Zmiri-disable-isolation` disables host host isolation. As a consequence,
|
|
|
|
the program has access to host resources such as environment variables and
|
|
|
|
randomness (and, eventually, file systems and more).
|
2019-08-28 17:20:50 -05:00
|
|
|
* `-Zmiri-env-exclude=<var>` keeps the `var` environment variable isolated from
|
2019-08-29 04:09:34 -05:00
|
|
|
the host. Can be used multiple times to exclude several variables. The `TERM`
|
|
|
|
environment variable is excluded by default.
|
2019-07-24 07:18:42 -05:00
|
|
|
* `-Zmir-opt-level` controls how many MIR optimizations are performed. Miri
|
|
|
|
overrides the default to be `0`; be advised that using any higher level can
|
|
|
|
make Miri miss bugs in your program because they got optimized away.
|
|
|
|
* `-Zalways-encode-mir` makes rustc dump MIR even for completely monomorphic
|
|
|
|
functions. This is needed so that Miri can execute such functions, so Miri
|
|
|
|
sets this flag per default.
|
|
|
|
* `-Zmir-emit-retag` controls whether `Retag` statements are emitted. Miri
|
|
|
|
enables this per default because it is needed for validation.
|
|
|
|
|
|
|
|
Moreover, Miri recognizes some environment variables:
|
|
|
|
|
|
|
|
* `MIRI_LOG`, `MIRI_BACKTRACE` control logging and backtrace printing during
|
|
|
|
Miri executions, also [see above][testing-miri].
|
|
|
|
* `MIRI_SYSROOT` (recognized by `cargo miri` and the test suite)
|
|
|
|
indicates the sysroot to use. To do the same thing with `miri`
|
|
|
|
directly, use the `--sysroot` flag.
|
|
|
|
* `MIRI_TEST_TARGET` (recognized by the test suite) indicates which target
|
|
|
|
architecture to test against. `miri` and `cargo miri` accept the `--target`
|
|
|
|
flag for the same purpose.
|
|
|
|
|
2016-06-19 01:04:11 -05:00
|
|
|
## Contributing and getting help
|
2016-04-14 03:21:32 -05:00
|
|
|
|
2019-09-14 06:40:37 -05:00
|
|
|
If you want to contribute to Miri, great! Please check out our
|
|
|
|
[contribution guide](CONTRIBUTING.md).
|
|
|
|
|
|
|
|
For help with running Miri, you can open an issue here on
|
2018-11-30 15:23:20 -06:00
|
|
|
GitHub or contact us (`oli-obk` and `RalfJ`) on the [Rust Zulip].
|
|
|
|
|
|
|
|
[Rust Zulip]: https://rust-lang.zulipchat.com
|
2016-02-02 04:47:28 -06:00
|
|
|
|
2018-11-13 07:16:08 -06:00
|
|
|
## History
|
|
|
|
|
|
|
|
This project began as part of an undergraduate research course in 2015 by
|
2018-11-14 01:23:43 -06:00
|
|
|
@solson at the [University of Saskatchewan][usask]. There are [slides] and a
|
|
|
|
[report] available from that project. In 2016, @oli-obk joined to prepare miri
|
|
|
|
for eventually being used as const evaluator in the Rust compiler itself
|
|
|
|
(basically, for `const` and `static` stuff), replacing the old evaluator that
|
|
|
|
worked directly on the AST. In 2017, @RalfJung did an internship with Mozilla
|
|
|
|
and began developing miri towards a tool for detecting undefined behavior, and
|
|
|
|
also using miri as a way to explore the consequences of various possible
|
|
|
|
definitions for undefined behavior in Rust. @oli-obk's move of the miri engine
|
|
|
|
into the compiler finally came to completion in early 2018. Meanwhile, later
|
|
|
|
that year, @RalfJung did a second internship, developing miri further with
|
|
|
|
support for checking basic type invariants and verifying that references are
|
|
|
|
used according to their aliasing restrictions.
|
2018-11-13 07:16:08 -06:00
|
|
|
|
|
|
|
[usask]: https://www.usask.ca/
|
2018-11-14 01:23:43 -06:00
|
|
|
[slides]: https://solson.me/miri-slides.pdf
|
|
|
|
[report]: https://solson.me/miri-report.pdf
|
2018-11-13 07:16:08 -06:00
|
|
|
|
2019-02-15 03:41:25 -06:00
|
|
|
## Bugs found by Miri
|
|
|
|
|
2019-04-26 02:08:30 -05:00
|
|
|
Miri has already found a number of bugs in the Rust standard library and beyond, which we collect here.
|
|
|
|
|
|
|
|
Definite bugs found:
|
2019-02-15 03:41:25 -06:00
|
|
|
|
2019-02-15 03:43:27 -06:00
|
|
|
* [`Debug for vec_deque::Iter` accessing uninitialized memory](https://github.com/rust-lang/rust/issues/53566)
|
2019-07-02 02:51:16 -05:00
|
|
|
* [`Vec::into_iter` doing an unaligned ZST read](https://github.com/rust-lang/rust/pull/53804)
|
2019-02-15 03:41:25 -06:00
|
|
|
* [`From<&[T]> for Rc` creating a not sufficiently aligned reference](https://github.com/rust-lang/rust/issues/54908)
|
|
|
|
* [`BTreeMap` creating a shared reference pointing to a too small allocation](https://github.com/rust-lang/rust/issues/54957)
|
2019-05-23 09:33:44 -05:00
|
|
|
* [`Vec::append` creating a dangling reference](https://github.com/rust-lang/rust/pull/61082)
|
2019-02-15 03:41:25 -06:00
|
|
|
* [Futures turning a shared reference into a mutable one](https://github.com/rust-lang/rust/pull/56319)
|
|
|
|
* [`str` turning a shared reference into a mutable one](https://github.com/rust-lang/rust/pull/58200)
|
2019-04-26 02:08:30 -05:00
|
|
|
* [`rand` performing unaligned reads](https://github.com/rust-random/rand/issues/779)
|
2019-06-30 14:23:48 -05:00
|
|
|
* [The Unix allocator calling `posix_memalign` in an invalid way](https://github.com/rust-lang/rust/issues/62251)
|
2019-08-04 08:09:13 -05:00
|
|
|
* [`getrandom` calling the `getrandom` syscall in an invalid way](https://github.com/rust-random/getrandom/pull/73)
|
2019-04-26 02:08:30 -05:00
|
|
|
|
|
|
|
Violations of Stacked Borrows found that are likely bugs (but Stacked Borrows is currently just an experiment):
|
|
|
|
|
|
|
|
* [`VecDeque` creating overlapping mutable references](https://github.com/rust-lang/rust/pull/56161)
|
2019-02-15 03:41:25 -06:00
|
|
|
* [`BTreeMap` creating mutable references that overlap with shared references](https://github.com/rust-lang/rust/pull/58431)
|
2019-04-18 04:54:21 -05:00
|
|
|
* [`LinkedList` creating overlapping mutable references](https://github.com/rust-lang/rust/pull/60072)
|
2019-05-23 09:33:44 -05:00
|
|
|
* [`Vec::push` invalidating existing references into the vector](https://github.com/rust-lang/rust/issues/60847)
|
2019-02-15 03:41:25 -06:00
|
|
|
|
2016-02-02 04:47:28 -06:00
|
|
|
## License
|
|
|
|
|
|
|
|
Licensed under either of
|
|
|
|
* Apache License, Version 2.0 ([LICENSE-APACHE](LICENSE-APACHE) or
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0)
|
|
|
|
* MIT license ([LICENSE-MIT](LICENSE-MIT) or
|
|
|
|
http://opensource.org/licenses/MIT) at your option.
|
|
|
|
|
|
|
|
### Contribution
|
|
|
|
|
|
|
|
Unless you explicitly state otherwise, any contribution intentionally submitted
|
|
|
|
for inclusion in the work by you shall be dual licensed as above, without any
|
|
|
|
additional terms or conditions.
|