|
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;
|
@@ -55,3 +56,24 @@ pub(super) fn check<'tcx>(
|
55 | 56 | }
|
56 | 57 | false
|
57 | 58 | }
|
| 59 | + |
| 60 | +pub(super) fn check_implicit_cast(cx: &LateContext<'_>, expr: &Expr<'_>) { |
| 61 | + if !expr.span.from_expansion() |
| 62 | + && let ExprKind::AddrOf(BorrowKind::Ref, _, target) = expr.kind |
| 63 | + && cx.typeck_results().expr_ty(expr).is_ref() |
| 64 | + && !matches!(get_parent_expr(cx, expr).map(|e| e.kind), Some(ExprKind::Cast(..))) |
| 65 | + && !is_expr_temporary_value(cx, target) |
| 66 | + && let [deref, borrow] = cx.typeck_results().expr_adjustments(expr) |
| 67 | + && matches!(deref.kind, Adjust::Deref(..)) |
| 68 | + && let Adjust::Borrow(AutoBorrow::RawPtr(mutability)) = borrow.kind |
| 69 | + { |
| 70 | + span_lint_and_then(cx, BORROW_AS_PTR, expr.span, "implicit borrow as raw pointer", |diag| { |
| 71 | + diag.span_suggestion_verbose( |
| 72 | + expr.span.until(target.span), |
| 73 | + "use a raw pointer instead", |
| 74 | + format!("&raw {} ", mutability.ptr_str()), |
| 75 | + Applicability::MachineApplicable, |
| 76 | + ); |
| 77 | + }); |
| 78 | + } |
| 79 | +} |
0 commit comments