Skip to content

Commit 61600ab

Browse files
committed
Make borrow_as_ptr flag implicit casts as well
1 parent f85331f commit 61600ab

File tree

6 files changed

+115
-4
lines changed

6 files changed

+115
-4
lines changed

clippy_lints/src/casts/borrow_as_ptr.rs

+32-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;
@@ -59,3 +60,32 @@ pub(super) fn check<'tcx>(
5960
}
6061
false
6162
}
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+
}

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::BORROW_AS_PTR) {
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, &raw const *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

+16
Original file line numberDiff line numberDiff line change
@@ -29,3 +29,19 @@ 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 r = &mut 1;
43+
let p: *const i32 = &raw const *r;
44+
//~^ borrow_as_ptr
45+
let p: *mut i32 = &raw mut *r;
46+
//~^ borrow_as_ptr
47+
}

tests/ui/borrow_as_ptr.rs

+16
Original file line numberDiff line numberDiff line change
@@ -29,3 +29,19 @@ 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 r = &mut 1;
43+
let p: *const i32 = r;
44+
//~^ borrow_as_ptr
45+
let p: *mut i32 = r;
46+
//~^ borrow_as_ptr
47+
}

tests/ui/borrow_as_ptr.stderr

+47-1
Original file line numberDiff line numberDiff line change
@@ -25,5 +25,51 @@ 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 = &val;
37+
LL + let p: *const i32 = &raw const val;
38+
|
39+
40+
error: implicit borrow as raw pointer
41+
--> tests/ui/borrow_as_ptr.rs:39:23
42+
|
43+
LL | let p: *mut i32 = &mut val;
44+
| ^^^^^^^^
45+
|
46+
help: use a raw pointer instead
47+
|
48+
LL - let p: *mut i32 = &mut val;
49+
LL + let p: *mut i32 = &raw mut val;
50+
|
51+
52+
error: implicit borrow as raw pointer
53+
--> tests/ui/borrow_as_ptr.rs:43:25
54+
|
55+
LL | let p: *const i32 = r;
56+
| ^
57+
|
58+
help: reborrow as a raw pointer
59+
|
60+
LL | let p: *const i32 = &raw const *r;
61+
| ++++++++++++
62+
63+
error: implicit borrow as raw pointer
64+
--> tests/ui/borrow_as_ptr.rs:45:23
65+
|
66+
LL | let p: *mut i32 = r;
67+
| ^
68+
|
69+
help: reborrow as a raw pointer
70+
|
71+
LL | let p: *mut i32 = &raw mut *r;
72+
| ++++++++++
73+
74+
error: aborting due to 8 previous errors
2975

0 commit comments

Comments
 (0)