|
1 |
| -use clippy_utils::diagnostics::span_lint_and_sugg; |
| 1 | +use clippy_utils::diagnostics::{span_lint_and_sugg, span_lint_and_then}; |
2 | 2 | use clippy_utils::msrvs::Msrv;
|
3 | 3 | use clippy_utils::source::{snippet_with_applicability, snippet_with_context};
|
4 | 4 | use clippy_utils::sugg::has_enclosing_paren;
|
5 |
| -use clippy_utils::{is_expr_temporary_value, is_lint_allowed, msrvs, std_or_core}; |
| 5 | +use clippy_utils::{get_parent_expr, is_expr_temporary_value, is_lint_allowed, msrvs, std_or_core}; |
6 | 6 | use rustc_errors::Applicability;
|
7 | 7 | use rustc_hir::{BorrowKind, Expr, ExprKind, Mutability, Ty, TyKind};
|
8 | 8 | use rustc_lint::LateContext;
|
| 9 | +use rustc_middle::ty::adjustment::{Adjust, AutoBorrow}; |
9 | 10 | use rustc_span::BytePos;
|
10 | 11 |
|
11 | 12 | use super::BORROW_AS_PTR;
|
@@ -59,3 +60,32 @@ pub(super) fn check<'tcx>(
|
59 | 60 | }
|
60 | 61 | false
|
61 | 62 | }
|
| 63 | + |
| 64 | +pub(super) fn check_implicit_cast(cx: &LateContext<'_>, expr: &Expr<'_>) { |
| 65 | + if !expr.span.from_expansion() |
| 66 | + && !matches!(get_parent_expr(cx, expr).map(|e| e.kind), Some(ExprKind::Cast(..))) |
| 67 | + && let [deref, borrow] = cx.typeck_results().expr_adjustments(expr) |
| 68 | + && matches!(deref.kind, Adjust::Deref(..)) |
| 69 | + && cx.typeck_results().expr_ty(expr).is_ref() |
| 70 | + && let Adjust::Borrow(AutoBorrow::RawPtr(mutability)) = borrow.kind |
| 71 | + { |
| 72 | + let mode = if mutability == Mutability::Not { "const" } else { "mut" }; |
| 73 | + span_lint_and_then(cx, BORROW_AS_PTR, expr.span, "implicit borrow as raw pointer", |diag| { |
| 74 | + let (msg, suggs) = if let ExprKind::AddrOf(BorrowKind::Ref, _, target) = expr.kind { |
| 75 | + ( |
| 76 | + "use a raw pointer instead", |
| 77 | + vec![ |
| 78 | + (expr.span.shrink_to_lo(), format!("&raw {mode} ")), |
| 79 | + (expr.span.until(target.span), String::new()), |
| 80 | + ], |
| 81 | + ) |
| 82 | + } else { |
| 83 | + ( |
| 84 | + "reborrow as a raw pointer", |
| 85 | + vec![(expr.span.shrink_to_lo(), format!("&raw {mode} *"))], |
| 86 | + ) |
| 87 | + }; |
| 88 | + diag.multipart_suggestion_verbose(msg, suggs, Applicability::MachineApplicable); |
| 89 | + }); |
| 90 | + } |
| 91 | +} |
0 commit comments