about summary refs log tree commit diff
path: root/compiler/rustc_lint/src/map_unit_fn.rs
blob: 18a947dc1ee062627bdfe0f5efee82572da88875 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
use rustc_hir::{Expr, ExprKind, Stmt, StmtKind};
use rustc_middle::ty::{self};
use rustc_session::{declare_lint, declare_lint_pass};
use rustc_span::sym;

use crate::lints::MappingToUnit;
use crate::{LateContext, LateLintPass, LintContext};

declare_lint! {
    /// The `map_unit_fn` lint checks for `Iterator::map` receive
    /// a callable that returns `()`.
    ///
    /// ### Example
    ///
    /// ```rust
    /// fn foo(items: &mut Vec<u8>) {
    ///     items.sort();
    /// }
    ///
    /// fn main() {
    ///     let mut x: Vec<Vec<u8>> = vec![
    ///         vec![0, 2, 1],
    ///         vec![5, 4, 3],
    ///     ];
    ///     x.iter_mut().map(foo);
    /// }
    /// ```
    ///
    /// {{produces}}
    ///
    /// ### Explanation
    ///
    /// Mapping to `()` is almost always a mistake.
    pub MAP_UNIT_FN,
    Warn,
    "`Iterator::map` call that discard the iterator's values"
}

declare_lint_pass!(MapUnitFn => [MAP_UNIT_FN]);

impl<'tcx> LateLintPass<'tcx> for MapUnitFn {
    fn check_stmt(&mut self, cx: &LateContext<'tcx>, stmt: &Stmt<'_>) {
        let StmtKind::Semi(expr) = stmt.kind else {
            return;
        };
        let ExprKind::MethodCall(path, receiver, [arg], span) = expr.kind else {
            return;
        };
        if path.ident.name != sym::map
            || stmt.span.from_expansion()
            || receiver.span.from_expansion()
            || arg.span.from_expansion()
            || !is_impl_slice(cx, receiver)
            || !cx
                .typeck_results()
                .type_dependent_def_id(expr.hir_id)
                .is_some_and(|id| cx.tcx.is_diagnostic_item(sym::IteratorMap, id))
        {
            return;
        }
        let (id, sig) = match *cx.typeck_results().expr_ty(arg).kind() {
            ty::Closure(id, subs) => (id, subs.as_closure().sig()),
            ty::FnDef(id, _) => (id, cx.tcx.fn_sig(id).skip_binder()),
            _ => return,
        };
        let ret_ty = sig.output().skip_binder();
        if !(ret_ty.is_unit() || ret_ty.is_never()) {
            return;
        }
        cx.emit_span_lint(
            MAP_UNIT_FN,
            span,
            MappingToUnit {
                function_label: cx.tcx.span_of_impl(id).unwrap_or(arg.span),
                argument_label: arg.span,
                map_label: span,
                suggestion: path.ident.span,
            },
        );
    }
}

fn is_impl_slice(cx: &LateContext<'_>, expr: &Expr<'_>) -> bool {
    if let Some(method_id) = cx.typeck_results().type_dependent_def_id(expr.hir_id)
        && let Some(impl_id) = cx.tcx.impl_of_assoc(method_id)
    {
        return cx.tcx.type_of(impl_id).skip_binder().is_slice();
    }
    false
}