Skip to content

Commit b996061

Browse files
committed
Tune the inlinability of unwrap
1 parent 62d7ed4 commit b996061

File tree

4 files changed

+39
-5
lines changed

4 files changed

+39
-5
lines changed

library/core/src/option.rs

+10-2
Original file line numberDiff line numberDiff line change
@@ -921,14 +921,14 @@ impl<T> Option<T> {
921921
/// let x: Option<&str> = None;
922922
/// assert_eq!(x.unwrap(), "air"); // fails
923923
/// ```
924-
#[inline]
924+
#[inline(always)]
925925
#[track_caller]
926926
#[stable(feature = "rust1", since = "1.0.0")]
927927
#[rustc_const_unstable(feature = "const_option", issue = "67441")]
928928
pub const fn unwrap(self) -> T {
929929
match self {
930930
Some(val) => val,
931-
None => panic("called `Option::unwrap()` on a `None` value"),
931+
None => unwrap_failed(),
932932
}
933933
}
934934

@@ -1970,6 +1970,14 @@ impl<T, E> Option<Result<T, E>> {
19701970
}
19711971
}
19721972

1973+
#[cfg_attr(not(feature = "panic_immediate_abort"), inline(never))]
1974+
#[cfg_attr(feature = "panic_immediate_abort", inline)]
1975+
#[cold]
1976+
#[track_caller]
1977+
const fn unwrap_failed() -> ! {
1978+
panic("called `Option::unwrap()` on a `None` value")
1979+
}
1980+
19731981
// This is a separate function to reduce the code size of .expect() itself.
19741982
#[cfg_attr(not(feature = "panic_immediate_abort"), inline(never))]
19751983
#[cfg_attr(feature = "panic_immediate_abort", inline)]

library/core/src/result.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1061,7 +1061,7 @@ impl<T, E> Result<T, E> {
10611061
/// let x: Result<u32, &str> = Err("emergency failure");
10621062
/// x.unwrap(); // panics with `emergency failure`
10631063
/// ```
1064-
#[inline]
1064+
#[inline(always)]
10651065
#[track_caller]
10661066
#[stable(feature = "rust1", since = "1.0.0")]
10671067
pub fn unwrap(self) -> T

tests/codegen/debuginfo-inline-callsite-location.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,9 @@
44
// can correctly merge the debug info if it merges the inlined code (e.g., for merging of tail
55
// calls to panic.
66

7-
// CHECK: tail call void @_ZN4core9panicking5panic17h{{([0-9a-z]{16})}}E
7+
// CHECK: tail call void @_ZN4core6option13unwrap_failed17h{{([0-9a-z]{16})}}E
88
// CHECK-SAME: !dbg ![[#first_dbg:]]
9-
// CHECK: tail call void @_ZN4core9panicking5panic17h{{([0-9a-z]{16})}}E
9+
// CHECK: tail call void @_ZN4core6option13unwrap_failed17h{{([0-9a-z]{16})}}E
1010
// CHECK-SAME: !dbg ![[#second_dbg:]]
1111

1212
// CHECK-DAG: ![[#func_dbg:]] = distinct !DISubprogram(name: "unwrap<i32>"
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
// compile-flags: -C opt-level=z --edition=2021
2+
// ignore-debug
3+
4+
#![crate_type = "lib"]
5+
6+
// From <https://github.com/rust-lang/rust/issues/115463>
7+
8+
// CHECK-LABEL: @read_up_to_8(
9+
#[no_mangle]
10+
pub fn read_up_to_8(buf: &[u8]) -> u64 {
11+
// CHECK-NOT: unwrap_failed
12+
if buf.len() < 4 {
13+
// actual instance has more code.
14+
return 0;
15+
}
16+
let lo = u32::from_le_bytes(buf[..4].try_into().unwrap()) as u64;
17+
let hi = u32::from_le_bytes(buf[buf.len() - 4..][..4].try_into().unwrap()) as u64;
18+
lo | (hi << 8 * (buf.len() as u64 - 4))
19+
}
20+
21+
// CHECK-LABEL: @checking_unwrap_expectation(
22+
#[no_mangle]
23+
pub fn checking_unwrap_expectation(buf: &[u8]) -> &[u8; 4] {
24+
// CHECK: call void @_ZN4core6result13unwrap_failed17h
25+
buf.try_into().unwrap()
26+
}

0 commit comments

Comments
 (0)