Skip to content

Commit 6032700

Browse files
committed
Auto merge of #109944 - cjgillot:lint-cstring-macro, r=Nilstrieb
Do not suppress temporary_cstring_as_ptr in macros. There isn't really a reason to skip the lint when part of the expression comes from an expansion. Fixes #94694
2 parents 700938c + 8528ac6 commit 6032700

File tree

3 files changed

+33
-43
lines changed

3 files changed

+33
-43
lines changed

compiler/rustc_lint/src/methods.rs

+8-41
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,9 @@ use crate::lints::CStringPtr;
22
use crate::LateContext;
33
use crate::LateLintPass;
44
use crate::LintContext;
5-
use rustc_hir::{Expr, ExprKind, PathSegment};
5+
use rustc_hir::{Expr, ExprKind};
66
use rustc_middle::ty;
7-
use rustc_span::{symbol::sym, ExpnKind, Span};
7+
use rustc_span::{symbol::sym, Span};
88

99
declare_lint! {
1010
/// The `temporary_cstring_as_ptr` lint detects getting the inner pointer of
@@ -34,47 +34,14 @@ declare_lint! {
3434

3535
declare_lint_pass!(TemporaryCStringAsPtr => [TEMPORARY_CSTRING_AS_PTR]);
3636

37-
fn in_macro(span: Span) -> bool {
38-
if span.from_expansion() {
39-
!matches!(span.ctxt().outer_expn_data().kind, ExpnKind::Desugaring(..))
40-
} else {
41-
false
42-
}
43-
}
44-
45-
fn first_method_call<'tcx>(
46-
expr: &'tcx Expr<'tcx>,
47-
) -> Option<(&'tcx PathSegment<'tcx>, &'tcx Expr<'tcx>)> {
48-
if let ExprKind::MethodCall(path, receiver, args, ..) = &expr.kind {
49-
if args.iter().any(|e| e.span.from_expansion()) || receiver.span.from_expansion() {
50-
None
51-
} else {
52-
Some((path, *receiver))
53-
}
54-
} else {
55-
None
56-
}
57-
}
58-
5937
impl<'tcx> LateLintPass<'tcx> for TemporaryCStringAsPtr {
6038
fn check_expr(&mut self, cx: &LateContext<'tcx>, expr: &'tcx Expr<'_>) {
61-
if in_macro(expr.span) {
62-
return;
63-
}
64-
65-
match first_method_call(expr) {
66-
Some((path, unwrap_arg)) if path.ident.name == sym::as_ptr => {
67-
let as_ptr_span = path.ident.span;
68-
match first_method_call(unwrap_arg) {
69-
Some((path, receiver))
70-
if path.ident.name == sym::unwrap || path.ident.name == sym::expect =>
71-
{
72-
lint_cstring_as_ptr(cx, as_ptr_span, receiver, unwrap_arg);
73-
}
74-
_ => return,
75-
}
76-
}
77-
_ => return,
39+
if let ExprKind::MethodCall(as_ptr_path, as_ptr_receiver, ..) = expr.kind
40+
&& as_ptr_path.ident.name == sym::as_ptr
41+
&& let ExprKind::MethodCall(unwrap_path, unwrap_receiver, ..) = as_ptr_receiver.kind
42+
&& (unwrap_path.ident.name == sym::unwrap || unwrap_path.ident.name == sym::expect)
43+
{
44+
lint_cstring_as_ptr(cx, as_ptr_path.ident.span, unwrap_receiver, as_ptr_receiver);
7845
}
7946
}
8047
}

tests/ui/lint/lint-temporary-cstring-as-ptr.rs

+8
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,15 @@
33

44
use std::ffi::CString;
55

6+
macro_rules! mymacro {
7+
() => {
8+
let s = CString::new("some text").unwrap().as_ptr();
9+
//~^ ERROR getting the inner pointer of a temporary `CString`
10+
}
11+
}
12+
613
fn main() {
714
let s = CString::new("some text").unwrap().as_ptr();
815
//~^ ERROR getting the inner pointer of a temporary `CString`
16+
mymacro!();
917
}

tests/ui/lint/lint-temporary-cstring-as-ptr.stderr

+17-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
error: getting the inner pointer of a temporary `CString`
2-
--> $DIR/lint-temporary-cstring-as-ptr.rs:7:48
2+
--> $DIR/lint-temporary-cstring-as-ptr.rs:14:48
33
|
44
LL | let s = CString::new("some text").unwrap().as_ptr();
55
| ---------------------------------- ^^^^^^ this pointer will be invalid
@@ -14,5 +14,20 @@ note: the lint level is defined here
1414
LL | #![deny(temporary_cstring_as_ptr)]
1515
| ^^^^^^^^^^^^^^^^^^^^^^^^
1616

17-
error: aborting due to previous error
17+
error: getting the inner pointer of a temporary `CString`
18+
--> $DIR/lint-temporary-cstring-as-ptr.rs:8:52
19+
|
20+
LL | let s = CString::new("some text").unwrap().as_ptr();
21+
| ---------------------------------- ^^^^^^ this pointer will be invalid
22+
| |
23+
| this `CString` is deallocated at the end of the statement, bind it to a variable to extend its lifetime
24+
...
25+
LL | mymacro!();
26+
| ---------- in this macro invocation
27+
|
28+
= note: pointers do not have a lifetime; when calling `as_ptr` the `CString` will be deallocated at the end of the statement because nothing is referencing it as far as the type system is concerned
29+
= help: for more information, see https://doc.rust-lang.org/reference/destructors.html
30+
= note: this error originates in the macro `mymacro` (in Nightly builds, run with -Z macro-backtrace for more info)
31+
32+
error: aborting due to 2 previous errors
1833

0 commit comments

Comments
 (0)