Skip to content

Commit 58e6447

Browse files
authored
Rollup merge of #71599 - ldm0:fnclo, r=nikomatsakis
Support coercion between (FnDef | Closure) and (FnDef | Closure) Fixes #46742, fixes #48109 Inject `Closure` into the `FnDef x FnDef` coercion special case, which makes coercion of `(FnDef | Closure) x (FnDef | Closure)` possible, where closures should be **non-capturing**.
2 parents 9e2a6a2 + 42396b1 commit 58e6447

14 files changed

+512
-51
lines changed

src/librustc_middle/ty/context.rs

+11-10
Original file line numberDiff line numberDiff line change
@@ -2069,24 +2069,25 @@ impl<'tcx> TyCtxt<'tcx> {
20692069
self.mk_fn_ptr(sig.map_bound(|sig| ty::FnSig { unsafety: hir::Unsafety::Unsafe, ..sig }))
20702070
}
20712071

2072-
/// Given a closure signature `sig`, returns an equivalent `fn`
2073-
/// type with the same signature. Detuples and so forth -- so
2074-
/// e.g., if we have a sig with `Fn<(u32, i32)>` then you would get
2075-
/// a `fn(u32, i32)`.
2076-
/// `unsafety` determines the unsafety of the `fn` type. If you pass
2072+
/// Given a closure signature, returns an equivalent fn signature. Detuples
2073+
/// and so forth -- so e.g., if we have a sig with `Fn<(u32, i32)>` then
2074+
/// you would get a `fn(u32, i32)`.
2075+
/// `unsafety` determines the unsafety of the fn signature. If you pass
20772076
/// `hir::Unsafety::Unsafe` in the previous example, then you would get
20782077
/// an `unsafe fn (u32, i32)`.
20792078
/// It cannot convert a closure that requires unsafe.
2080-
pub fn coerce_closure_fn_ty(self, sig: PolyFnSig<'tcx>, unsafety: hir::Unsafety) -> Ty<'tcx> {
2081-
let converted_sig = sig.map_bound(|s| {
2079+
pub fn signature_unclosure(
2080+
self,
2081+
sig: PolyFnSig<'tcx>,
2082+
unsafety: hir::Unsafety,
2083+
) -> PolyFnSig<'tcx> {
2084+
sig.map_bound(|s| {
20822085
let params_iter = match s.inputs()[0].kind {
20832086
ty::Tuple(params) => params.into_iter().map(|k| k.expect_ty()),
20842087
_ => bug!(),
20852088
};
20862089
self.mk_fn_sig(params_iter, s.output(), s.c_variadic, unsafety, abi::Abi::Rust)
2087-
});
2088-
2089-
self.mk_fn_ptr(converted_sig)
2090+
})
20902091
}
20912092

20922093
#[allow(rustc::usage_of_ty_tykind)]

src/librustc_mir/borrow_check/type_check/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -2088,7 +2088,7 @@ impl<'a, 'tcx> TypeChecker<'a, 'tcx> {
20882088
ty::Closure(_, substs) => substs.as_closure().sig(),
20892089
_ => bug!(),
20902090
};
2091-
let ty_fn_ptr_from = tcx.coerce_closure_fn_ty(sig, *unsafety);
2091+
let ty_fn_ptr_from = tcx.mk_fn_ptr(tcx.signature_unclosure(sig, *unsafety));
20922092

20932093
if let Err(terr) = self.eq_types(
20942094
ty_fn_ptr_from,

src/librustc_typeck/check/coercion.rs

+69-22
Original file line numberDiff line numberDiff line change
@@ -793,7 +793,8 @@ impl<'f, 'tcx> Coerce<'f, 'tcx> {
793793
// `unsafe fn(arg0,arg1,...) -> _`
794794
let closure_sig = substs_a.as_closure().sig();
795795
let unsafety = fn_ty.unsafety();
796-
let pointer_ty = self.tcx.coerce_closure_fn_ty(closure_sig, unsafety);
796+
let pointer_ty =
797+
self.tcx.mk_fn_ptr(self.tcx.signature_unclosure(closure_sig, unsafety));
797798
debug!("coerce_closure_to_fn(a={:?}, b={:?}, pty={:?})", a, b, pointer_ty);
798799
self.unify_and(
799800
pointer_ty,
@@ -909,23 +910,63 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
909910
debug!("coercion::try_find_coercion_lub({:?}, {:?})", prev_ty, new_ty);
910911

911912
// Special-case that coercion alone cannot handle:
912-
// Two function item types of differing IDs or InternalSubsts.
913-
if let (&ty::FnDef(..), &ty::FnDef(..)) = (&prev_ty.kind, &new_ty.kind) {
914-
// Don't reify if the function types have a LUB, i.e., they
915-
// are the same function and their parameters have a LUB.
916-
let lub_ty = self
917-
.commit_if_ok(|_| self.at(cause, self.param_env).lub(prev_ty, new_ty))
918-
.map(|ok| self.register_infer_ok_obligations(ok));
919-
920-
if lub_ty.is_ok() {
921-
// We have a LUB of prev_ty and new_ty, just return it.
922-
return lub_ty;
913+
// Function items or non-capturing closures of differing IDs or InternalSubsts.
914+
let (a_sig, b_sig) = {
915+
let is_capturing_closure = |ty| {
916+
if let &ty::Closure(_, substs) = ty {
917+
substs.as_closure().upvar_tys().next().is_some()
918+
} else {
919+
false
920+
}
921+
};
922+
if is_capturing_closure(&prev_ty.kind) || is_capturing_closure(&new_ty.kind) {
923+
(None, None)
924+
} else {
925+
match (&prev_ty.kind, &new_ty.kind) {
926+
(&ty::FnDef(..), &ty::FnDef(..)) => {
927+
// Don't reify if the function types have a LUB, i.e., they
928+
// are the same function and their parameters have a LUB.
929+
match self
930+
.commit_if_ok(|_| self.at(cause, self.param_env).lub(prev_ty, new_ty))
931+
{
932+
// We have a LUB of prev_ty and new_ty, just return it.
933+
Ok(ok) => return Ok(self.register_infer_ok_obligations(ok)),
934+
Err(_) => {
935+
(Some(prev_ty.fn_sig(self.tcx)), Some(new_ty.fn_sig(self.tcx)))
936+
}
937+
}
938+
}
939+
(&ty::Closure(_, substs), &ty::FnDef(..)) => {
940+
let b_sig = new_ty.fn_sig(self.tcx);
941+
let a_sig = self
942+
.tcx
943+
.signature_unclosure(substs.as_closure().sig(), b_sig.unsafety());
944+
(Some(a_sig), Some(b_sig))
945+
}
946+
(&ty::FnDef(..), &ty::Closure(_, substs)) => {
947+
let a_sig = prev_ty.fn_sig(self.tcx);
948+
let b_sig = self
949+
.tcx
950+
.signature_unclosure(substs.as_closure().sig(), a_sig.unsafety());
951+
(Some(a_sig), Some(b_sig))
952+
}
953+
(&ty::Closure(_, substs_a), &ty::Closure(_, substs_b)) => (
954+
Some(self.tcx.signature_unclosure(
955+
substs_a.as_closure().sig(),
956+
hir::Unsafety::Normal,
957+
)),
958+
Some(self.tcx.signature_unclosure(
959+
substs_b.as_closure().sig(),
960+
hir::Unsafety::Normal,
961+
)),
962+
),
963+
_ => (None, None),
964+
}
923965
}
924-
966+
};
967+
if let (Some(a_sig), Some(b_sig)) = (a_sig, b_sig) {
925968
// The signature must match.
926-
let a_sig = prev_ty.fn_sig(self.tcx);
927969
let a_sig = self.normalize_associated_types_in(new.span, &a_sig);
928-
let b_sig = new_ty.fn_sig(self.tcx);
929970
let b_sig = self.normalize_associated_types_in(new.span, &b_sig);
930971
let sig = self
931972
.at(cause, self.param_env)
@@ -935,17 +976,23 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
935976

936977
// Reify both sides and return the reified fn pointer type.
937978
let fn_ptr = self.tcx.mk_fn_ptr(sig);
938-
for expr in exprs.iter().map(|e| e.as_coercion_site()).chain(Some(new)) {
939-
// The only adjustment that can produce an fn item is
940-
// `NeverToAny`, so this should always be valid.
979+
let prev_adjustment = match prev_ty.kind {
980+
ty::Closure(..) => Adjust::Pointer(PointerCast::ClosureFnPointer(a_sig.unsafety())),
981+
ty::FnDef(..) => Adjust::Pointer(PointerCast::ReifyFnPointer),
982+
_ => unreachable!(),
983+
};
984+
let next_adjustment = match new_ty.kind {
985+
ty::Closure(..) => Adjust::Pointer(PointerCast::ClosureFnPointer(b_sig.unsafety())),
986+
ty::FnDef(..) => Adjust::Pointer(PointerCast::ReifyFnPointer),
987+
_ => unreachable!(),
988+
};
989+
for expr in exprs.iter().map(|e| e.as_coercion_site()) {
941990
self.apply_adjustments(
942991
expr,
943-
vec![Adjustment {
944-
kind: Adjust::Pointer(PointerCast::ReifyFnPointer),
945-
target: fn_ptr,
946-
}],
992+
vec![Adjustment { kind: prev_adjustment.clone(), target: fn_ptr }],
947993
);
948994
}
995+
self.apply_adjustments(new, vec![Adjustment { kind: next_adjustment, target: fn_ptr }]);
949996
return Ok(fn_ptr);
950997
}
951998

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,39 @@
1+
fn add(a: i32, b: i32) -> i32 {
2+
a + b
3+
}
4+
fn main() {
5+
// We shouldn't coerce capturing closure to a function
6+
let cap = 0;
7+
let _ = match "+" {
8+
"+" => add,
9+
"-" => |a, b| (a - b + cap) as i32,
10+
_ => unimplemented!(),
11+
};
12+
//~^^^ ERROR `match` arms have incompatible types
13+
14+
15+
// We shouldn't coerce capturing closure to a non-capturing closure
16+
let _ = match "+" {
17+
"+" => |a, b| (a + b) as i32,
18+
"-" => |a, b| (a - b + cap) as i32,
19+
_ => unimplemented!(),
20+
};
21+
//~^^^ ERROR `match` arms have incompatible types
22+
23+
24+
// We shouldn't coerce non-capturing closure to a capturing closure
25+
let _ = match "+" {
26+
"+" => |a, b| (a + b + cap) as i32,
27+
"-" => |a, b| (a - b) as i32,
28+
_ => unimplemented!(),
29+
};
30+
//~^^^ ERROR `match` arms have incompatible types
31+
32+
// We shouldn't coerce capturing closure to a capturing closure
33+
let _ = match "+" {
34+
"+" => |a, b| (a + b + cap) as i32,
35+
"-" => |a, b| (a - b + cap) as i32,
36+
_ => unimplemented!(),
37+
};
38+
//~^^^ ERROR `match` arms have incompatible types
39+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,73 @@
1+
error[E0308]: `match` arms have incompatible types
2+
--> $DIR/closure_cap_coerce_many_fail.rs:9:16
3+
|
4+
LL | let _ = match "+" {
5+
| _____________-
6+
LL | | "+" => add,
7+
| | --- this is found to be of type `fn(i32, i32) -> i32 {add}`
8+
LL | | "-" => |a, b| (a - b + cap) as i32,
9+
| | ^^^^^^^^^^^^^^^^^^^^^^^^^^^ expected fn item, found closure
10+
LL | | _ => unimplemented!(),
11+
LL | | };
12+
| |_____- `match` arms have incompatible types
13+
|
14+
= note: expected type `fn(i32, i32) -> i32 {add}`
15+
found closure `[closure@$DIR/closure_cap_coerce_many_fail.rs:9:16: 9:43 cap:_]`
16+
17+
error[E0308]: `match` arms have incompatible types
18+
--> $DIR/closure_cap_coerce_many_fail.rs:18:16
19+
|
20+
LL | let _ = match "+" {
21+
| _____________-
22+
LL | | "+" => |a, b| (a + b) as i32,
23+
| | --------------------- this is found to be of type `[closure@$DIR/closure_cap_coerce_many_fail.rs:17:16: 17:37]`
24+
LL | | "-" => |a, b| (a - b + cap) as i32,
25+
| | ^^^^^^^^^^^^^^^^^^^^^^^^^^^ expected closure, found a different closure
26+
LL | | _ => unimplemented!(),
27+
LL | | };
28+
| |_____- `match` arms have incompatible types
29+
|
30+
= note: expected type `[closure@$DIR/closure_cap_coerce_many_fail.rs:17:16: 17:37]`
31+
found closure `[closure@$DIR/closure_cap_coerce_many_fail.rs:18:16: 18:43 cap:_]`
32+
= note: no two closures, even if identical, have the same type
33+
= help: consider boxing your closure and/or using it as a trait object
34+
35+
error[E0308]: `match` arms have incompatible types
36+
--> $DIR/closure_cap_coerce_many_fail.rs:27:16
37+
|
38+
LL | let _ = match "+" {
39+
| _____________-
40+
LL | | "+" => |a, b| (a + b + cap) as i32,
41+
| | --------------------------- this is found to be of type `[closure@$DIR/closure_cap_coerce_many_fail.rs:26:16: 26:43 cap:_]`
42+
LL | | "-" => |a, b| (a - b) as i32,
43+
| | ^^^^^^^^^^^^^^^^^^^^^ expected closure, found a different closure
44+
LL | | _ => unimplemented!(),
45+
LL | | };
46+
| |_____- `match` arms have incompatible types
47+
|
48+
= note: expected type `[closure@$DIR/closure_cap_coerce_many_fail.rs:26:16: 26:43 cap:_]`
49+
found closure `[closure@$DIR/closure_cap_coerce_many_fail.rs:27:16: 27:37]`
50+
= note: no two closures, even if identical, have the same type
51+
= help: consider boxing your closure and/or using it as a trait object
52+
53+
error[E0308]: `match` arms have incompatible types
54+
--> $DIR/closure_cap_coerce_many_fail.rs:35:16
55+
|
56+
LL | let _ = match "+" {
57+
| _____________-
58+
LL | | "+" => |a, b| (a + b + cap) as i32,
59+
| | --------------------------- this is found to be of type `[closure@$DIR/closure_cap_coerce_many_fail.rs:34:16: 34:43 cap:_]`
60+
LL | | "-" => |a, b| (a - b + cap) as i32,
61+
| | ^^^^^^^^^^^^^^^^^^^^^^^^^^^ expected closure, found a different closure
62+
LL | | _ => unimplemented!(),
63+
LL | | };
64+
| |_____- `match` arms have incompatible types
65+
|
66+
= note: expected type `[closure@$DIR/closure_cap_coerce_many_fail.rs:34:16: 34:43 cap:_]`
67+
found closure `[closure@$DIR/closure_cap_coerce_many_fail.rs:35:16: 35:43 cap:_]`
68+
= note: no two closures, even if identical, have the same type
69+
= help: consider boxing your closure and/or using it as a trait object
70+
71+
error: aborting due to 4 previous errors
72+
73+
For more information about this error, try `rustc --explain E0308`.

0 commit comments

Comments
 (0)