Skip to content

Commit a9aff03

Browse files
committed
Make borrow_as_ptr flag implicit casts as well
1 parent 4d0d6f5 commit a9aff03

File tree

6 files changed

+98
-4
lines changed

6 files changed

+98
-4
lines changed

clippy_lints/src/casts/borrow_as_ptr.rs

+24-2
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,12 @@
1-
use clippy_utils::diagnostics::span_lint_and_sugg;
1+
use clippy_utils::diagnostics::{span_lint_and_sugg, span_lint_and_then};
22
use clippy_utils::msrvs::Msrv;
33
use clippy_utils::source::{snippet_with_applicability, snippet_with_context};
44
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};
66
use rustc_errors::Applicability;
77
use rustc_hir::{BorrowKind, Expr, ExprKind, Mutability, Ty, TyKind};
88
use rustc_lint::LateContext;
9+
use rustc_middle::ty::adjustment::{Adjust, AutoBorrow};
910
use rustc_span::BytePos;
1011

1112
use super::BORROW_AS_PTR;
@@ -55,3 +56,24 @@ pub(super) fn check<'tcx>(
5556
}
5657
false
5758
}
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+
}

clippy_lints/src/casts/mod.rs

+3
Original file line numberDiff line numberDiff line change
@@ -846,6 +846,9 @@ impl<'tcx> LateLintPass<'tcx> for Casts {
846846
}
847847
}
848848

849+
if self.msrv.meets(cx, msrvs::RAW_REF_OP) {
850+
borrow_as_ptr::check_implicit_cast(cx, expr);
851+
}
849852
cast_ptr_alignment::check(cx, expr);
850853
char_lit_as_u8::check(cx, expr);
851854
ptr_as_ptr::check(cx, expr, self.msrv);

clippy_lints/src/non_copy_const.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -449,7 +449,7 @@ impl<'tcx> LateLintPass<'tcx> for NonCopyConst<'tcx> {
449449

450450
dereferenced_expr = parent_expr;
451451
},
452-
ExprKind::Index(e, _, _) if ptr::eq(&**e, cur_expr) => {
452+
ExprKind::Index(e, _, _) if ptr::eq(&raw const **e, cur_expr) => {
453453
// `e[i]` => desugared to `*Index::index(&e, i)`,
454454
// meaning `e` must be referenced.
455455
// no need to go further up since a method call is involved now.

tests/ui/borrow_as_ptr.fixed

+18
Original file line numberDiff line numberDiff line change
@@ -29,3 +29,21 @@ fn issue_13882() {
2929
let _raw = (&raw mut x[1]).wrapping_offset(-1);
3030
//~^ borrow_as_ptr
3131
}
32+
33+
fn implicit_cast() {
34+
let val = 1;
35+
let p: *const i32 = &raw const val;
36+
//~^ borrow_as_ptr
37+
38+
let mut val = 1;
39+
let p: *mut i32 = &raw mut val;
40+
//~^ borrow_as_ptr
41+
42+
let mut val = 1;
43+
// Only lint the leftmost argument, the rightmost is ref to a temporary
44+
core::ptr::eq(&raw const val, &1);
45+
//~^ borrow_as_ptr
46+
47+
// Do not lint references to temporaries
48+
core::ptr::eq(&0i32, &1i32);
49+
}

tests/ui/borrow_as_ptr.rs

+18
Original file line numberDiff line numberDiff line change
@@ -29,3 +29,21 @@ fn issue_13882() {
2929
let _raw = (&mut x[1] as *mut i32).wrapping_offset(-1);
3030
//~^ borrow_as_ptr
3131
}
32+
33+
fn implicit_cast() {
34+
let val = 1;
35+
let p: *const i32 = &val;
36+
//~^ borrow_as_ptr
37+
38+
let mut val = 1;
39+
let p: *mut i32 = &mut val;
40+
//~^ borrow_as_ptr
41+
42+
let mut val = 1;
43+
// Only lint the leftmost argument, the rightmost is ref to a temporary
44+
core::ptr::eq(&val, &1);
45+
//~^ borrow_as_ptr
46+
47+
// Do not lint references to temporaries
48+
core::ptr::eq(&0i32, &1i32);
49+
}

tests/ui/borrow_as_ptr.stderr

+34-1
Original file line numberDiff line numberDiff line change
@@ -25,5 +25,38 @@ error: borrow as raw pointer
2525
LL | let _raw = (&mut x[1] as *mut i32).wrapping_offset(-1);
2626
| ^^^^^^^^^^^^^^^^^^^^^ help: try: `&raw mut x[1]`
2727

28-
error: aborting due to 4 previous errors
28+
error: implicit borrow as raw pointer
29+
--> tests/ui/borrow_as_ptr.rs:35:25
30+
|
31+
LL | let p: *const i32 = &val;
32+
| ^^^^
33+
|
34+
help: use a raw pointer instead
35+
|
36+
LL | let p: *const i32 = &raw const val;
37+
| +++++++++
38+
39+
error: implicit borrow as raw pointer
40+
--> tests/ui/borrow_as_ptr.rs:39:23
41+
|
42+
LL | let p: *mut i32 = &mut val;
43+
| ^^^^^^^^
44+
|
45+
help: use a raw pointer instead
46+
|
47+
LL | let p: *mut i32 = &raw mut val;
48+
| +++
49+
50+
error: implicit borrow as raw pointer
51+
--> tests/ui/borrow_as_ptr.rs:44:19
52+
|
53+
LL | core::ptr::eq(&val, &1);
54+
| ^^^^
55+
|
56+
help: use a raw pointer instead
57+
|
58+
LL | core::ptr::eq(&raw const val, &1);
59+
| +++++++++
60+
61+
error: aborting due to 7 previous errors
2962

0 commit comments

Comments
 (0)