bevy_lint/lints/nursery/zst_query.rs
1//! Checks for queries that query the data for a zero-sized type.
2//!
3//! # Motivation
4//!
5//! Zero-sized types (ZSTs) are types that have no size because they contain no runtime data. Any
6//! information they may hold is known at compile-time in the form of [constant generics], which do
7//! not need to be queried. As such, ZSTs are better used as query filters instead of query data.
8//!
9//! [constant generics]: https://doc.rust-lang.org/reference/items/generics.html#const-generics
10//!
11//! # Known Issues
12//!
13//! This lint raises false positives on queries like `Has<T>` and `AnyOf<T>` because they are ZSTs,
14//! even though they still retrieve data from the ECS. Please see [#279] for more information.
15//!
16//! [#279]: https://github.com/TheBevyFlock/bevy_cli/issues/279
17//!
18//! # Example
19//!
20//! ```
21//! # use bevy::prelude::*;
22//! #
23//! // This is a zero-sized type, sometimes known as a "marker component".
24//! #[derive(Component)]
25//! struct Player;
26//!
27//! fn move_player(mut query: Query<(&mut Transform, &Player)>) {
28//! for (transform, _) in query.iter_mut() {
29//! // ...
30//! }
31//! }
32//! #
33//! # assert_eq!(std::mem::size_of::<Player>(), 0);
34//! ```
35//!
36//! Use instead:
37//!
38//! ```
39//! # use bevy::prelude::*;
40//! #
41//! #[derive(Component)]
42//! struct Player;
43//!
44//! fn move_player(mut query: Query<&mut Transform, With<Player>>) {
45//! for transform in query.iter_mut() {
46//! // ...
47//! }
48//! }
49//! #
50//! # assert_eq!(std::mem::size_of::<Player>(), 0);
51//! ```
52
53use crate::{
54 declare_bevy_lint, declare_bevy_lint_pass,
55 utils::hir_parse::{detuple, generic_type_at},
56};
57use clippy_utils::{
58 diagnostics::span_lint_and_help,
59 ty::{is_normalizable, match_type, ty_from_hir_ty},
60};
61use rustc_abi::Size;
62use rustc_hir::AmbigArg;
63use rustc_lint::{LateContext, LateLintPass};
64use rustc_middle::ty::{
65 Ty,
66 layout::{LayoutOf, TyAndLayout},
67};
68
69declare_bevy_lint! {
70 pub ZST_QUERY,
71 // This will eventually be a `RESTRICTION` lint, but due to
72 // <https://github.com/TheBevyFlock/bevy_cli/issues/279> it is not yet ready for production.
73 super::NURSERY,
74 "queried a zero-sized type",
75}
76
77declare_bevy_lint_pass! {
78 pub ZstQuery => [ZST_QUERY.lint],
79}
80
81impl<'tcx> LateLintPass<'tcx> for ZstQuery {
82 fn check_ty(&mut self, cx: &LateContext<'tcx>, hir_ty: &'tcx rustc_hir::Ty<'tcx, AmbigArg>) {
83 if hir_ty.span.in_external_macro(cx.tcx.sess.source_map()) {
84 return;
85 }
86 let ty = ty_from_hir_ty(cx, hir_ty.as_unambig_ty());
87
88 let Some(query_kind) = QueryKind::try_from_ty(cx, ty) else {
89 return;
90 };
91
92 let Some(query_data_ty) = generic_type_at(cx, hir_ty.as_unambig_ty(), 2) else {
93 return;
94 };
95
96 for hir_ty in detuple(*query_data_ty) {
97 let ty = ty_from_hir_ty(cx, &hir_ty);
98
99 // We want to make sure we're evaluating `Foo` and not `&Foo`/`&mut Foo`
100 let peeled = ty.peel_refs();
101
102 if !is_zero_sized(cx, peeled).unwrap_or_default() {
103 continue;
104 }
105
106 // TODO: We can also special case `Option<&Foo>`/`Option<&mut Foo>` to
107 // instead suggest `Has<Foo>`
108 span_lint_and_help(
109 cx,
110 ZST_QUERY.lint,
111 hir_ty.span,
112 ZST_QUERY.lint.desc,
113 None,
114 query_kind.help(peeled),
115 );
116 }
117 }
118}
119
120enum QueryKind {
121 Query,
122}
123
124impl QueryKind {
125 fn try_from_ty<'tcx>(cx: &LateContext<'tcx>, ty: Ty<'tcx>) -> Option<Self> {
126 if match_type(cx, ty, &crate::paths::QUERY) {
127 Some(Self::Query)
128 } else {
129 None
130 }
131 }
132
133 fn help(&self, ty: Ty<'_>) -> String {
134 // It should be noted that `With<Foo>` is not always the best filter to suggest.
135 // While it's most often going to be what users want, there's also `Added<Foo>`
136 // and `Changed<Foo>` which might be more appropriate in some cases
137 // (i.e. users are calling `foo.is_added()` or `foo.is_changed()` in the body of
138 // the system).
139 // In the future, we might want to span the usage site of `is_added()`/`is_changed()`
140 // and suggest to use `Added<Foo>`/`Changed<Foo>` instead.
141 match self {
142 Self::Query => format!(
143 // NOTE: This isn't actually true, please see #279 for more info and how this will
144 // be fixed!
145 "zero-sized types do not retrieve any data, consider using a filter instead: `With<{ty}>`"
146 ),
147 }
148 }
149}
150
151/// Checks if a type is zero-sized.
152///
153/// Returns:
154/// - `Some(true)` if the type is most likely a ZST
155/// - `Some(false)` if the type is most likely not a ZST
156/// - `None` if we cannot determine the size (e.g., type is not normalizable)
157fn is_zero_sized<'tcx>(cx: &LateContext<'tcx>, ty: Ty<'tcx>) -> Option<bool> {
158 // `cx.layout_of()` panics if the type is not normalizable.
159 if !is_normalizable(cx, cx.param_env, ty) {
160 return None;
161 }
162
163 // Note: we don't use `approx_ty_size` from `clippy_utils` here
164 // because it will return `0` as the default value if the type is not
165 // normalizable, which will put us at risk of emitting more false positives.
166 if let Ok(TyAndLayout { layout, .. }) = cx.layout_of(ty) {
167 Some(layout.size() == Size::ZERO)
168 } else {
169 None
170 }
171}