Rollup merge of #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. #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 #91370. r? ``@cjgillot``
This commit is contained in:
commit
464a0813d1
@ -520,9 +520,16 @@ fn visit_foreign_item(&mut self, foreign_item: &'ast ForeignItem) {
|
||||
}
|
||||
fn visit_fn(&mut self, fn_kind: FnKind<'ast>, sp: Span, _: NodeId) {
|
||||
let rib_kind = match fn_kind {
|
||||
// Bail if there's no body.
|
||||
FnKind::Fn(.., None) => return visit::walk_fn(self, fn_kind, sp),
|
||||
FnKind::Fn(FnCtxt::Free | FnCtxt::Foreign, ..) => FnItemRibKind,
|
||||
// Bail if the function is foreign, and thus cannot validly have
|
||||
// a body, or if there's no body for some other reason.
|
||||
FnKind::Fn(FnCtxt::Foreign, _, sig, ..) | FnKind::Fn(_, _, sig, .., None) => {
|
||||
// We don't need to deal with patterns in parameters, because
|
||||
// they are not possible for foreign or bodiless functions.
|
||||
self.visit_fn_header(&sig.header);
|
||||
visit::walk_fn_decl(self, &sig.decl);
|
||||
return;
|
||||
}
|
||||
FnKind::Fn(FnCtxt::Free, ..) => FnItemRibKind,
|
||||
FnKind::Fn(FnCtxt::Assoc(_), ..) => NormalRibKind,
|
||||
FnKind::Closure(..) => ClosureOrAsyncRibKind,
|
||||
};
|
||||
|
12
src/test/ui/foreign/issue-91370-foreign-fn-block-impl.rs
Normal file
12
src/test/ui/foreign/issue-91370-foreign-fn-block-impl.rs
Normal file
@ -0,0 +1,12 @@
|
||||
// Regression test for issue #91370.
|
||||
|
||||
extern {
|
||||
//~^ `extern` blocks define existing foreign functions
|
||||
fn f() {
|
||||
//~^ incorrect function inside `extern` block
|
||||
//~| cannot have a body
|
||||
impl Copy for u8 {}
|
||||
}
|
||||
}
|
||||
|
||||
fn main() {}
|
21
src/test/ui/foreign/issue-91370-foreign-fn-block-impl.stderr
Normal file
21
src/test/ui/foreign/issue-91370-foreign-fn-block-impl.stderr
Normal file
@ -0,0 +1,21 @@
|
||||
error: incorrect function inside `extern` block
|
||||
--> $DIR/issue-91370-foreign-fn-block-impl.rs:5:8
|
||||
|
|
||||
LL | extern {
|
||||
| ------ `extern` blocks define existing foreign functions and functions inside of them cannot have a body
|
||||
LL |
|
||||
LL | fn f() {
|
||||
| ________^___-
|
||||
| | |
|
||||
| | cannot have a body
|
||||
LL | |
|
||||
LL | |
|
||||
LL | | impl Copy for u8 {}
|
||||
LL | | }
|
||||
| |_____- help: remove the invalid body: `;`
|
||||
|
|
||||
= 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
|
||||
= note: for more information, visit https://doc.rust-lang.org/std/keyword.extern.html
|
||||
|
||||
error: aborting due to previous error
|
||||
|
Loading…
Reference in New Issue
Block a user