Only implement Fn* traits for extern "Rust" safe function pointers.
This commit is contained in:
parent
84c47b8279
commit
fb9e171ab7
4 changed files with 98 additions and 2 deletions
|
@ -2,6 +2,7 @@ use rustc_data_structures::fx::FxHashMap;
|
|||
use rustc_hir::{def_id::DefId, Movability, Mutability};
|
||||
use rustc_infer::traits::query::NoSolution;
|
||||
use rustc_middle::ty::{self, Ty, TyCtxt, TypeFoldable, TypeFolder, TypeSuperFoldable};
|
||||
use rustc_target::spec::abi::Abi;
|
||||
|
||||
use crate::solve::EvalCtxt;
|
||||
|
||||
|
@ -194,7 +195,20 @@ pub(crate) fn extract_tupled_inputs_and_output_from_callable<'tcx>(
|
|||
.subst(tcx, substs)
|
||||
.map_bound(|sig| (tcx.mk_tup(sig.inputs()), sig.output())),
|
||||
)),
|
||||
ty::FnPtr(sig) => Ok(Some(sig.map_bound(|sig| (tcx.mk_tup(sig.inputs()), sig.output())))),
|
||||
// keep this in sync with assemble_fn_pointer_candidates until the old solver is removed.
|
||||
ty::FnPtr(sig) => {
|
||||
if let ty::FnSig {
|
||||
unsafety: rustc_hir::Unsafety::Normal,
|
||||
abi: Abi::Rust,
|
||||
c_variadic: false,
|
||||
..
|
||||
} = sig.skip_binder()
|
||||
{
|
||||
Ok(Some(sig.map_bound(|sig| (tcx.mk_tup(sig.inputs()), sig.output()))))
|
||||
} else {
|
||||
Err(NoSolution)
|
||||
}
|
||||
}
|
||||
ty::Closure(_, substs) => {
|
||||
let closure_substs = substs.as_closure();
|
||||
match closure_substs.kind_ty().to_opt_closure_kind() {
|
||||
|
|
|
@ -291,6 +291,9 @@ impl<'cx, 'tcx> SelectionContext<'cx, 'tcx> {
|
|||
return;
|
||||
}
|
||||
|
||||
// Keep this funtion in sync with extract_tupled_inputs_and_output_from_callable
|
||||
// until the old solver (and thus this function) is removed.
|
||||
|
||||
// Okay to skip binder because what we are inspecting doesn't involve bound regions.
|
||||
let self_ty = obligation.self_ty().skip_binder();
|
||||
match *self_ty.kind() {
|
||||
|
|
|
@ -1,5 +1,4 @@
|
|||
// compile-flags: -Ztrait-solver=next
|
||||
// check-pass
|
||||
|
||||
fn require_fn(_: impl Fn() -> i32) {}
|
||||
|
||||
|
@ -7,7 +6,23 @@ fn f() -> i32 {
|
|||
1i32
|
||||
}
|
||||
|
||||
extern "C" fn g() -> i32 {
|
||||
2i32
|
||||
}
|
||||
|
||||
unsafe fn h() -> i32 {
|
||||
2i32
|
||||
}
|
||||
|
||||
fn main() {
|
||||
require_fn(f);
|
||||
require_fn(f as fn() -> i32);
|
||||
require_fn(f as unsafe fn() -> i32);
|
||||
//~^ ERROR: expected a `Fn<()>` closure, found `unsafe fn() -> i32`
|
||||
//~| ERROR: type mismatch resolving `<unsafe fn() -> i32 as FnOnce<()>>::Output == i32`
|
||||
require_fn(g);
|
||||
require_fn(g as extern "C" fn() -> i32);
|
||||
//~^ ERROR: expected a `Fn<()>` closure, found `extern "C" fn() -> i32`
|
||||
//~| ERROR: type mismatch resolving `<extern "C" fn() -> i32 as FnOnce<()>>::Output == i32`
|
||||
require_fn(h);
|
||||
}
|
||||
|
|
64
tests/ui/traits/new-solver/fn-trait.stderr
Normal file
64
tests/ui/traits/new-solver/fn-trait.stderr
Normal file
|
@ -0,0 +1,64 @@
|
|||
error[E0277]: expected a `Fn<()>` closure, found `unsafe fn() -> i32`
|
||||
--> $DIR/fn-trait.rs:20:16
|
||||
|
|
||||
LL | require_fn(f as unsafe fn() -> i32);
|
||||
| ---------- ^^^^^^^^^^^^^^^^^^^^^^^ call the function in a closure: `|| unsafe { /* code */ }`
|
||||
| |
|
||||
| required by a bound introduced by this call
|
||||
|
|
||||
= help: the trait `Fn<()>` is not implemented for `unsafe fn() -> i32`
|
||||
= note: wrap the `unsafe fn() -> i32` in a closure with no arguments: `|| { /* code */ }`
|
||||
note: required by a bound in `require_fn`
|
||||
--> $DIR/fn-trait.rs:3:23
|
||||
|
|
||||
LL | fn require_fn(_: impl Fn() -> i32) {}
|
||||
| ^^^^^^^^^^^ required by this bound in `require_fn`
|
||||
|
||||
error[E0271]: type mismatch resolving `<unsafe fn() -> i32 as FnOnce<()>>::Output == i32`
|
||||
--> $DIR/fn-trait.rs:20:16
|
||||
|
|
||||
LL | require_fn(f as unsafe fn() -> i32);
|
||||
| ---------- ^^^^^^^^^^^^^^^^^^^^^^^ types differ
|
||||
| |
|
||||
| required by a bound introduced by this call
|
||||
|
|
||||
note: required by a bound in `require_fn`
|
||||
--> $DIR/fn-trait.rs:3:31
|
||||
|
|
||||
LL | fn require_fn(_: impl Fn() -> i32) {}
|
||||
| ^^^ required by this bound in `require_fn`
|
||||
|
||||
error[E0277]: expected a `Fn<()>` closure, found `extern "C" fn() -> i32`
|
||||
--> $DIR/fn-trait.rs:24:16
|
||||
|
|
||||
LL | require_fn(g as extern "C" fn() -> i32);
|
||||
| ---------- ^^^^^^^^^^^^^^^^^^^^^^^^^^^ expected an `Fn<()>` closure, found `extern "C" fn() -> i32`
|
||||
| |
|
||||
| required by a bound introduced by this call
|
||||
|
|
||||
= help: the trait `Fn<()>` is not implemented for `extern "C" fn() -> i32`
|
||||
= note: wrap the `extern "C" fn() -> i32` in a closure with no arguments: `|| { /* code */ }`
|
||||
note: required by a bound in `require_fn`
|
||||
--> $DIR/fn-trait.rs:3:23
|
||||
|
|
||||
LL | fn require_fn(_: impl Fn() -> i32) {}
|
||||
| ^^^^^^^^^^^ required by this bound in `require_fn`
|
||||
|
||||
error[E0271]: type mismatch resolving `<extern "C" fn() -> i32 as FnOnce<()>>::Output == i32`
|
||||
--> $DIR/fn-trait.rs:24:16
|
||||
|
|
||||
LL | require_fn(g as extern "C" fn() -> i32);
|
||||
| ---------- ^^^^^^^^^^^^^^^^^^^^^^^^^^^ types differ
|
||||
| |
|
||||
| required by a bound introduced by this call
|
||||
|
|
||||
note: required by a bound in `require_fn`
|
||||
--> $DIR/fn-trait.rs:3:31
|
||||
|
|
||||
LL | fn require_fn(_: impl Fn() -> i32) {}
|
||||
| ^^^ required by this bound in `require_fn`
|
||||
|
||||
error: aborting due to 4 previous errors
|
||||
|
||||
Some errors have detailed explanations: E0271, E0277.
|
||||
For more information about an error, try `rustc --explain E0271`.
|
Loading…
Add table
Reference in a new issue