Skip to content

Commit 6a0998d

Browse files
committed
lint plain b"...".as_ptr() outside of CStr constructors
1 parent d66b5ae commit 6a0998d

File tree

7 files changed

+139
-23
lines changed

7 files changed

+139
-23
lines changed

book/src/lint_configuration.md

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -151,6 +151,7 @@ The minimum rust version that the project supports. Defaults to the `rust-versio
151151
* [`manual_try_fold`](https://rust-lang.github.io/rust-clippy/master/index.html#manual_try_fold)
152152
* [`manual_hash_one`](https://rust-lang.github.io/rust-clippy/master/index.html#manual_hash_one)
153153
* [`iter_kv_map`](https://rust-lang.github.io/rust-clippy/master/index.html#iter_kv_map)
154+
* [`manual_c_str_literals`](https://rust-lang.github.io/rust-clippy/master/index.html#manual_c_str_literals)
154155

155156

156157
## `cognitive-complexity-threshold`

clippy_config/src/conf.rs

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -249,7 +249,7 @@ define_Conf! {
249249
///
250250
/// Suppress lints whenever the suggested change would cause breakage for other crates.
251251
(avoid_breaking_exported_api: bool = true),
252-
/// Lint: MANUAL_SPLIT_ONCE, MANUAL_STR_REPEAT, CLONED_INSTEAD_OF_COPIED, REDUNDANT_FIELD_NAMES, OPTION_MAP_UNWRAP_OR, REDUNDANT_STATIC_LIFETIMES, FILTER_MAP_NEXT, CHECKED_CONVERSIONS, MANUAL_RANGE_CONTAINS, USE_SELF, MEM_REPLACE_WITH_DEFAULT, MANUAL_NON_EXHAUSTIVE, OPTION_AS_REF_DEREF, MAP_UNWRAP_OR, MATCH_LIKE_MATCHES_MACRO, MANUAL_STRIP, MISSING_CONST_FOR_FN, UNNESTED_OR_PATTERNS, FROM_OVER_INTO, PTR_AS_PTR, IF_THEN_SOME_ELSE_NONE, APPROX_CONSTANT, DEPRECATED_CFG_ATTR, INDEX_REFUTABLE_SLICE, MAP_CLONE, BORROW_AS_PTR, MANUAL_BITS, ERR_EXPECT, CAST_ABS_TO_UNSIGNED, UNINLINED_FORMAT_ARGS, MANUAL_CLAMP, MANUAL_LET_ELSE, UNCHECKED_DURATION_SUBTRACTION, COLLAPSIBLE_STR_REPLACE, SEEK_FROM_CURRENT, SEEK_REWIND, UNNECESSARY_LAZY_EVALUATIONS, TRANSMUTE_PTR_TO_REF, ALMOST_COMPLETE_RANGE, NEEDLESS_BORROW, DERIVABLE_IMPLS, MANUAL_IS_ASCII_CHECK, MANUAL_REM_EUCLID, MANUAL_RETAIN, TYPE_REPETITION_IN_BOUNDS, TUPLE_ARRAY_CONVERSIONS, MANUAL_TRY_FOLD, MANUAL_HASH_ONE, ITER_KV_MAP.
252+
/// Lint: MANUAL_SPLIT_ONCE, MANUAL_STR_REPEAT, CLONED_INSTEAD_OF_COPIED, REDUNDANT_FIELD_NAMES, OPTION_MAP_UNWRAP_OR, REDUNDANT_STATIC_LIFETIMES, FILTER_MAP_NEXT, CHECKED_CONVERSIONS, MANUAL_RANGE_CONTAINS, USE_SELF, MEM_REPLACE_WITH_DEFAULT, MANUAL_NON_EXHAUSTIVE, OPTION_AS_REF_DEREF, MAP_UNWRAP_OR, MATCH_LIKE_MATCHES_MACRO, MANUAL_STRIP, MISSING_CONST_FOR_FN, UNNESTED_OR_PATTERNS, FROM_OVER_INTO, PTR_AS_PTR, IF_THEN_SOME_ELSE_NONE, APPROX_CONSTANT, DEPRECATED_CFG_ATTR, INDEX_REFUTABLE_SLICE, MAP_CLONE, BORROW_AS_PTR, MANUAL_BITS, ERR_EXPECT, CAST_ABS_TO_UNSIGNED, UNINLINED_FORMAT_ARGS, MANUAL_CLAMP, MANUAL_LET_ELSE, UNCHECKED_DURATION_SUBTRACTION, COLLAPSIBLE_STR_REPLACE, SEEK_FROM_CURRENT, SEEK_REWIND, UNNECESSARY_LAZY_EVALUATIONS, TRANSMUTE_PTR_TO_REF, ALMOST_COMPLETE_RANGE, NEEDLESS_BORROW, DERIVABLE_IMPLS, MANUAL_IS_ASCII_CHECK, MANUAL_REM_EUCLID, MANUAL_RETAIN, TYPE_REPETITION_IN_BOUNDS, TUPLE_ARRAY_CONVERSIONS, MANUAL_TRY_FOLD, MANUAL_HASH_ONE, ITER_KV_MAP, MANUAL_C_STR_LITERALS.
253253
///
254254
/// The minimum rust version that the project supports. Defaults to the `rust-version` field in `Cargo.toml`
255255
#[default_text = ""]

clippy_lints/src/methods/manual_c_str_literals.rs

Lines changed: 93 additions & 15 deletions
Original file line numberDiff line numberDiff line change
@@ -4,24 +4,69 @@ use clippy_utils::get_parent_expr;
44
use clippy_utils::source::snippet;
55
use rustc_ast::{LitKind, StrStyle};
66
use rustc_errors::Applicability;
7-
use rustc_hir::{Expr, ExprKind, QPath, TyKind};
7+
use rustc_hir::{Expr, ExprKind, Node, QPath, TyKind};
88
use rustc_lint::LateContext;
9-
use rustc_span::{sym, Span};
9+
use rustc_span::{sym, Span, Symbol};
1010

1111
use super::MANUAL_C_STR_LITERALS;
1212

13-
pub(super) fn check(cx: &LateContext<'_>, expr: &Expr<'_>, func: &Expr<'_>, args: &[Expr<'_>], msrv: &Msrv) {
13+
/// Checks:
14+
/// - `b"...".as_ptr()`
15+
/// - `b"...".as_ptr().cast()`
16+
/// - `"...".as_ptr()`
17+
/// - `"...".as_ptr().cast()`
18+
///
19+
/// Iff the parent call of `.cast()` isn't `CStr::from_ptr`, to avoid linting twice.
20+
pub(super) fn check_as_ptr<'tcx>(cx: &LateContext<'tcx>, expr: &'tcx Expr<'tcx>, receiver: &'tcx Expr<'tcx>) {
21+
if let ExprKind::Lit(lit) = receiver.kind
22+
&& let LitKind::ByteStr(_, StrStyle::Cooked) | LitKind::Str(_, StrStyle::Cooked) = lit.node
23+
&& let casts_removed = peel_ptr_cast_ancestors(cx, expr)
24+
&& !get_parent_expr(cx, casts_removed).is_some_and(
25+
|parent| matches!(parent.kind, ExprKind::Call(func, _) if is_c_str_function(cx, func).is_some()),
26+
)
27+
&& let Some(sugg) = rewrite_as_cstr(cx, lit.span)
28+
{
29+
span_lint_and_sugg(
30+
cx,
31+
MANUAL_C_STR_LITERALS,
32+
receiver.span,
33+
"manually constructing a nul-terminated string",
34+
r#"use a `c""` literal"#,
35+
sugg,
36+
// an additional cast may be needed, since the type of `CStr::as_ptr` and
37+
// `"".as_ptr()` can differ and is platform dependent
38+
Applicability::MaybeIncorrect,
39+
);
40+
}
41+
}
42+
43+
/// Checks if the callee is a "relevant" `CStr` function considered by this lint.
44+
/// Returns the method name.
45+
fn is_c_str_function(cx: &LateContext<'_>, func: &Expr<'_>) -> Option<Symbol> {
1446
if let ExprKind::Path(QPath::TypeRelative(cstr, fn_name)) = &func.kind
1547
&& let TyKind::Path(QPath::Resolved(_, ty_path)) = &cstr.kind
1648
&& cx.tcx.lang_items().c_str() == ty_path.res.opt_def_id()
49+
{
50+
Some(fn_name.ident.name)
51+
} else {
52+
None
53+
}
54+
}
55+
56+
/// Checks:
57+
/// - `CStr::from_bytes_with_nul(..)`
58+
/// - `CStr::from_bytes_with_nul_unchecked(..)`
59+
/// - `CStr::from_ptr(..)`
60+
pub(super) fn check(cx: &LateContext<'_>, expr: &Expr<'_>, func: &Expr<'_>, args: &[Expr<'_>], msrv: &Msrv) {
61+
if let Some(fn_name) = is_c_str_function(cx, func)
1762
&& let [arg] = args
1863
&& msrv.meets(msrvs::C_STR_LITERALS)
1964
{
20-
match fn_name.ident.name.as_str() {
65+
match fn_name.as_str() {
2166
name @ ("from_bytes_with_nul" | "from_bytes_with_nul_unchecked")
2267
if !arg.span.from_expansion()
2368
&& let ExprKind::Lit(lit) = arg.kind
24-
&& let LitKind::ByteStr(_, StrStyle::Cooked) = lit.node =>
69+
&& let LitKind::ByteStr(_, StrStyle::Cooked) | LitKind::Str(_, StrStyle::Cooked) = lit.node =>
2570
{
2671
check_from_bytes(cx, expr, arg, name);
2772
},
@@ -33,19 +78,20 @@ pub(super) fn check(cx: &LateContext<'_>, expr: &Expr<'_>, func: &Expr<'_>, args
3378

3479
/// Checks `CStr::from_bytes_with_nul(b"foo\0")`
3580
fn check_from_ptr(cx: &LateContext<'_>, expr: &Expr<'_>, arg: &Expr<'_>) {
36-
if let ExprKind::MethodCall(method, lit, [], _) = peel_ptr_cast(arg).kind
81+
if let ExprKind::MethodCall(method, lit, ..) = peel_ptr_cast(arg).kind
3782
&& method.ident.name == sym::as_ptr
3883
&& !lit.span.from_expansion()
3984
&& let ExprKind::Lit(lit) = lit.kind
4085
&& let LitKind::ByteStr(_, StrStyle::Cooked) = lit.node
86+
&& let Some(sugg) = rewrite_as_cstr(cx, lit.span)
4187
{
4288
span_lint_and_sugg(
4389
cx,
4490
MANUAL_C_STR_LITERALS,
4591
expr.span,
4692
"calling `CStr::from_ptr` with a byte string literal",
4793
r#"use a `c""` literal"#,
48-
rewrite_as_cstr(cx, lit.span),
94+
sugg,
4995
Applicability::MachineApplicable,
5096
);
5197
}
@@ -66,24 +112,29 @@ fn check_from_bytes(cx: &LateContext<'_>, expr: &Expr<'_>, arg: &Expr<'_>, metho
66112
(expr.span, Applicability::MaybeIncorrect)
67113
};
68114

115+
let Some(sugg) = rewrite_as_cstr(cx, arg.span) else {
116+
return;
117+
};
118+
69119
span_lint_and_sugg(
70120
cx,
71121
MANUAL_C_STR_LITERALS,
72122
span,
73123
"calling `CStr::new` with a byte string literal",
74124
r#"use a `c""` literal"#,
75-
rewrite_as_cstr(cx, arg.span),
125+
sugg,
76126
applicability,
77127
);
78128
}
79129

80130
/// Rewrites a byte string literal to a c-str literal.
81131
/// `b"foo\0"` -> `c"foo"`
82-
pub fn rewrite_as_cstr(cx: &LateContext<'_>, span: Span) -> String {
132+
///
133+
/// Returns `None` if it doesn't end in a NUL byte.
134+
fn rewrite_as_cstr(cx: &LateContext<'_>, span: Span) -> Option<String> {
83135
let mut sugg = String::from("c") + snippet(cx, span.source_callsite(), "..").trim_start_matches('b');
84136

85137
// NUL byte should always be right before the closing quote.
86-
// (Can rfind ever return `None`?)
87138
if let Some(quote_pos) = sugg.rfind('"') {
88139
// Possible values right before the quote:
89140
// - literal NUL value
@@ -98,17 +149,44 @@ pub fn rewrite_as_cstr(cx: &LateContext<'_>, span: Span) -> String {
98149
else if sugg[..quote_pos].ends_with("\\0") {
99150
sugg.replace_range(quote_pos - 2..quote_pos, "");
100151
}
152+
// No known suffix, so assume it's not a C-string.
153+
else {
154+
return None;
155+
}
101156
}
102157

103-
sugg
158+
Some(sugg)
159+
}
160+
161+
fn get_cast_target<'tcx>(e: &'tcx Expr<'tcx>) -> Option<&'tcx Expr<'tcx>> {
162+
match &e.kind {
163+
ExprKind::MethodCall(method, receiver, [], _) if method.ident.as_str() == "cast" => Some(receiver),
164+
ExprKind::Cast(expr, _) => Some(expr),
165+
_ => None,
166+
}
104167
}
105168

106169
/// `x.cast()` -> `x`
107170
/// `x as *const _` -> `x`
171+
/// `x` -> `x` (returns the same expression for non-cast exprs)
108172
fn peel_ptr_cast<'tcx>(e: &'tcx Expr<'tcx>) -> &'tcx Expr<'tcx> {
109-
match &e.kind {
110-
ExprKind::MethodCall(method, receiver, [], _) if method.ident.as_str() == "cast" => peel_ptr_cast(receiver),
111-
ExprKind::Cast(expr, _) => peel_ptr_cast(expr),
112-
_ => e,
173+
get_cast_target(e).map_or(e, peel_ptr_cast)
174+
}
175+
176+
/// Same as `peel_ptr_cast`, but the other way around, by walking up the ancestor cast expressions:
177+
///
178+
/// `foo(x.cast() as *const _)`
179+
/// ^ given this `x` expression, returns the `foo(...)` expression
180+
fn peel_ptr_cast_ancestors<'tcx>(cx: &LateContext<'tcx>, e: &'tcx Expr<'tcx>) -> &'tcx Expr<'tcx> {
181+
let mut prev = e;
182+
for (_, node) in cx.tcx.hir().parent_iter(e.hir_id) {
183+
if let Node::Expr(e) = node
184+
&& get_cast_target(e).is_some()
185+
{
186+
prev = e;
187+
} else {
188+
break;
189+
}
113190
}
191+
prev
114192
}

clippy_lints/src/methods/mod.rs

Lines changed: 15 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -3755,24 +3755,32 @@ declare_clippy_lint! {
37553755

37563756
declare_clippy_lint! {
37573757
/// ### What it does
3758-
/// Checks for calls to `CStr::from_ptr` and `CStr::from_bytes_with_nul` with byte string literals as arguments.
3758+
/// Checks for the manual creation of C strings (a string with a `NUL` byte at the end), either
3759+
/// through one of the `CStr` constructor functions, or more plainly by calling `.as_ptr()`
3760+
/// on a (byte) string literal.
37593761
///
37603762
/// ### Why is this bad?
37613763
/// This can be written more concisely using `c"str"` literals and is also less error-prone,
3762-
/// because the compiler checks for interior nul bytes.
3764+
/// because the compiler checks for interior `NUL` bytes and the terminating `NUL` byte is inserted automatically.
37633765
///
37643766
/// ### Example
37653767
/// ```no_run
37663768
/// # use std::ffi::CStr;
3767-
/// # fn needs_cstr(_: &CStr) {}
3768-
/// needs_cstr(CStr::from_bytes_with_nul(b":)").unwrap());
3769+
/// # mod libc { pub unsafe fn puts(_: *const i8) {} }
3770+
/// fn needs_cstr(_: &CStr) {}
3771+
///
3772+
/// needs_cstr(CStr::from_bytes_with_nul(b"Hello\0").unwrap());
3773+
/// unsafe { libc::puts("World\0".as_ptr().cast()) }
37693774
/// ```
37703775
/// Use instead:
37713776
/// ```no_run
37723777
/// # #![feature(c_str_literals)]
37733778
/// # use std::ffi::CStr;
3774-
/// # fn needs_cstr(_: &CStr) {}
3775-
/// needs_cstr(c":)");
3779+
/// # mod libc { pub unsafe fn puts(_: *const i8) {} }
3780+
/// fn needs_cstr(_: &CStr) {}
3781+
///
3782+
/// needs_cstr(c"Hello");
3783+
/// unsafe { libc::puts(c"World".as_ptr()) }
37763784
/// ```
37773785
#[clippy::version = "1.76.0"]
37783786
pub MANUAL_C_STR_LITERALS,
@@ -4178,6 +4186,7 @@ impl Methods {
41784186
}
41794187
},
41804188
("as_mut", []) => useless_asref::check(cx, expr, "as_mut", recv),
4189+
("as_ptr", []) => manual_c_str_literals::check_as_ptr(cx, expr, recv),
41814190
("as_ref", []) => useless_asref::check(cx, expr, "as_ref", recv),
41824191
("assume_init", []) => uninit_assumed_init::check(cx, expr, recv),
41834192
("cloned", []) => cloned_instead_of_copied::check(cx, expr, recv, span, &self.msrv),

tests/ui/manual_c_str_literals.fixed

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,11 @@ fn main() {
4343

4444
unsafe { c"foo" };
4545
unsafe { c"foo" };
46+
let _: *const _ = c"foo".as_ptr();
47+
let _: *const _ = c"foo".as_ptr();
48+
let _: *const _ = "foo".as_ptr(); // not a C-string
49+
let _: *const _ = "".as_ptr();
50+
let _: *const _ = c"foo".as_ptr().cast::<i8>();
4651

4752
// Macro cases, don't lint:
4853
cstr!("foo");

tests/ui/manual_c_str_literals.rs

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,11 @@ fn main() {
4343

4444
unsafe { CStr::from_ptr(b"foo\0".as_ptr().cast()) };
4545
unsafe { CStr::from_ptr(b"foo\0".as_ptr() as *const _) };
46+
let _: *const _ = b"foo\0".as_ptr();
47+
let _: *const _ = "foo\0".as_ptr();
48+
let _: *const _ = "foo".as_ptr(); // not a C-string
49+
let _: *const _ = "".as_ptr();
50+
let _: *const _ = b"foo\0".as_ptr().cast::<i8>();
4651

4752
// Macro cases, don't lint:
4853
cstr!("foo");

tests/ui/manual_c_str_literals.stderr

Lines changed: 19 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -43,5 +43,23 @@ error: calling `CStr::from_ptr` with a byte string literal
4343
LL | unsafe { CStr::from_ptr(b"foo\0".as_ptr() as *const _) };
4444
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ help: use a `c""` literal: `c"foo"`
4545

46-
error: aborting due to 7 previous errors
46+
error: manually constructing a nul-terminated string
47+
--> $DIR/manual_c_str_literals.rs:46:23
48+
|
49+
LL | let _: *const _ = b"foo\0".as_ptr();
50+
| ^^^^^^^^ help: use a `c""` literal: `c"foo"`
51+
52+
error: manually constructing a nul-terminated string
53+
--> $DIR/manual_c_str_literals.rs:47:23
54+
|
55+
LL | let _: *const _ = "foo\0".as_ptr();
56+
| ^^^^^^^ help: use a `c""` literal: `c"foo"`
57+
58+
error: manually constructing a nul-terminated string
59+
--> $DIR/manual_c_str_literals.rs:50:23
60+
|
61+
LL | let _: *const _ = b"foo\0".as_ptr().cast::<i8>();
62+
| ^^^^^^^^ help: use a `c""` literal: `c"foo"`
63+
64+
error: aborting due to 10 previous errors
4765

0 commit comments

Comments
 (0)