blob: dfe6f6de996de0ea38ece321fecd2eb3ed778431 [file] [log] [blame] [view] [edit]
## Unrecoverable Errors with `panic!`
Sometimes bad things happen in your code, and theres nothing you can do about
it. In these cases, Rust has the `panic!` macro. There are two ways to cause a
panic in practice: by taking an action that causes our code to panic (such as
accessing an array past the end) or by explicitly calling the `panic!` macro. In
both cases, we cause a panic in our program. By default, these panics will print
a failure message, unwind, clean up the stack, and quit. Via an environment
variable, you can also have Rust display the call stack when a panic occurs to
make it easier to track down the source of the panic.
> ### Unwinding the Stack or Aborting in Response to a Panic
>
> By default, when a panic occurs the program starts _unwinding_, which means
> Rust walks back up the stack and cleans up the data from each function it
> encounters. However, walking back and cleaning up is a lot of work. Rust,
> therefore, allows you to choose the alternative of immediately _aborting_,
> which ends the program without cleaning up.
>
> Memory that the program was using will then need to be cleaned up by the
> operating system. If in your project you need to make the resultant binary as
> small as possible, you can switch from unwinding to aborting upon a panic by
> adding `panic = 'abort'` to the appropriate `[profile]` sections in your
> _Cargo.toml_ file. For example, if you want to abort on panic in release mode,
> add this:
>
> ```toml
> [profile.release]
> panic = 'abort'
> ```
Lets try calling `panic!` in a simple program:
<Listing file-name="src/main.rs">
```rust,should_panic,panics
{{#rustdoc_include ../listings/ch09-error-handling/no-listing-01-panic/src/main.rs}}
```
</Listing>
When you run the program, youll see something like this:
```console
{{#include ../listings/ch09-error-handling/no-listing-01-panic/output.txt}}
```
The call to `panic!` causes the error message contained in the last two lines.
The first line shows our panic message and the place in our source code where
the panic occurred: _src/main.rs:2:5_ indicates that its the second line, fifth
character of our _src/main.rs_ file.
In this case, the line indicated is part of our code, and if we go to that line,
we see the `panic!` macro call. In other cases, the `panic!` call might be in
code that our code calls, and the filename and line number reported by the error
message will be someone elses code where the `panic!` macro is called, not the
line of our code that eventually led to the `panic!` call.
<!-- Old heading. Do not remove or links may break. -->
<a id="using-a-panic-backtrace"></a>
We can use the backtrace of the functions the `panic!` call came from to figure
out the part of our code that is causing the problem. To understand how to use a
`panic!` backtrace, lets look at another example and see what its like when a
`panic!` call comes from a library because of a bug in our code instead of from
our code calling the macro directly. Listing 9-1 has some code that attempts to
access an index in a vector beyond the range of valid indexes.
<Listing number="9-1" file-name="src/main.rs" caption="Attempting to access an element beyond the end of a vector, which will cause a call to `panic!`">
```rust,should_panic,panics
{{#rustdoc_include ../listings/ch09-error-handling/listing-09-01/src/main.rs}}
```
</Listing>
Here, were attempting to access the 100th element of our vector (which is at
index 99 because indexing starts at zero), but the vector has only three
elements. In this situation, Rust will panic. Using `[]` is supposed to return
an element, but if you pass an invalid index, theres no element that Rust could
return here that would be correct.
In C, attempting to read beyond the end of a data structure is undefined
behavior. You might get whatever is at the location in memory that would
correspond to that element in the data structure, even though the memory doesnt
belong to that structure. This is called a _buffer overread_ and can lead to
security vulnerabilities if an attacker is able to manipulate the index in such
a way as to read data they shouldnt be allowed to that is stored after the data
structure.
To protect your program from this sort of vulnerability, if you try to read an
element at an index that doesnt exist, Rust will stop execution and refuse to
continue. Lets try it and see:
```console
{{#include ../listings/ch09-error-handling/listing-09-01/output.txt}}
```
This error points at line 4 of our _main.rs_ where we attempt to access index
`99` of the vector in `v`.
The `note:` line tells us that we can set the `RUST_BACKTRACE` environment
variable to get a backtrace of exactly what happened to cause the error. A
_backtrace_ is a list of all the functions that have been called to get to this
point. Backtraces in Rust work as they do in other languages: the key to reading
the backtrace is to start from the top and read until you see files you wrote.
Thats the spot where the problem originated. The lines above that spot are code
that your code has called; the lines below are code that called your code. These
before-and-after lines might include core Rust code, standard library code, or
crates that youre using. Lets try getting a backtrace by setting the
`RUST_BACKTRACE` environment variable to any value except `0`. Listing 9-2 shows
output similar to what youll see.
<!-- manual-regeneration
cd listings/ch09-error-handling/listing-09-01
RUST_BACKTRACE=1 cargo run
copy the backtrace output below
check the backtrace number mentioned in the text below the listing
-->
<Listing number="9-2" caption="The backtrace generated by a call to `panic!` displayed when the environment variable `RUST_BACKTRACE` is set">
```console
$ RUST_BACKTRACE=1 cargo run
thread 'main' panicked at src/main.rs:4:6:
index out of bounds: the len is 3 but the index is 99
stack backtrace:
0: rust_begin_unwind
at /rustc/f6e511eec7342f59a25f7c0534f1dbea00d01b14/library/std/src/panicking.rs:662:5
1: core::panicking::panic_fmt
at /rustc/f6e511eec7342f59a25f7c0534f1dbea00d01b14/library/core/src/panicking.rs:74:14
2: core::panicking::panic_bounds_check
at /rustc/f6e511eec7342f59a25f7c0534f1dbea00d01b14/library/core/src/panicking.rs:276:5
3: <usize as core::slice::index::SliceIndex<[T]>>::index
at /rustc/f6e511eec7342f59a25f7c0534f1dbea00d01b14/library/core/src/slice/index.rs:302:10
4: core::slice::index::<impl core::ops::index::Index<I> for [T]>::index
at /rustc/f6e511eec7342f59a25f7c0534f1dbea00d01b14/library/core/src/slice/index.rs:16:9
5: <alloc::vec::Vec<T,A> as core::ops::index::Index<I>>::index
at /rustc/f6e511eec7342f59a25f7c0534f1dbea00d01b14/library/alloc/src/vec/mod.rs:2920:9
6: panic::main
at ./src/main.rs:4:6
7: core::ops::function::FnOnce::call_once
at /rustc/f6e511eec7342f59a25f7c0534f1dbea00d01b14/library/core/src/ops/function.rs:250:5
note: Some details are omitted, run with `RUST_BACKTRACE=full` for a verbose backtrace.
```
</Listing>
Thats a lot of output! The exact output you see might be different depending on
your operating system and Rust version. In order to get backtraces with this
information, debug symbols must be enabled. Debug symbols are enabled by default
when using `cargo build` or `cargo run` without the `--release` flag, as we have
here.
In the output in Listing 9-2, line 6 of the backtrace points to the line in our
project thats causing the problem: line 4 of _src/main.rs_. If we dont want
our program to panic, we should start our investigation at the location pointed
to by the first line mentioning a file we wrote. In Listing 9-1, where we
deliberately wrote code that would panic, the way to fix the panic is to not
request an element beyond the range of the vector indexes. When your code panics
in the future, youll need to figure out what action the code is taking with
what values to cause the panic and what the code should do instead.
Well come back to `panic!` and when we should and should not use `panic!` to
handle error conditions in the
[“To `panic!` or Not to `panic!`”][to-panic-or-not-to-panic]<!-- ignore -->
section later in this chapter. Next, well look at how to recover from an error
using `Result`.
[to-panic-or-not-to-panic]: ch09-03-to-panic-or-not-to-panic.html#to-panic-or-not-to-panic