Skip to content

Commit c412528

Browse files
committed
Auto merge of rust-lang#12840 - tesuji:const-asserts, r=llogiq
Don't lint `assertions_on_constants` on any const assertions close rust-lang#12816 close rust-lang#12847 cc rust-lang#12817 ---- changelog: Fix false positives in consts for `assertions_on_constants` and `unnecessary_operation`.
2 parents 0505dad + a0234b4 commit c412528

9 files changed

+105
-22
lines changed

clippy_lints/src/assertions_on_constants.rs

+10-10
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,8 @@
1-
use clippy_utils::consts::{constant_with_source, Constant, ConstantSource};
1+
use clippy_utils::consts::{constant, Constant};
22
use clippy_utils::diagnostics::span_lint_and_help;
3+
use clippy_utils::is_inside_always_const_context;
34
use clippy_utils::macros::{find_assert_args, root_macro_call_first_node, PanicExpn};
4-
use rustc_hir::{Expr, Item, ItemKind, Node};
5+
use rustc_hir::{Expr, ExprKind};
56
use rustc_lint::{LateContext, LateLintPass};
67
use rustc_session::declare_lint_pass;
78
use rustc_span::sym;
@@ -42,17 +43,16 @@ impl<'tcx> LateLintPass<'tcx> for AssertionsOnConstants {
4243
let Some((condition, panic_expn)) = find_assert_args(cx, e, macro_call.expn) else {
4344
return;
4445
};
45-
let Some((Constant::Bool(val), source)) = constant_with_source(cx, cx.typeck_results(), condition) else {
46+
let Some(Constant::Bool(val)) = constant(cx, cx.typeck_results(), condition) else {
4647
return;
4748
};
48-
if let ConstantSource::Constant = source
49-
&& let Node::Item(Item {
50-
kind: ItemKind::Const(..),
51-
..
52-
}) = cx.tcx.parent_hir_node(e.hir_id)
53-
{
54-
return;
49+
50+
match condition.kind {
51+
ExprKind::Path(..) | ExprKind::Lit(_) => {},
52+
_ if is_inside_always_const_context(cx.tcx, e.hir_id) => return,
53+
_ => {},
5554
}
55+
5656
if val {
5757
span_lint_and_help(
5858
cx,

clippy_lints/src/default_numeric_fallback.rs

+2
Original file line numberDiff line numberDiff line change
@@ -50,6 +50,8 @@ declare_lint_pass!(DefaultNumericFallback => [DEFAULT_NUMERIC_FALLBACK]);
5050
impl<'tcx> LateLintPass<'tcx> for DefaultNumericFallback {
5151
fn check_body(&mut self, cx: &LateContext<'tcx>, body: &Body<'tcx>) {
5252
let hir = cx.tcx.hir();
53+
// NOTE: this is different from `clippy_utils::is_inside_always_const_context`.
54+
// Inline const supports type inference.
5355
let is_parent_const = matches!(
5456
hir.body_const_context(hir.body_owner_def_id(body.id())),
5557
Some(ConstContext::Const { inline: false } | ConstContext::Static(_))

clippy_lints/src/no_effect.rs

+7-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,9 @@
11
use clippy_utils::diagnostics::{span_lint_hir, span_lint_hir_and_then};
22
use clippy_utils::source::snippet_opt;
33
use clippy_utils::ty::has_drop;
4-
use clippy_utils::{any_parent_is_automatically_derived, is_lint_allowed, path_to_local, peel_blocks};
4+
use clippy_utils::{
5+
any_parent_is_automatically_derived, is_inside_always_const_context, is_lint_allowed, path_to_local, peel_blocks,
6+
};
57
use rustc_errors::Applicability;
68
use rustc_hir::def::{DefKind, Res};
79
use rustc_hir::{
@@ -258,13 +260,16 @@ fn has_no_effect(cx: &LateContext<'_>, expr: &Expr<'_>) -> bool {
258260

259261
fn check_unnecessary_operation(cx: &LateContext<'_>, stmt: &Stmt<'_>) {
260262
if let StmtKind::Semi(expr) = stmt.kind
263+
&& !in_external_macro(cx.sess(), stmt.span)
261264
&& let ctxt = stmt.span.ctxt()
262265
&& expr.span.ctxt() == ctxt
263266
&& let Some(reduced) = reduce_expression(cx, expr)
264-
&& !in_external_macro(cx.sess(), stmt.span)
265267
&& reduced.iter().all(|e| e.span.ctxt() == ctxt)
266268
{
267269
if let ExprKind::Index(..) = &expr.kind {
270+
if is_inside_always_const_context(cx.tcx, expr.hir_id) {
271+
return;
272+
}
268273
let snippet =
269274
if let (Some(arr), Some(func)) = (snippet_opt(cx, reduced[0].span), snippet_opt(cx, reduced[1].span)) {
270275
format!("assert!({}.len() > {});", &arr, &func)

clippy_utils/src/lib.rs

+26-5
Original file line numberDiff line numberDiff line change
@@ -101,10 +101,10 @@ use rustc_hir::hir_id::{HirIdMap, HirIdSet};
101101
use rustc_hir::intravisit::{walk_expr, FnKind, Visitor};
102102
use rustc_hir::LangItem::{OptionNone, OptionSome, ResultErr, ResultOk};
103103
use rustc_hir::{
104-
self as hir, def, Arm, ArrayLen, BindingMode, Block, BlockCheckMode, Body, ByRef, Closure, Destination, Expr,
105-
ExprField, ExprKind, FnDecl, FnRetTy, GenericArgs, HirId, Impl, ImplItem, ImplItemKind, ImplItemRef, Item,
106-
ItemKind, LangItem, LetStmt, MatchSource, Mutability, Node, OwnerId, Param, Pat, PatKind, Path, PathSegment,
107-
PrimTy, QPath, Stmt, StmtKind, TraitItem, TraitItemKind, TraitItemRef, TraitRef, TyKind, UnOp,
104+
self as hir, def, Arm, ArrayLen, BindingMode, Block, BlockCheckMode, Body, ByRef, Closure, ConstContext,
105+
Destination, Expr, ExprField, ExprKind, FnDecl, FnRetTy, GenericArgs, HirId, Impl, ImplItem, ImplItemKind,
106+
ImplItemRef, Item, ItemKind, LangItem, LetStmt, MatchSource, Mutability, Node, OwnerId, Param, Pat, PatKind, Path,
107+
PathSegment, PrimTy, QPath, Stmt, StmtKind, TraitItem, TraitItemKind, TraitItemRef, TraitRef, TyKind, UnOp,
108108
};
109109
use rustc_lexer::{tokenize, TokenKind};
110110
use rustc_lint::{LateContext, Level, Lint, LintContext};
@@ -209,7 +209,10 @@ pub fn local_is_initialized(cx: &LateContext<'_>, local: HirId) -> bool {
209209
false
210210
}
211211

212-
/// Returns `true` if the given `NodeId` is inside a constant context
212+
/// Returns `true` if the given `HirId` is inside a constant context.
213+
///
214+
/// This is the same as `is_inside_always_const_context`, but also includes
215+
/// `const fn`.
213216
///
214217
/// # Example
215218
///
@@ -222,6 +225,24 @@ pub fn in_constant(cx: &LateContext<'_>, id: HirId) -> bool {
222225
cx.tcx.hir().is_inside_const_context(id)
223226
}
224227

228+
/// Returns `true` if the given `HirId` is inside an always constant context.
229+
///
230+
/// This context includes:
231+
/// * const/static items
232+
/// * const blocks (or inline consts)
233+
/// * associated constants
234+
pub fn is_inside_always_const_context(tcx: TyCtxt<'_>, hir_id: HirId) -> bool {
235+
use ConstContext::{Const, ConstFn, Static};
236+
let hir = tcx.hir();
237+
let Some(ctx) = hir.body_const_context(hir.enclosing_body_owner(hir_id)) else {
238+
return false;
239+
};
240+
match ctx {
241+
ConstFn => false,
242+
Static(_) | Const { inline: _ } => true,
243+
}
244+
}
245+
225246
/// Checks if a `Res` refers to a constructor of a `LangItem`
226247
/// For example, use this to check whether a function call or a pattern is `Some(..)`.
227248
pub fn is_res_lang_ctor(cx: &LateContext<'_>, res: Res, lang_item: LangItem) -> bool {

tests/ui/assertions_on_constants.rs

+10-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
#![allow(non_fmt_panics, clippy::needless_bool)]
1+
#![allow(non_fmt_panics, clippy::needless_bool, clippy::eq_op)]
22

33
macro_rules! assert_const {
44
($len:expr) => {
@@ -49,7 +49,16 @@ fn main() {
4949
const _: () = assert!(true);
5050
//~^ ERROR: `assert!(true)` will be optimized out by the compiler
5151

52+
assert!(8 == (7 + 1));
53+
//~^ ERROR: `assert!(true)` will be optimized out by the compiler
54+
5255
// Don't lint if the value is dependent on a defined constant:
5356
const N: usize = 1024;
5457
const _: () = assert!(N.is_power_of_two());
5558
}
59+
60+
const _: () = {
61+
assert!(true);
62+
//~^ ERROR: `assert!(true)` will be optimized out by the compiler
63+
assert!(8 == (7 + 1));
64+
};

tests/ui/assertions_on_constants.stderr

+17-1
Original file line numberDiff line numberDiff line change
@@ -80,5 +80,21 @@ LL | const _: () = assert!(true);
8080
|
8181
= help: remove it
8282

83-
error: aborting due to 10 previous errors
83+
error: `assert!(true)` will be optimized out by the compiler
84+
--> tests/ui/assertions_on_constants.rs:52:5
85+
|
86+
LL | assert!(8 == (7 + 1));
87+
| ^^^^^^^^^^^^^^^^^^^^^
88+
|
89+
= help: remove it
90+
91+
error: `assert!(true)` will be optimized out by the compiler
92+
--> tests/ui/assertions_on_constants.rs:61:5
93+
|
94+
LL | assert!(true);
95+
| ^^^^^^^^^^^^^
96+
|
97+
= help: remove it
98+
99+
error: aborting due to 12 previous errors
84100

tests/ui/unnecessary_operation.fixed

+13-1
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,7 @@ fn get_number() -> i32 {
4343
0
4444
}
4545

46-
fn get_usize() -> usize {
46+
const fn get_usize() -> usize {
4747
0
4848
}
4949
fn get_struct() -> Struct {
@@ -113,4 +113,16 @@ fn main() {
113113
'label: {
114114
break 'label
115115
};
116+
let () = const {
117+
[42, 55][get_usize()];
118+
};
119+
}
120+
121+
const _: () = {
122+
[42, 55][get_usize()];
123+
};
124+
125+
const fn foo() {
126+
assert!([42, 55].len() > get_usize());
127+
//~^ ERROR: unnecessary operation
116128
}

tests/ui/unnecessary_operation.rs

+13-1
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,7 @@ fn get_number() -> i32 {
4343
0
4444
}
4545

46-
fn get_usize() -> usize {
46+
const fn get_usize() -> usize {
4747
0
4848
}
4949
fn get_struct() -> Struct {
@@ -117,4 +117,16 @@ fn main() {
117117
'label: {
118118
break 'label
119119
};
120+
let () = const {
121+
[42, 55][get_usize()];
122+
};
123+
}
124+
125+
const _: () = {
126+
[42, 55][get_usize()];
127+
};
128+
129+
const fn foo() {
130+
[42, 55][get_usize()];
131+
//~^ ERROR: unnecessary operation
120132
}

tests/ui/unnecessary_operation.stderr

+7-1
Original file line numberDiff line numberDiff line change
@@ -119,5 +119,11 @@ LL | | s: String::from("blah"),
119119
LL | | };
120120
| |______^ help: statement can be reduced to: `String::from("blah");`
121121

122-
error: aborting due to 19 previous errors
122+
error: unnecessary operation
123+
--> tests/ui/unnecessary_operation.rs:130:5
124+
|
125+
LL | [42, 55][get_usize()];
126+
| ^^^^^^^^^^^^^^^^^^^^^^ help: statement can be written as: `assert!([42, 55].len() > get_usize());`
127+
128+
error: aborting due to 20 previous errors
123129

0 commit comments

Comments
 (0)