Update unop path, fix tests
This commit is contained in:
parent
4d0fe27896
commit
dc41dbaf8e
5 changed files with 51 additions and 94 deletions
|
@ -456,25 +456,32 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
|
||||||
// the resulting predicate generates a more specific
|
// the resulting predicate generates a more specific
|
||||||
// suggestion for the user.
|
// suggestion for the user.
|
||||||
let errors = self
|
let errors = self
|
||||||
.lookup_op_method(lhs_ty, &[rhs_ty], Op::Binary(op, is_assign))
|
.lookup_op_method(
|
||||||
.unwrap_err();
|
lhs_ty,
|
||||||
let predicates = errors
|
Some(rhs_ty),
|
||||||
.into_iter()
|
Some(rhs_expr),
|
||||||
.filter_map(|error| error.obligation.predicate.to_opt_poly_trait_pred())
|
Op::Binary(op, is_assign),
|
||||||
.collect::<Vec<_>>();
|
)
|
||||||
if !predicates.is_empty() {
|
.unwrap_err();
|
||||||
for pred in predicates {
|
let predicates = errors
|
||||||
self.infcx.suggest_restricting_param_bound(&mut err,
|
.into_iter()
|
||||||
pred,
|
.filter_map(|error| error.obligation.predicate.to_opt_poly_trait_pred())
|
||||||
self.body_id,
|
.collect::<Vec<_>>();
|
||||||
);
|
if !predicates.is_empty() {
|
||||||
|
for pred in predicates {
|
||||||
|
self.infcx.suggest_restricting_param_bound(
|
||||||
|
&mut err,
|
||||||
|
pred,
|
||||||
|
self.body_id,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} else if *ty != lhs_ty {
|
||||||
|
// When we know that a missing bound is responsible, we don't show
|
||||||
|
// this note as it is redundant.
|
||||||
|
err.note(&format!(
|
||||||
|
"the trait `{missing_trait}` is not implemented for `{lhs_ty}`"
|
||||||
|
));
|
||||||
}
|
}
|
||||||
} else if *ty != lhs_ty {
|
|
||||||
// When we know that a missing bound is responsible, we don't show
|
|
||||||
// this note as it is redundant.
|
|
||||||
err.note(&format!(
|
|
||||||
"the trait `{missing_trait}` is not implemented for `{lhs_ty}`"
|
|
||||||
));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
err.emit();
|
err.emit();
|
||||||
|
@ -663,24 +670,22 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
|
||||||
ex.span,
|
ex.span,
|
||||||
format!("cannot apply unary operator `{}`", op.as_str()),
|
format!("cannot apply unary operator `{}`", op.as_str()),
|
||||||
);
|
);
|
||||||
let missing_trait = match op {
|
|
||||||
hir::UnOp::Deref => unreachable!("check unary op `-` or `!` only"),
|
|
||||||
hir::UnOp::Not => "std::ops::Not",
|
|
||||||
hir::UnOp::Neg => "std::ops::Neg",
|
|
||||||
};
|
|
||||||
let mut visitor = TypeParamVisitor(vec![]);
|
let mut visitor = TypeParamVisitor(vec![]);
|
||||||
visitor.visit_ty(operand_ty);
|
visitor.visit_ty(operand_ty);
|
||||||
if let [ty] = &visitor.0[..] && let ty::Param(p) = *operand_ty.kind() {
|
if let [_] = &visitor.0[..] && let ty::Param(_) = *operand_ty.kind() {
|
||||||
suggest_constraining_param(
|
let predicates = errors
|
||||||
self.tcx,
|
.iter()
|
||||||
self.body_id,
|
.filter_map(|error| {
|
||||||
&mut err,
|
error.obligation.predicate.clone().to_opt_poly_trait_pred()
|
||||||
*ty,
|
});
|
||||||
operand_ty,
|
for pred in predicates {
|
||||||
missing_trait,
|
self.infcx.suggest_restricting_param_bound(
|
||||||
p,
|
&mut err,
|
||||||
true,
|
pred,
|
||||||
);
|
self.body_id,
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let sp = self.tcx.sess.source_map().start_point(ex.span);
|
let sp = self.tcx.sess.source_map().start_point(ex.span);
|
||||||
|
|
|
@ -1,46 +0,0 @@
|
||||||
// run-rustfix
|
|
||||||
|
|
||||||
use std::ops::Add;
|
|
||||||
|
|
||||||
struct A<B>(B);
|
|
||||||
|
|
||||||
impl<B> Add for A<B> where B: Add + Add<Output = B> {
|
|
||||||
type Output = Self;
|
|
||||||
|
|
||||||
fn add(self, rhs: Self) -> Self {
|
|
||||||
A(self.0 + rhs.0) //~ ERROR mismatched types
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct C<B>(B);
|
|
||||||
|
|
||||||
impl<B: Add + Add<Output = B>> Add for C<B> {
|
|
||||||
type Output = Self;
|
|
||||||
|
|
||||||
fn add(self, rhs: Self) -> Self {
|
|
||||||
Self(self.0 + rhs.0) //~ ERROR mismatched types
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct D<B>(B);
|
|
||||||
|
|
||||||
impl<B: std::ops::Add> Add for D<B> {
|
|
||||||
type Output = Self;
|
|
||||||
|
|
||||||
fn add(self, rhs: Self) -> Self {
|
|
||||||
Self(self.0 + rhs.0) //~ ERROR cannot add `B` to `B`
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct E<B>(B);
|
|
||||||
|
|
||||||
impl<B: Add> Add for E<B> where B: Add<Output = B>, B: Add<Output = B> {
|
|
||||||
//~^ ERROR equality constraints are not yet supported in `where` clauses
|
|
||||||
type Output = Self;
|
|
||||||
|
|
||||||
fn add(self, rhs: Self) -> Self {
|
|
||||||
Self(self.0 + rhs.0) //~ ERROR mismatched types
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn main() {}
|
|
|
@ -1,5 +1,3 @@
|
||||||
// run-rustfix
|
|
||||||
|
|
||||||
use std::ops::Add;
|
use std::ops::Add;
|
||||||
|
|
||||||
struct A<B>(B);
|
struct A<B>(B);
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
error: equality constraints are not yet supported in `where` clauses
|
error: equality constraints are not yet supported in `where` clauses
|
||||||
--> $DIR/missing-bounds.rs:37:33
|
--> $DIR/missing-bounds.rs:35:33
|
||||||
|
|
|
|
||||||
LL | impl<B: Add> Add for E<B> where <B as Add>::Output = B {
|
LL | impl<B: Add> Add for E<B> where <B as Add>::Output = B {
|
||||||
| ^^^^^^^^^^^^^^^^^^^^^^ not supported
|
| ^^^^^^^^^^^^^^^^^^^^^^ not supported
|
||||||
|
@ -11,7 +11,7 @@ LL | impl<B: Add> Add for E<B> where B: Add<Output = B> {
|
||||||
| ~~~~~~~~~~~~~~~~~~
|
| ~~~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
error[E0308]: mismatched types
|
error[E0308]: mismatched types
|
||||||
--> $DIR/missing-bounds.rs:11:11
|
--> $DIR/missing-bounds.rs:9:11
|
||||||
|
|
|
|
||||||
LL | impl<B> Add for A<B> where B: Add {
|
LL | impl<B> Add for A<B> where B: Add {
|
||||||
| - this type parameter
|
| - this type parameter
|
||||||
|
@ -24,7 +24,7 @@ LL | A(self.0 + rhs.0)
|
||||||
= note: expected type parameter `B`
|
= note: expected type parameter `B`
|
||||||
found associated type `<B as Add>::Output`
|
found associated type `<B as Add>::Output`
|
||||||
note: tuple struct defined here
|
note: tuple struct defined here
|
||||||
--> $DIR/missing-bounds.rs:5:8
|
--> $DIR/missing-bounds.rs:3:8
|
||||||
|
|
|
|
||||||
LL | struct A<B>(B);
|
LL | struct A<B>(B);
|
||||||
| ^
|
| ^
|
||||||
|
@ -34,7 +34,7 @@ LL | impl<B> Add for A<B> where B: Add + Add<Output = B> {
|
||||||
| +++++++++++++++++
|
| +++++++++++++++++
|
||||||
|
|
||||||
error[E0308]: mismatched types
|
error[E0308]: mismatched types
|
||||||
--> $DIR/missing-bounds.rs:21:14
|
--> $DIR/missing-bounds.rs:19:14
|
||||||
|
|
|
|
||||||
LL | impl<B: Add> Add for C<B> {
|
LL | impl<B: Add> Add for C<B> {
|
||||||
| - this type parameter
|
| - this type parameter
|
||||||
|
@ -47,7 +47,7 @@ LL | Self(self.0 + rhs.0)
|
||||||
= note: expected type parameter `B`
|
= note: expected type parameter `B`
|
||||||
found associated type `<B as Add>::Output`
|
found associated type `<B as Add>::Output`
|
||||||
note: tuple struct defined here
|
note: tuple struct defined here
|
||||||
--> $DIR/missing-bounds.rs:15:8
|
--> $DIR/missing-bounds.rs:13:8
|
||||||
|
|
|
|
||||||
LL | struct C<B>(B);
|
LL | struct C<B>(B);
|
||||||
| ^
|
| ^
|
||||||
|
@ -57,7 +57,7 @@ LL | impl<B: Add + Add<Output = B>> Add for C<B> {
|
||||||
| +++++++++++++++++
|
| +++++++++++++++++
|
||||||
|
|
||||||
error[E0369]: cannot add `B` to `B`
|
error[E0369]: cannot add `B` to `B`
|
||||||
--> $DIR/missing-bounds.rs:31:21
|
--> $DIR/missing-bounds.rs:29:21
|
||||||
|
|
|
|
||||||
LL | Self(self.0 + rhs.0)
|
LL | Self(self.0 + rhs.0)
|
||||||
| ------ ^ ----- B
|
| ------ ^ ----- B
|
||||||
|
@ -70,7 +70,7 @@ LL | impl<B: std::ops::Add> Add for D<B> {
|
||||||
| +++++++++++++++
|
| +++++++++++++++
|
||||||
|
|
||||||
error[E0308]: mismatched types
|
error[E0308]: mismatched types
|
||||||
--> $DIR/missing-bounds.rs:42:14
|
--> $DIR/missing-bounds.rs:40:14
|
||||||
|
|
|
|
||||||
LL | impl<B: Add> Add for E<B> where <B as Add>::Output = B {
|
LL | impl<B: Add> Add for E<B> where <B as Add>::Output = B {
|
||||||
| - this type parameter
|
| - this type parameter
|
||||||
|
@ -83,7 +83,7 @@ LL | Self(self.0 + rhs.0)
|
||||||
= note: expected type parameter `B`
|
= note: expected type parameter `B`
|
||||||
found associated type `<B as Add>::Output`
|
found associated type `<B as Add>::Output`
|
||||||
note: tuple struct defined here
|
note: tuple struct defined here
|
||||||
--> $DIR/missing-bounds.rs:35:8
|
--> $DIR/missing-bounds.rs:33:8
|
||||||
|
|
|
|
||||||
LL | struct E<B>(B);
|
LL | struct E<B>(B);
|
||||||
| ^
|
| ^
|
||||||
|
|
|
@ -32,8 +32,8 @@ LL | let y = -x;
|
||||||
|
|
|
|
||||||
help: consider restricting type parameter `T`
|
help: consider restricting type parameter `T`
|
||||||
|
|
|
|
||||||
LL | fn baz<T: std::ops::Neg<Output = T>>(x: T) {
|
LL | fn baz<T: std::ops::Neg>(x: T) {
|
||||||
| +++++++++++++++++++++++++++
|
| +++++++++++++++
|
||||||
|
|
||||||
error[E0600]: cannot apply unary operator `!` to type `T`
|
error[E0600]: cannot apply unary operator `!` to type `T`
|
||||||
--> $DIR/missing_trait_impl.rs:14:13
|
--> $DIR/missing_trait_impl.rs:14:13
|
||||||
|
@ -43,8 +43,8 @@ LL | let y = !x;
|
||||||
|
|
|
|
||||||
help: consider restricting type parameter `T`
|
help: consider restricting type parameter `T`
|
||||||
|
|
|
|
||||||
LL | fn baz<T: std::ops::Not<Output = T>>(x: T) {
|
LL | fn baz<T: std::ops::Not>(x: T) {
|
||||||
| +++++++++++++++++++++++++++
|
| +++++++++++++++
|
||||||
|
|
||||||
error[E0614]: type `T` cannot be dereferenced
|
error[E0614]: type `T` cannot be dereferenced
|
||||||
--> $DIR/missing_trait_impl.rs:15:13
|
--> $DIR/missing_trait_impl.rs:15:13
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue