Skip to content

Commit 157e0a0

Browse files
committed
Validate that naked functions are never inlined
Reject all uses of the inline attribute on naked functions. rust-lang/rfcs#2774 rust-lang/rfcs#2972
1 parent 6be8a06 commit 157e0a0

File tree

4 files changed

+112
-2
lines changed

4 files changed

+112
-2
lines changed

compiler/rustc_lint_defs/src/builtin.rs

+3
Original file line numberDiff line numberDiff line change
@@ -2720,6 +2720,9 @@ declare_lint! {
27202720
/// The asm block must not contain any operands other than `const` and
27212721
/// `sym`. Additionally, naked function should specify a non-Rust ABI.
27222722
///
2723+
/// Naked functions cannot be inlined. All forms of the `inline` attribute
2724+
/// are prohibited.
2725+
///
27232726
/// While other definitions of naked functions were previously accepted,
27242727
/// they are unsupported and might not work reliably. This is a
27252728
/// [future-incompatible] lint that will transition into hard error in

compiler/rustc_passes/src/naked_functions.rs

+11-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
//! Checks validity of naked functions.
22
3-
use rustc_ast::InlineAsmOptions;
3+
use rustc_ast::{Attribute, InlineAsmOptions};
44
use rustc_hir as hir;
55
use rustc_hir::def_id::LocalDefId;
66
use rustc_hir::intravisit::{ErasedMap, FnKind, NestedVisitorMap, Visitor};
@@ -70,10 +70,20 @@ impl<'tcx> Visitor<'tcx> for CheckNakedFunctions<'tcx> {
7070
check_no_patterns(self.tcx, body.params);
7171
check_no_parameters_use(self.tcx, body);
7272
check_asm(self.tcx, hir_id, body, span);
73+
check_inline(self.tcx, hir_id, attrs);
7374
}
7475
}
7576
}
7677

78+
/// Check that the function isn't inlined.
79+
fn check_inline(tcx: TyCtxt<'_>, hir_id: HirId, attrs: &[Attribute]) {
80+
for attr in attrs.iter().filter(|attr| attr.has_name(sym::inline)) {
81+
tcx.struct_span_lint_hir(UNSUPPORTED_NAKED_FUNCTIONS, hir_id, attr.span, |lint| {
82+
lint.build("naked functions cannot be inlined").emit();
83+
});
84+
}
85+
}
86+
7787
/// Checks that function uses non-Rust ABI.
7888
fn check_abi(tcx: TyCtxt<'_>, hir_id: HirId, abi: Abi, fn_ident_span: Span) {
7989
if abi == Abi::Rust {

src/test/ui/asm/naked-functions.rs

+43
Original file line numberDiff line numberDiff line change
@@ -167,3 +167,46 @@ pub unsafe extern "C" fn valid_c() {
167167
pub unsafe extern "C" fn valid_att_syntax() {
168168
asm!("", options(noreturn, att_syntax));
169169
}
170+
171+
#[naked]
172+
pub unsafe extern "C" fn inline_none() {
173+
asm!("", options(noreturn));
174+
}
175+
176+
#[naked]
177+
#[inline]
178+
//~^ WARN naked functions cannot be inlined
179+
//~| WARN this was previously accepted
180+
pub unsafe extern "C" fn inline_hint() {
181+
asm!("", options(noreturn));
182+
}
183+
184+
#[naked]
185+
#[inline(always)]
186+
//~^ WARN naked functions cannot be inlined
187+
//~| WARN this was previously accepted
188+
pub unsafe extern "C" fn inline_always() {
189+
asm!("", options(noreturn));
190+
}
191+
192+
#[naked]
193+
#[inline(never)]
194+
//~^ WARN naked functions cannot be inlined
195+
//~| WARN this was previously accepted
196+
pub unsafe extern "C" fn inline_never() {
197+
asm!("", options(noreturn));
198+
}
199+
200+
#[naked]
201+
#[inline]
202+
//~^ WARN naked functions cannot be inlined
203+
//~| WARN this was previously accepted
204+
#[inline(always)]
205+
//~^ WARN naked functions cannot be inlined
206+
//~| WARN this was previously accepted
207+
#[inline(never)]
208+
//~^ WARN naked functions cannot be inlined
209+
//~| WARN this was previously accepted
210+
pub unsafe extern "C" fn inline_all() {
211+
asm!("", options(noreturn));
212+
}

src/test/ui/asm/naked-functions.stderr

+55-1
Original file line numberDiff line numberDiff line change
@@ -296,5 +296,59 @@ LL | pub unsafe extern "Rust" fn rust_abi() {
296296
= warning: this was previously accepted by the compiler but is being phased out; it will become a hard error in a future release!
297297
= note: for more information, see issue #32408 <https://github.com/rust-lang/rust/issues/32408>
298298

299-
error: aborting due to 8 previous errors; 19 warnings emitted
299+
warning: naked functions cannot be inlined
300+
--> $DIR/naked-functions.rs:177:1
301+
|
302+
LL | #[inline]
303+
| ^^^^^^^^^
304+
|
305+
= warning: this was previously accepted by the compiler but is being phased out; it will become a hard error in a future release!
306+
= note: for more information, see issue #32408 <https://github.com/rust-lang/rust/issues/32408>
307+
308+
warning: naked functions cannot be inlined
309+
--> $DIR/naked-functions.rs:185:1
310+
|
311+
LL | #[inline(always)]
312+
| ^^^^^^^^^^^^^^^^^
313+
|
314+
= warning: this was previously accepted by the compiler but is being phased out; it will become a hard error in a future release!
315+
= note: for more information, see issue #32408 <https://github.com/rust-lang/rust/issues/32408>
316+
317+
warning: naked functions cannot be inlined
318+
--> $DIR/naked-functions.rs:193:1
319+
|
320+
LL | #[inline(never)]
321+
| ^^^^^^^^^^^^^^^^
322+
|
323+
= warning: this was previously accepted by the compiler but is being phased out; it will become a hard error in a future release!
324+
= note: for more information, see issue #32408 <https://github.com/rust-lang/rust/issues/32408>
325+
326+
warning: naked functions cannot be inlined
327+
--> $DIR/naked-functions.rs:201:1
328+
|
329+
LL | #[inline]
330+
| ^^^^^^^^^
331+
|
332+
= warning: this was previously accepted by the compiler but is being phased out; it will become a hard error in a future release!
333+
= note: for more information, see issue #32408 <https://github.com/rust-lang/rust/issues/32408>
334+
335+
warning: naked functions cannot be inlined
336+
--> $DIR/naked-functions.rs:204:1
337+
|
338+
LL | #[inline(always)]
339+
| ^^^^^^^^^^^^^^^^^
340+
|
341+
= warning: this was previously accepted by the compiler but is being phased out; it will become a hard error in a future release!
342+
= note: for more information, see issue #32408 <https://github.com/rust-lang/rust/issues/32408>
343+
344+
warning: naked functions cannot be inlined
345+
--> $DIR/naked-functions.rs:207:1
346+
|
347+
LL | #[inline(never)]
348+
| ^^^^^^^^^^^^^^^^
349+
|
350+
= warning: this was previously accepted by the compiler but is being phased out; it will become a hard error in a future release!
351+
= note: for more information, see issue #32408 <https://github.com/rust-lang/rust/issues/32408>
352+
353+
error: aborting due to 8 previous errors; 25 warnings emitted
300354

0 commit comments

Comments
 (0)