Implement ambiguous_negative_literals
lint
This commit is contained in:
parent
0c81f94b9a
commit
de560c3065
6 changed files with 321 additions and 0 deletions
|
@ -5,6 +5,11 @@ lint_ambiguous_glob_reexport = ambiguous glob re-exports
|
||||||
.label_first_reexport = the name `{$name}` in the {$namespace} namespace is first re-exported here
|
.label_first_reexport = the name `{$name}` in the {$namespace} namespace is first re-exported here
|
||||||
.label_duplicate_reexport = but the name `{$name}` in the {$namespace} namespace is also re-exported here
|
.label_duplicate_reexport = but the name `{$name}` in the {$namespace} namespace is also re-exported here
|
||||||
|
|
||||||
|
lint_ambiguous_negative_literals = `-` has lower precedence than method calls, which might be unexpected
|
||||||
|
.example = e.g. `-4.abs()` equals `-4`; while `(-4).abs()` equals `4`
|
||||||
|
.negative_literal = add parentheses around the `-` and the literal to call the method on a negative literal
|
||||||
|
.current_behavior = add parentheses around the literal and the method call to keep the current behavior
|
||||||
|
|
||||||
lint_ambiguous_wide_pointer_comparisons = ambiguous wide pointer comparison, the comparison includes metadata which may not be expected
|
lint_ambiguous_wide_pointer_comparisons = ambiguous wide pointer comparison, the comparison includes metadata which may not be expected
|
||||||
.addr_metadata_suggestion = use explicit `std::ptr::eq` method to compare metadata and addresses
|
.addr_metadata_suggestion = use explicit `std::ptr::eq` method to compare metadata and addresses
|
||||||
.addr_suggestion = use `std::ptr::addr_eq` or untyped pointers to only compare their addresses
|
.addr_suggestion = use `std::ptr::addr_eq` or untyped pointers to only compare their addresses
|
||||||
|
|
|
@ -72,6 +72,7 @@ mod noop_method_call;
|
||||||
mod opaque_hidden_inferred_bound;
|
mod opaque_hidden_inferred_bound;
|
||||||
mod pass_by_value;
|
mod pass_by_value;
|
||||||
mod passes;
|
mod passes;
|
||||||
|
mod precedence;
|
||||||
mod ptr_nulls;
|
mod ptr_nulls;
|
||||||
mod redundant_semicolon;
|
mod redundant_semicolon;
|
||||||
mod reference_casting;
|
mod reference_casting;
|
||||||
|
@ -109,6 +110,7 @@ use nonstandard_style::*;
|
||||||
use noop_method_call::*;
|
use noop_method_call::*;
|
||||||
use opaque_hidden_inferred_bound::*;
|
use opaque_hidden_inferred_bound::*;
|
||||||
use pass_by_value::*;
|
use pass_by_value::*;
|
||||||
|
use precedence::*;
|
||||||
use ptr_nulls::*;
|
use ptr_nulls::*;
|
||||||
use redundant_semicolon::*;
|
use redundant_semicolon::*;
|
||||||
use reference_casting::*;
|
use reference_casting::*;
|
||||||
|
@ -173,6 +175,7 @@ early_lint_methods!(
|
||||||
RedundantSemicolons: RedundantSemicolons,
|
RedundantSemicolons: RedundantSemicolons,
|
||||||
UnusedDocComment: UnusedDocComment,
|
UnusedDocComment: UnusedDocComment,
|
||||||
Expr2024: Expr2024,
|
Expr2024: Expr2024,
|
||||||
|
Precedence: Precedence,
|
||||||
]
|
]
|
||||||
]
|
]
|
||||||
);
|
);
|
||||||
|
|
|
@ -1489,6 +1489,35 @@ pub struct NonLocalDefinitionsCargoUpdateNote {
|
||||||
pub crate_name: Symbol,
|
pub crate_name: Symbol,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// precedence.rs
|
||||||
|
#[derive(LintDiagnostic)]
|
||||||
|
#[diag(lint_ambiguous_negative_literals)]
|
||||||
|
#[note(lint_example)]
|
||||||
|
pub struct AmbiguousNegativeLiteralsDiag {
|
||||||
|
#[subdiagnostic]
|
||||||
|
pub negative_literal: AmbiguousNegativeLiteralsNegativeLiteralSuggestion,
|
||||||
|
#[subdiagnostic]
|
||||||
|
pub current_behavior: AmbiguousNegativeLiteralsCurrentBehaviorSuggestion,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Subdiagnostic)]
|
||||||
|
#[multipart_suggestion(lint_negative_literal, applicability = "maybe-incorrect")]
|
||||||
|
pub struct AmbiguousNegativeLiteralsNegativeLiteralSuggestion {
|
||||||
|
#[suggestion_part(code = "(")]
|
||||||
|
pub start_span: Span,
|
||||||
|
#[suggestion_part(code = ")")]
|
||||||
|
pub end_span: Span,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Subdiagnostic)]
|
||||||
|
#[multipart_suggestion(lint_current_behavior, applicability = "maybe-incorrect")]
|
||||||
|
pub struct AmbiguousNegativeLiteralsCurrentBehaviorSuggestion {
|
||||||
|
#[suggestion_part(code = "(")]
|
||||||
|
pub start_span: Span,
|
||||||
|
#[suggestion_part(code = ")")]
|
||||||
|
pub end_span: Span,
|
||||||
|
}
|
||||||
|
|
||||||
// pass_by_value.rs
|
// pass_by_value.rs
|
||||||
#[derive(LintDiagnostic)]
|
#[derive(LintDiagnostic)]
|
||||||
#[diag(lint_pass_by_value)]
|
#[diag(lint_pass_by_value)]
|
||||||
|
|
70
compiler/rustc_lint/src/precedence.rs
Normal file
70
compiler/rustc_lint/src/precedence.rs
Normal file
|
@ -0,0 +1,70 @@
|
||||||
|
use rustc_ast::token::LitKind;
|
||||||
|
use rustc_ast::{Expr, ExprKind, MethodCall, UnOp};
|
||||||
|
use rustc_session::{declare_lint, declare_lint_pass};
|
||||||
|
|
||||||
|
use crate::lints::{
|
||||||
|
AmbiguousNegativeLiteralsCurrentBehaviorSuggestion, AmbiguousNegativeLiteralsDiag,
|
||||||
|
AmbiguousNegativeLiteralsNegativeLiteralSuggestion,
|
||||||
|
};
|
||||||
|
use crate::{EarlyContext, EarlyLintPass, LintContext};
|
||||||
|
|
||||||
|
declare_lint! {
|
||||||
|
/// The `ambiguous_negative_literals` lint checks for cases that are
|
||||||
|
/// confusing between a negative literal and a negation that's not part
|
||||||
|
/// of the literal.
|
||||||
|
///
|
||||||
|
/// ### Example
|
||||||
|
///
|
||||||
|
/// ```rust,compile_fail
|
||||||
|
/// # #![allow(unused)]
|
||||||
|
/// -1i32.abs(); // equals -1, while `(-1i32).abs()` equals 1
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// {{produces}}
|
||||||
|
///
|
||||||
|
/// ### Explanation
|
||||||
|
///
|
||||||
|
/// Method calls take precedence over unary precedence. Setting the
|
||||||
|
/// precedence explicitly makes the code clearer and avoid potential bugs.
|
||||||
|
pub AMBIGUOUS_NEGATIVE_LITERALS,
|
||||||
|
Deny,
|
||||||
|
"ambiguous negative literals operations",
|
||||||
|
report_in_external_macro
|
||||||
|
}
|
||||||
|
|
||||||
|
declare_lint_pass!(Precedence => [AMBIGUOUS_NEGATIVE_LITERALS]);
|
||||||
|
|
||||||
|
impl EarlyLintPass for Precedence {
|
||||||
|
fn check_expr(&mut self, cx: &EarlyContext<'_>, expr: &Expr) {
|
||||||
|
let ExprKind::Unary(UnOp::Neg, operand) = &expr.kind else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut arg = operand;
|
||||||
|
let mut at_least_one = false;
|
||||||
|
while let ExprKind::MethodCall(box MethodCall { receiver, .. }) = &arg.kind {
|
||||||
|
at_least_one = true;
|
||||||
|
arg = receiver;
|
||||||
|
}
|
||||||
|
|
||||||
|
if at_least_one
|
||||||
|
&& let ExprKind::Lit(lit) = &arg.kind
|
||||||
|
&& let LitKind::Integer | LitKind::Float = &lit.kind
|
||||||
|
{
|
||||||
|
cx.emit_span_lint(
|
||||||
|
AMBIGUOUS_NEGATIVE_LITERALS,
|
||||||
|
expr.span,
|
||||||
|
AmbiguousNegativeLiteralsDiag {
|
||||||
|
negative_literal: AmbiguousNegativeLiteralsNegativeLiteralSuggestion {
|
||||||
|
start_span: expr.span.shrink_to_lo(),
|
||||||
|
end_span: arg.span.shrink_to_hi(),
|
||||||
|
},
|
||||||
|
current_behavior: AmbiguousNegativeLiteralsCurrentBehaviorSuggestion {
|
||||||
|
start_span: operand.span.shrink_to_lo(),
|
||||||
|
end_span: operand.span.shrink_to_hi(),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
35
tests/ui/lint/negative_literals.rs
Normal file
35
tests/ui/lint/negative_literals.rs
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
//@ check-fail
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
let _ = -1i32.abs();
|
||||||
|
//~^ ERROR `-` has lower precedence than method calls
|
||||||
|
let _ = -1f32.abs();
|
||||||
|
//~^ ERROR `-` has lower precedence than method calls
|
||||||
|
let _ = -1f64.asin();
|
||||||
|
//~^ ERROR `-` has lower precedence than method calls
|
||||||
|
let _ = -1f64.asinh();
|
||||||
|
//~^ ERROR `-` has lower precedence than method calls
|
||||||
|
let _ = -1f64.tan();
|
||||||
|
//~^ ERROR `-` has lower precedence than method calls
|
||||||
|
let _ = -1f64.tanh();
|
||||||
|
//~^ ERROR `-` has lower precedence than method calls
|
||||||
|
let _ = -1.0_f64.cos().cos();
|
||||||
|
//~^ ERROR `-` has lower precedence than method calls
|
||||||
|
let _ = -1.0_f64.cos().sin();
|
||||||
|
//~^ ERROR `-` has lower precedence than method calls
|
||||||
|
let _ = -1.0_f64.sin().cos();
|
||||||
|
//~^ ERROR `-` has lower precedence than method calls
|
||||||
|
let _ = -1f64.sin().sin();
|
||||||
|
//~^ ERROR `-` has lower precedence than method calls
|
||||||
|
|
||||||
|
dbg!( -1.0_f32.cos() );
|
||||||
|
//~^ ERROR `-` has lower precedence than method calls
|
||||||
|
|
||||||
|
// should not warn
|
||||||
|
let _ = (-1i32).abs();
|
||||||
|
let _ = (-1f32).abs();
|
||||||
|
let _ = -(1i32).abs();
|
||||||
|
let _ = -(1f32).abs();
|
||||||
|
let _ = -(1i32.abs());
|
||||||
|
let _ = -(1f32.abs());
|
||||||
|
}
|
179
tests/ui/lint/negative_literals.stderr
Normal file
179
tests/ui/lint/negative_literals.stderr
Normal file
|
@ -0,0 +1,179 @@
|
||||||
|
error: `-` has lower precedence than method calls, which might be unexpected
|
||||||
|
--> $DIR/negative_literals.rs:4:13
|
||||||
|
|
|
||||||
|
LL | let _ = -1i32.abs();
|
||||||
|
| ^^^^^^^^^^^
|
||||||
|
|
|
||||||
|
= note: e.g. `-4.abs()` equals `-4`; while `(-4).abs()` equals `4`
|
||||||
|
= note: `#[deny(ambiguous_negative_literals)]` on by default
|
||||||
|
help: add parentheses around the `-` and the literal to call the method on a negative literal
|
||||||
|
|
|
||||||
|
LL | let _ = (-1i32).abs();
|
||||||
|
| + +
|
||||||
|
help: add parentheses around the literal and the method call to keep the current behavior
|
||||||
|
|
|
||||||
|
LL | let _ = -(1i32.abs());
|
||||||
|
| + +
|
||||||
|
|
||||||
|
error: `-` has lower precedence than method calls, which might be unexpected
|
||||||
|
--> $DIR/negative_literals.rs:6:13
|
||||||
|
|
|
||||||
|
LL | let _ = -1f32.abs();
|
||||||
|
| ^^^^^^^^^^^
|
||||||
|
|
|
||||||
|
= note: e.g. `-4.abs()` equals `-4`; while `(-4).abs()` equals `4`
|
||||||
|
help: add parentheses around the `-` and the literal to call the method on a negative literal
|
||||||
|
|
|
||||||
|
LL | let _ = (-1f32).abs();
|
||||||
|
| + +
|
||||||
|
help: add parentheses around the literal and the method call to keep the current behavior
|
||||||
|
|
|
||||||
|
LL | let _ = -(1f32.abs());
|
||||||
|
| + +
|
||||||
|
|
||||||
|
error: `-` has lower precedence than method calls, which might be unexpected
|
||||||
|
--> $DIR/negative_literals.rs:8:13
|
||||||
|
|
|
||||||
|
LL | let _ = -1f64.asin();
|
||||||
|
| ^^^^^^^^^^^^
|
||||||
|
|
|
||||||
|
= note: e.g. `-4.abs()` equals `-4`; while `(-4).abs()` equals `4`
|
||||||
|
help: add parentheses around the `-` and the literal to call the method on a negative literal
|
||||||
|
|
|
||||||
|
LL | let _ = (-1f64).asin();
|
||||||
|
| + +
|
||||||
|
help: add parentheses around the literal and the method call to keep the current behavior
|
||||||
|
|
|
||||||
|
LL | let _ = -(1f64.asin());
|
||||||
|
| + +
|
||||||
|
|
||||||
|
error: `-` has lower precedence than method calls, which might be unexpected
|
||||||
|
--> $DIR/negative_literals.rs:10:13
|
||||||
|
|
|
||||||
|
LL | let _ = -1f64.asinh();
|
||||||
|
| ^^^^^^^^^^^^^
|
||||||
|
|
|
||||||
|
= note: e.g. `-4.abs()` equals `-4`; while `(-4).abs()` equals `4`
|
||||||
|
help: add parentheses around the `-` and the literal to call the method on a negative literal
|
||||||
|
|
|
||||||
|
LL | let _ = (-1f64).asinh();
|
||||||
|
| + +
|
||||||
|
help: add parentheses around the literal and the method call to keep the current behavior
|
||||||
|
|
|
||||||
|
LL | let _ = -(1f64.asinh());
|
||||||
|
| + +
|
||||||
|
|
||||||
|
error: `-` has lower precedence than method calls, which might be unexpected
|
||||||
|
--> $DIR/negative_literals.rs:12:13
|
||||||
|
|
|
||||||
|
LL | let _ = -1f64.tan();
|
||||||
|
| ^^^^^^^^^^^
|
||||||
|
|
|
||||||
|
= note: e.g. `-4.abs()` equals `-4`; while `(-4).abs()` equals `4`
|
||||||
|
help: add parentheses around the `-` and the literal to call the method on a negative literal
|
||||||
|
|
|
||||||
|
LL | let _ = (-1f64).tan();
|
||||||
|
| + +
|
||||||
|
help: add parentheses around the literal and the method call to keep the current behavior
|
||||||
|
|
|
||||||
|
LL | let _ = -(1f64.tan());
|
||||||
|
| + +
|
||||||
|
|
||||||
|
error: `-` has lower precedence than method calls, which might be unexpected
|
||||||
|
--> $DIR/negative_literals.rs:14:13
|
||||||
|
|
|
||||||
|
LL | let _ = -1f64.tanh();
|
||||||
|
| ^^^^^^^^^^^^
|
||||||
|
|
|
||||||
|
= note: e.g. `-4.abs()` equals `-4`; while `(-4).abs()` equals `4`
|
||||||
|
help: add parentheses around the `-` and the literal to call the method on a negative literal
|
||||||
|
|
|
||||||
|
LL | let _ = (-1f64).tanh();
|
||||||
|
| + +
|
||||||
|
help: add parentheses around the literal and the method call to keep the current behavior
|
||||||
|
|
|
||||||
|
LL | let _ = -(1f64.tanh());
|
||||||
|
| + +
|
||||||
|
|
||||||
|
error: `-` has lower precedence than method calls, which might be unexpected
|
||||||
|
--> $DIR/negative_literals.rs:16:13
|
||||||
|
|
|
||||||
|
LL | let _ = -1.0_f64.cos().cos();
|
||||||
|
| ^^^^^^^^^^^^^^^^^^^^
|
||||||
|
|
|
||||||
|
= note: e.g. `-4.abs()` equals `-4`; while `(-4).abs()` equals `4`
|
||||||
|
help: add parentheses around the `-` and the literal to call the method on a negative literal
|
||||||
|
|
|
||||||
|
LL | let _ = (-1.0_f64).cos().cos();
|
||||||
|
| + +
|
||||||
|
help: add parentheses around the literal and the method call to keep the current behavior
|
||||||
|
|
|
||||||
|
LL | let _ = -(1.0_f64.cos().cos());
|
||||||
|
| + +
|
||||||
|
|
||||||
|
error: `-` has lower precedence than method calls, which might be unexpected
|
||||||
|
--> $DIR/negative_literals.rs:18:13
|
||||||
|
|
|
||||||
|
LL | let _ = -1.0_f64.cos().sin();
|
||||||
|
| ^^^^^^^^^^^^^^^^^^^^
|
||||||
|
|
|
||||||
|
= note: e.g. `-4.abs()` equals `-4`; while `(-4).abs()` equals `4`
|
||||||
|
help: add parentheses around the `-` and the literal to call the method on a negative literal
|
||||||
|
|
|
||||||
|
LL | let _ = (-1.0_f64).cos().sin();
|
||||||
|
| + +
|
||||||
|
help: add parentheses around the literal and the method call to keep the current behavior
|
||||||
|
|
|
||||||
|
LL | let _ = -(1.0_f64.cos().sin());
|
||||||
|
| + +
|
||||||
|
|
||||||
|
error: `-` has lower precedence than method calls, which might be unexpected
|
||||||
|
--> $DIR/negative_literals.rs:20:13
|
||||||
|
|
|
||||||
|
LL | let _ = -1.0_f64.sin().cos();
|
||||||
|
| ^^^^^^^^^^^^^^^^^^^^
|
||||||
|
|
|
||||||
|
= note: e.g. `-4.abs()` equals `-4`; while `(-4).abs()` equals `4`
|
||||||
|
help: add parentheses around the `-` and the literal to call the method on a negative literal
|
||||||
|
|
|
||||||
|
LL | let _ = (-1.0_f64).sin().cos();
|
||||||
|
| + +
|
||||||
|
help: add parentheses around the literal and the method call to keep the current behavior
|
||||||
|
|
|
||||||
|
LL | let _ = -(1.0_f64.sin().cos());
|
||||||
|
| + +
|
||||||
|
|
||||||
|
error: `-` has lower precedence than method calls, which might be unexpected
|
||||||
|
--> $DIR/negative_literals.rs:22:13
|
||||||
|
|
|
||||||
|
LL | let _ = -1f64.sin().sin();
|
||||||
|
| ^^^^^^^^^^^^^^^^^
|
||||||
|
|
|
||||||
|
= note: e.g. `-4.abs()` equals `-4`; while `(-4).abs()` equals `4`
|
||||||
|
help: add parentheses around the `-` and the literal to call the method on a negative literal
|
||||||
|
|
|
||||||
|
LL | let _ = (-1f64).sin().sin();
|
||||||
|
| + +
|
||||||
|
help: add parentheses around the literal and the method call to keep the current behavior
|
||||||
|
|
|
||||||
|
LL | let _ = -(1f64.sin().sin());
|
||||||
|
| + +
|
||||||
|
|
||||||
|
error: `-` has lower precedence than method calls, which might be unexpected
|
||||||
|
--> $DIR/negative_literals.rs:25:11
|
||||||
|
|
|
||||||
|
LL | dbg!( -1.0_f32.cos() );
|
||||||
|
| ^^^^^^^^^^^^^^
|
||||||
|
|
|
||||||
|
= note: e.g. `-4.abs()` equals `-4`; while `(-4).abs()` equals `4`
|
||||||
|
help: add parentheses around the `-` and the literal to call the method on a negative literal
|
||||||
|
|
|
||||||
|
LL | dbg!( (-1.0_f32).cos() );
|
||||||
|
| + +
|
||||||
|
help: add parentheses around the literal and the method call to keep the current behavior
|
||||||
|
|
|
||||||
|
LL | dbg!( -(1.0_f32.cos()) );
|
||||||
|
| + +
|
||||||
|
|
||||||
|
error: aborting due to 11 previous errors
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue