Skip to content

Commit 21fe214

Browse files
authored
Rollup merge of rust-lang#92510 - inquisitivecrystal:foreign-block, r=cjgillot
Don't resolve blocks in foreign functions Although it is an error for a foreign function to have a block, it is still possible at the level of the AST. rust-lang#74204 made AST lowering skip over blocks belonging to foreign functions, since they're invalid. However, resolve still treated these blocks normally, resulting in a mismatch between the HIR and resolve, which could cause an ICE under certain circumstances. This PR changes resolve to skip over blocks belonging to foreign functions, as AST lowering does. Fixes rust-lang#91370. r? `@cjgillot`
2 parents 1786cba + 42de973 commit 21fe214

File tree

3 files changed

+43
-3
lines changed

3 files changed

+43
-3
lines changed

compiler/rustc_resolve/src/late.rs

+10-3
Original file line numberDiff line numberDiff line change
@@ -520,9 +520,16 @@ impl<'a: 'ast, 'ast> Visitor<'ast> for LateResolutionVisitor<'a, '_, 'ast> {
520520
}
521521
fn visit_fn(&mut self, fn_kind: FnKind<'ast>, sp: Span, _: NodeId) {
522522
let rib_kind = match fn_kind {
523-
// Bail if there's no body.
524-
FnKind::Fn(.., None) => return visit::walk_fn(self, fn_kind, sp),
525-
FnKind::Fn(FnCtxt::Free | FnCtxt::Foreign, ..) => FnItemRibKind,
523+
// Bail if the function is foreign, and thus cannot validly have
524+
// a body, or if there's no body for some other reason.
525+
FnKind::Fn(FnCtxt::Foreign, _, sig, ..) | FnKind::Fn(_, _, sig, .., None) => {
526+
// We don't need to deal with patterns in parameters, because
527+
// they are not possible for foreign or bodiless functions.
528+
self.visit_fn_header(&sig.header);
529+
visit::walk_fn_decl(self, &sig.decl);
530+
return;
531+
}
532+
FnKind::Fn(FnCtxt::Free, ..) => FnItemRibKind,
526533
FnKind::Fn(FnCtxt::Assoc(_), ..) => NormalRibKind,
527534
FnKind::Closure(..) => ClosureOrAsyncRibKind,
528535
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
// Regression test for issue #91370.
2+
3+
extern {
4+
//~^ `extern` blocks define existing foreign functions
5+
fn f() {
6+
//~^ incorrect function inside `extern` block
7+
//~| cannot have a body
8+
impl Copy for u8 {}
9+
}
10+
}
11+
12+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
error: incorrect function inside `extern` block
2+
--> $DIR/issue-91370-foreign-fn-block-impl.rs:5:8
3+
|
4+
LL | extern {
5+
| ------ `extern` blocks define existing foreign functions and functions inside of them cannot have a body
6+
LL |
7+
LL | fn f() {
8+
| ________^___-
9+
| | |
10+
| | cannot have a body
11+
LL | |
12+
LL | |
13+
LL | | impl Copy for u8 {}
14+
LL | | }
15+
| |_____- help: remove the invalid body: `;`
16+
|
17+
= help: you might have meant to write a function accessible through FFI, which can be done by writing `extern fn` outside of the `extern` block
18+
= note: for more information, visit https://doc.rust-lang.org/std/keyword.extern.html
19+
20+
error: aborting due to previous error
21+

0 commit comments

Comments
 (0)