Suggest manual_div_ceil even when right operand is a constant (#13951)

changelog: [`manual_div_ceil`]: lint constants as well

Fix #13950
This commit is contained in:
Alejandra González 2025-01-14 19:06:32 +00:00 committed by GitHub
commit 7e83ec57c6
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
7 changed files with 124 additions and 2 deletions

View file

@ -102,12 +102,48 @@ impl<'tcx> LateLintPass<'tcx> for ManualDivCeil {
{
build_suggestion(cx, expr, add_lhs, div_rhs, &mut applicability);
}
// (x + (Y - 1)) / Y
if inner_op.node == BinOpKind::Add && differ_by_one(inner_rhs, div_rhs) {
build_suggestion(cx, expr, inner_lhs, div_rhs, &mut applicability);
}
// ((Y - 1) + x) / Y
if inner_op.node == BinOpKind::Add && differ_by_one(inner_lhs, div_rhs) {
build_suggestion(cx, expr, inner_rhs, div_rhs, &mut applicability);
}
// (x - (-Y - 1)) / Y
if inner_op.node == BinOpKind::Sub
&& let ExprKind::Unary(UnOp::Neg, abs_div_rhs) = div_rhs.kind
&& differ_by_one(abs_div_rhs, inner_rhs)
{
build_suggestion(cx, expr, inner_lhs, div_rhs, &mut applicability);
}
}
}
extract_msrv_attr!(LateContext);
}
/// Checks if two expressions represent non-zero integer literals such that `small_expr + 1 ==
/// large_expr`.
fn differ_by_one(small_expr: &Expr<'_>, large_expr: &Expr<'_>) -> bool {
if let ExprKind::Lit(small) = small_expr.kind
&& let ExprKind::Lit(large) = large_expr.kind
&& let LitKind::Int(s, _) = small.node
&& let LitKind::Int(l, _) = large.node
{
Some(l.get()) == s.get().checked_add(1)
} else if let ExprKind::Unary(UnOp::Neg, small_inner_expr) = small_expr.kind
&& let ExprKind::Unary(UnOp::Neg, large_inner_expr) = large_expr.kind
{
differ_by_one(large_inner_expr, small_inner_expr)
} else {
false
}
}
fn check_int_ty_and_feature(cx: &LateContext<'_>, expr: &Expr<'_>) -> bool {
let expr_ty = cx.typeck_results().expr_ty(expr);
match expr_ty.peel_refs().kind() {

View file

@ -50,3 +50,14 @@ fn issue_13843() {
let _ = 1_000_000_u32.div_ceil(6u32);
}
fn issue_13950() {
let x = 33u32;
let _ = x.div_ceil(8);
let _ = x.div_ceil(8);
let y = -33i32;
let _ = (y + -8) / -7;
let _ = (-8 + y) / -7;
let _ = (y - 8) / -7;
}

View file

@ -50,3 +50,14 @@ fn issue_13843() {
let _ = (1_000_000 + 6u32 - 1) / 6u32;
}
fn issue_13950() {
let x = 33u32;
let _ = (x + 7) / 8;
let _ = (7 + x) / 8;
let y = -33i32;
let _ = (y + -8) / -7;
let _ = (-8 + y) / -7;
let _ = (y - 8) / -7;
}

View file

@ -85,5 +85,17 @@ error: manually reimplementing `div_ceil`
LL | let _ = (1_000_000 + 6u32 - 1) / 6u32;
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: consider using `.div_ceil()`: `1_000_000_u32.div_ceil(6u32)`
error: aborting due to 14 previous errors
error: manually reimplementing `div_ceil`
--> tests/ui/manual_div_ceil.rs:56:13
|
LL | let _ = (x + 7) / 8;
| ^^^^^^^^^^^ help: consider using `.div_ceil()`: `x.div_ceil(8)`
error: manually reimplementing `div_ceil`
--> tests/ui/manual_div_ceil.rs:57:13
|
LL | let _ = (7 + x) / 8;
| ^^^^^^^^^^^ help: consider using `.div_ceil()`: `x.div_ceil(8)`
error: aborting due to 16 previous errors

View file

@ -50,3 +50,14 @@ fn issue_13843() {
let _ = 1_000_000_u32.div_ceil(6u32);
}
fn issue_13950() {
let x = 33u32;
let _ = x.div_ceil(8);
let _ = x.div_ceil(8);
let y = -33i32;
let _ = y.div_ceil(-7);
let _ = y.div_ceil(-7);
let _ = y.div_ceil(-7);
}

View file

@ -50,3 +50,14 @@ fn issue_13843() {
let _ = (1_000_000 + 6u32 - 1) / 6u32;
}
fn issue_13950() {
let x = 33u32;
let _ = (x + 7) / 8;
let _ = (7 + x) / 8;
let y = -33i32;
let _ = (y + -8) / -7;
let _ = (-8 + y) / -7;
let _ = (y - 8) / -7;
}

View file

@ -109,5 +109,35 @@ error: manually reimplementing `div_ceil`
LL | let _ = (1_000_000 + 6u32 - 1) / 6u32;
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: consider using `.div_ceil()`: `1_000_000_u32.div_ceil(6u32)`
error: aborting due to 18 previous errors
error: manually reimplementing `div_ceil`
--> tests/ui/manual_div_ceil_with_feature.rs:56:13
|
LL | let _ = (x + 7) / 8;
| ^^^^^^^^^^^ help: consider using `.div_ceil()`: `x.div_ceil(8)`
error: manually reimplementing `div_ceil`
--> tests/ui/manual_div_ceil_with_feature.rs:57:13
|
LL | let _ = (7 + x) / 8;
| ^^^^^^^^^^^ help: consider using `.div_ceil()`: `x.div_ceil(8)`
error: manually reimplementing `div_ceil`
--> tests/ui/manual_div_ceil_with_feature.rs:60:13
|
LL | let _ = (y + -8) / -7;
| ^^^^^^^^^^^^^ help: consider using `.div_ceil()`: `y.div_ceil(-7)`
error: manually reimplementing `div_ceil`
--> tests/ui/manual_div_ceil_with_feature.rs:61:13
|
LL | let _ = (-8 + y) / -7;
| ^^^^^^^^^^^^^ help: consider using `.div_ceil()`: `y.div_ceil(-7)`
error: manually reimplementing `div_ceil`
--> tests/ui/manual_div_ceil_with_feature.rs:62:13
|
LL | let _ = (y - 8) / -7;
| ^^^^^^^^^^^^ help: consider using `.div_ceil()`: `y.div_ceil(-7)`
error: aborting due to 23 previous errors