Skip to content

Commit 231f935

Browse files
committed
Detect misparsed binop caused by missing semi
When encountering ```rust foo() *bar = baz; ``` We currently emit potentially two errors, one for the return type of `foo` not being multiplyiable by the type of `bar`, and another for `foo() * bar` not being assignable. We now check for this case and suggest adding a semicolon in the right place. Fix #80446.
1 parent 6f65201 commit 231f935

File tree

4 files changed

+77
-0
lines changed

4 files changed

+77
-0
lines changed

compiler/rustc_hir_typeck/src/expr.rs

+21
Original file line numberDiff line numberDiff line change
@@ -953,12 +953,33 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
953953
Applicability::MachineApplicable,
954954
);
955955
});
956+
self.check_for_missing_semi(lhs, &mut err);
956957

957958
adjust_err(&mut err);
958959

959960
err.emit();
960961
}
961962

963+
/// Check if the expression that could not be assigned to was a typoed expression that
964+
fn check_for_missing_semi(&self, expr: &'tcx hir::Expr<'tcx>, err: &mut Diagnostic) {
965+
if let hir::ExprKind::Binary(binop, lhs, rhs) = expr.kind
966+
&& let hir::BinOpKind::Mul = binop.node
967+
&& self.tcx.sess.source_map().is_multiline(lhs.span.between(rhs.span))
968+
&& rhs.is_syntactic_place_expr()
969+
{
970+
// v missing semicolon here
971+
// foo()
972+
// *bar = baz;
973+
// (#80446).
974+
err.span_suggestion_verbose(
975+
lhs.span.shrink_to_hi(),
976+
"you might have meant to write a semicolon here",
977+
";".to_string(),
978+
Applicability::MachineApplicable,
979+
);
980+
}
981+
}
982+
962983
// Check if an expression `original_expr_id` comes from the condition of a while loop,
963984
/// as opposed from the body of a while loop, which we can naively check by iterating
964985
/// parents until we find a loop...

compiler/rustc_hir_typeck/src/op.rs

+16
Original file line numberDiff line numberDiff line change
@@ -379,6 +379,22 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
379379
(err, output_def_id)
380380
}
381381
};
382+
if self.tcx.sess.source_map().is_multiline(lhs_expr.span.between(rhs_expr.span))
383+
&& let IsAssign::No = is_assign
384+
&& let hir::BinOpKind::Mul = op.node
385+
&& rhs_expr.is_syntactic_place_expr()
386+
{
387+
// v missing semicolon here
388+
// foo()
389+
// *bar = baz;
390+
// (#80446).
391+
err.span_suggestion_verbose(
392+
lhs_expr.span.shrink_to_hi(),
393+
"you might have meant to write a semicolon here",
394+
";".to_string(),
395+
Applicability::MaybeIncorrect,
396+
);
397+
}
382398

383399
let suggest_deref_binop =
384400
|err: &mut DiagnosticBuilder<'_, _>, lhs_deref_ty: Ty<'tcx>| {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
fn foo() {}
2+
fn main() {
3+
let mut y = 42;
4+
let x = &mut y;
5+
foo()
6+
*x = 0; //~ ERROR invalid left-hand side of assignment
7+
//~^ ERROR cannot multiply `()` by `&mut {integer}`
8+
println!("{y}");
9+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
error[E0369]: cannot multiply `()` by `&mut {integer}`
2+
--> $DIR/false-binop-caused-by-missing-semi.rs:6:5
3+
|
4+
LL | foo()
5+
| ----- ()
6+
LL | *x = 0;
7+
| ^- &mut {integer}
8+
|
9+
help: you might have meant to write a semicolon here
10+
|
11+
LL | foo();
12+
| +
13+
14+
error[E0070]: invalid left-hand side of assignment
15+
--> $DIR/false-binop-caused-by-missing-semi.rs:6:8
16+
|
17+
LL | / foo()
18+
LL | | *x = 0;
19+
| | - ^
20+
| |______|
21+
| cannot assign to this expression
22+
|
23+
help: you might have meant to write a semicolon here
24+
|
25+
LL | foo();
26+
| +
27+
28+
error: aborting due to 2 previous errors
29+
30+
Some errors have detailed explanations: E0070, E0369.
31+
For more information about an error, try `rustc --explain E0070`.

0 commit comments

Comments
 (0)