Path: blob/main/crates/bevy_ecs/macros/src/query_data.rs
6600 views
use bevy_macro_utils::ensure_no_collision;1use proc_macro::TokenStream;2use proc_macro2::{Ident, Span};3use quote::{format_ident, quote};4use syn::{5parse_macro_input, parse_quote, punctuated::Punctuated, token, token::Comma, Attribute, Data,6DataStruct, DeriveInput, Field, Index, Meta,7};89use crate::{10bevy_ecs_path,11world_query::{item_struct, world_query_impl},12};1314#[derive(Default)]15struct QueryDataAttributes {16pub is_mutable: bool,1718pub derive_args: Punctuated<Meta, Comma>,19}2021static MUTABLE_ATTRIBUTE_NAME: &str = "mutable";22static DERIVE_ATTRIBUTE_NAME: &str = "derive";2324mod field_attr_keywords {25syn::custom_keyword!(ignore);26}2728pub static QUERY_DATA_ATTRIBUTE_NAME: &str = "query_data";2930pub fn derive_query_data_impl(input: TokenStream) -> TokenStream {31let tokens = input.clone();3233let ast = parse_macro_input!(input as DeriveInput);34let visibility = ast.vis;3536let mut attributes = QueryDataAttributes::default();37for attr in &ast.attrs {38if attr39.path()40.get_ident()41.is_none_or(|ident| ident != QUERY_DATA_ATTRIBUTE_NAME)42{43continue;44}4546let result = attr.parse_nested_meta(|meta| {47if meta.path.is_ident(MUTABLE_ATTRIBUTE_NAME) {48attributes.is_mutable = true;49if meta.input.peek(token::Paren) {50Err(meta.error(format_args!("`{MUTABLE_ATTRIBUTE_NAME}` does not take any arguments")))51} else {52Ok(())53}54} else if meta.path.is_ident(DERIVE_ATTRIBUTE_NAME) {55meta.parse_nested_meta(|meta| {56attributes.derive_args.push(Meta::Path(meta.path));57Ok(())58}).map_err(|_| {59meta.error(format_args!("`{DERIVE_ATTRIBUTE_NAME}` requires at least one argument"))60})61} else {62Err(meta.error(format_args!("invalid attribute, expected `{MUTABLE_ATTRIBUTE_NAME}` or `{DERIVE_ATTRIBUTE_NAME}`")))63}64});6566if let Err(err) = result {67return err.to_compile_error().into();68}69}7071let path = bevy_ecs_path();7273let user_generics = ast.generics.clone();74let (user_impl_generics, user_ty_generics, user_where_clauses) = user_generics.split_for_impl();75let user_generics_with_world = {76let mut generics = ast.generics.clone();77generics.params.insert(0, parse_quote!('__w));78generics79};80let (user_impl_generics_with_world, user_ty_generics_with_world, user_where_clauses_with_world) =81user_generics_with_world.split_for_impl();82let user_generics_with_world_and_state = {83let mut generics = ast.generics;84generics.params.insert(0, parse_quote!('__w));85generics.params.insert(1, parse_quote!('__s));86generics87};88let (89user_impl_generics_with_world_and_state,90user_ty_generics_with_world_and_state,91user_where_clauses_with_world_and_state,92) = user_generics_with_world_and_state.split_for_impl();9394let struct_name = ast.ident;95let read_only_struct_name = if attributes.is_mutable {96Ident::new(&format!("{struct_name}ReadOnly"), Span::call_site())97} else {98struct_name.clone()99};100101let item_struct_name = Ident::new(&format!("{struct_name}Item"), Span::call_site());102let read_only_item_struct_name = if attributes.is_mutable {103Ident::new(&format!("{struct_name}ReadOnlyItem"), Span::call_site())104} else {105item_struct_name.clone()106};107108let fetch_struct_name = Ident::new(&format!("{struct_name}Fetch"), Span::call_site());109let fetch_struct_name = ensure_no_collision(fetch_struct_name, tokens.clone());110let read_only_fetch_struct_name = if attributes.is_mutable {111let new_ident = Ident::new(&format!("{struct_name}ReadOnlyFetch"), Span::call_site());112ensure_no_collision(new_ident, tokens.clone())113} else {114fetch_struct_name.clone()115};116117let marker_name =118ensure_no_collision(format_ident!("_world_query_derive_marker"), tokens.clone());119120// Generate a name for the state struct that doesn't conflict121// with the struct definition.122let state_struct_name = Ident::new(&format!("{struct_name}State"), Span::call_site());123let state_struct_name = ensure_no_collision(state_struct_name, tokens);124125let Data::Struct(DataStruct { fields, .. }) = &ast.data else {126return syn::Error::new(127Span::call_site(),128"#[derive(QueryData)]` only supports structs",129)130.into_compile_error()131.into();132};133134let mut field_attrs = Vec::new();135let mut field_visibilities = Vec::new();136let mut field_idents = Vec::new();137let mut named_field_idents = Vec::new();138let mut field_types = Vec::new();139let mut read_only_field_types = Vec::new();140for (i, field) in fields.iter().enumerate() {141let attrs = match read_world_query_field_info(field) {142Ok(QueryDataFieldInfo { attrs }) => attrs,143Err(e) => return e.into_compile_error().into(),144};145146let named_field_ident = field147.ident148.as_ref()149.cloned()150.unwrap_or_else(|| format_ident!("f{i}"));151let i = Index::from(i);152let field_ident = field153.ident154.as_ref()155.map_or(quote! { #i }, |i| quote! { #i });156field_idents.push(field_ident);157named_field_idents.push(named_field_ident);158field_attrs.push(attrs);159field_visibilities.push(field.vis.clone());160let field_ty = field.ty.clone();161field_types.push(quote!(#field_ty));162read_only_field_types.push(quote!(<#field_ty as #path::query::QueryData>::ReadOnly));163}164165let derive_args = &attributes.derive_args;166// `#[derive()]` is valid syntax167let derive_macro_call = quote! { #[derive(#derive_args)] };168169let mutable_item_struct = item_struct(170&path,171fields,172&derive_macro_call,173&struct_name,174&visibility,175&item_struct_name,176&field_types,177&user_impl_generics_with_world_and_state,178&field_attrs,179&field_visibilities,180&field_idents,181&user_ty_generics,182&user_ty_generics_with_world_and_state,183user_where_clauses_with_world_and_state,184);185let mutable_world_query_impl = world_query_impl(186&path,187&struct_name,188&visibility,189&fetch_struct_name,190&field_types,191&user_impl_generics,192&user_impl_generics_with_world,193&user_ty_generics,194&user_ty_generics_with_world,195&named_field_idents,196&marker_name,197&state_struct_name,198user_where_clauses,199user_where_clauses_with_world,200);201202let (read_only_struct, read_only_impl) = if attributes.is_mutable {203// If the query is mutable, we need to generate a separate readonly version of some things204let readonly_item_struct = item_struct(205&path,206fields,207&derive_macro_call,208&read_only_struct_name,209&visibility,210&read_only_item_struct_name,211&read_only_field_types,212&user_impl_generics_with_world_and_state,213&field_attrs,214&field_visibilities,215&field_idents,216&user_ty_generics,217&user_ty_generics_with_world_and_state,218user_where_clauses_with_world_and_state,219);220let readonly_world_query_impl = world_query_impl(221&path,222&read_only_struct_name,223&visibility,224&read_only_fetch_struct_name,225&read_only_field_types,226&user_impl_generics,227&user_impl_generics_with_world,228&user_ty_generics,229&user_ty_generics_with_world,230&named_field_idents,231&marker_name,232&state_struct_name,233user_where_clauses,234user_where_clauses_with_world,235);236let read_only_structs = quote! {237#[doc = concat!(238"Automatically generated [`WorldQuery`](",239stringify!(#path),240"::query::WorldQuery) type for a read-only variant of [`",241stringify!(#struct_name),242"`]."243)]244#[automatically_derived]245#visibility struct #read_only_struct_name #user_impl_generics #user_where_clauses {246#(247#[doc = "Automatically generated read-only field for accessing `"]248#[doc = stringify!(#field_types)]249#[doc = "`."]250#field_visibilities #named_field_idents: #read_only_field_types,251)*252}253254#readonly_item_struct255};256(read_only_structs, readonly_world_query_impl)257} else {258(quote! {}, quote! {})259};260261let data_impl = {262let read_only_data_impl = if attributes.is_mutable {263quote! {264/// SAFETY: we assert fields are readonly below265unsafe impl #user_impl_generics #path::query::QueryData266for #read_only_struct_name #user_ty_generics #user_where_clauses {267const IS_READ_ONLY: bool = true;268type ReadOnly = #read_only_struct_name #user_ty_generics;269type Item<'__w, '__s> = #read_only_item_struct_name #user_ty_generics_with_world_and_state;270271fn shrink<'__wlong: '__wshort, '__wshort, '__s>(272item: Self::Item<'__wlong, '__s>273) -> Self::Item<'__wshort, '__s> {274#read_only_item_struct_name {275#(276#field_idents: <#read_only_field_types>::shrink(item.#field_idents),277)*278}279}280281fn provide_extra_access(282state: &mut Self::State,283access: &mut #path::query::Access,284available_access: &#path::query::Access,285) {286#(<#field_types>::provide_extra_access(&mut state.#named_field_idents, access, available_access);)*287}288289/// SAFETY: we call `fetch` for each member that implements `Fetch`.290#[inline(always)]291unsafe fn fetch<'__w, '__s>(292_state: &'__s Self::State,293_fetch: &mut <Self as #path::query::WorldQuery>::Fetch<'__w>,294_entity: #path::entity::Entity,295_table_row: #path::storage::TableRow,296) -> Self::Item<'__w, '__s> {297Self::Item {298#(#field_idents: <#read_only_field_types>::fetch(&_state.#named_field_idents, &mut _fetch.#named_field_idents, _entity, _table_row),)*299}300}301}302303impl #user_impl_generics #path::query::ReleaseStateQueryData304for #read_only_struct_name #user_ty_generics #user_where_clauses305// Make these HRTBs with an unused lifetime parameter to allow trivial constraints306// See https://github.com/rust-lang/rust/issues/48214307where #(for<'__a> #field_types: #path::query::QueryData<ReadOnly: #path::query::ReleaseStateQueryData>,)* {308fn release_state<'__w>(_item: Self::Item<'__w, '_>) -> Self::Item<'__w, 'static> {309Self::Item {310#(#field_idents: <#read_only_field_types>::release_state(_item.#field_idents),)*311}312}313}314}315} else {316quote! {}317};318319let is_read_only = !attributes.is_mutable;320321quote! {322/// SAFETY: we assert fields are readonly below323unsafe impl #user_impl_generics #path::query::QueryData324for #struct_name #user_ty_generics #user_where_clauses {325const IS_READ_ONLY: bool = #is_read_only;326type ReadOnly = #read_only_struct_name #user_ty_generics;327type Item<'__w, '__s> = #item_struct_name #user_ty_generics_with_world_and_state;328329fn shrink<'__wlong: '__wshort, '__wshort, '__s>(330item: Self::Item<'__wlong, '__s>331) -> Self::Item<'__wshort, '__s> {332#item_struct_name {333#(334#field_idents: <#field_types>::shrink(item.#field_idents),335)*336}337}338339fn provide_extra_access(340state: &mut Self::State,341access: &mut #path::query::Access,342available_access: &#path::query::Access,343) {344#(<#field_types>::provide_extra_access(&mut state.#named_field_idents, access, available_access);)*345}346347/// SAFETY: we call `fetch` for each member that implements `Fetch`.348#[inline(always)]349unsafe fn fetch<'__w, '__s>(350_state: &'__s Self::State,351_fetch: &mut <Self as #path::query::WorldQuery>::Fetch<'__w>,352_entity: #path::entity::Entity,353_table_row: #path::storage::TableRow,354) -> Self::Item<'__w, '__s> {355Self::Item {356#(#field_idents: <#field_types>::fetch(&_state.#named_field_idents, &mut _fetch.#named_field_idents, _entity, _table_row),)*357}358}359}360361impl #user_impl_generics #path::query::ReleaseStateQueryData362for #struct_name #user_ty_generics #user_where_clauses363// Make these HRTBs with an unused lifetime parameter to allow trivial constraints364// See https://github.com/rust-lang/rust/issues/48214365where #(for<'__a> #field_types: #path::query::ReleaseStateQueryData,)* {366fn release_state<'__w>(_item: Self::Item<'__w, '_>) -> Self::Item<'__w, 'static> {367Self::Item {368#(#field_idents: <#field_types>::release_state(_item.#field_idents),)*369}370}371}372373#read_only_data_impl374}375};376377let read_only_data_impl = quote! {378/// SAFETY: we assert fields are readonly below379unsafe impl #user_impl_generics #path::query::ReadOnlyQueryData380for #read_only_struct_name #user_ty_generics #user_where_clauses {}381};382383let read_only_asserts = if attributes.is_mutable {384quote! {385// Double-check that the data fetched by `<_ as WorldQuery>::ReadOnly` is read-only.386// This is technically unnecessary as `<_ as WorldQuery>::ReadOnly: ReadOnlyQueryData`387// but to protect against future mistakes we assert the assoc type implements `ReadOnlyQueryData` anyway388#( assert_readonly::<#read_only_field_types>(); )*389}390} else {391quote! {392// Statically checks that the safety guarantee of `ReadOnlyQueryData` for `$fetch_struct_name` actually holds true.393// We need this to make sure that we don't compile `ReadOnlyQueryData` if our struct contains nested `QueryData`394// members that don't implement it. I.e.:395// ```396// #[derive(QueryData)]397// pub struct Foo { a: &'static mut MyComponent }398// ```399#( assert_readonly::<#field_types>(); )*400}401};402403let data_asserts = quote! {404#( assert_data::<#field_types>(); )*405};406407TokenStream::from(quote! {408#mutable_item_struct409410#read_only_struct411412const _: () = {413#[doc(hidden)]414#[doc = concat!(415"Automatically generated internal [`WorldQuery`](",416stringify!(#path),417"::query::WorldQuery) state type for [`",418stringify!(#struct_name),419"`], used for caching."420)]421#[automatically_derived]422#visibility struct #state_struct_name #user_impl_generics #user_where_clauses {423#(#named_field_idents: <#field_types as #path::query::WorldQuery>::State,)*424}425426#mutable_world_query_impl427428#read_only_impl429430#data_impl431432#read_only_data_impl433};434435#[allow(dead_code)]436const _: () = {437fn assert_readonly<T>()438where439T: #path::query::ReadOnlyQueryData,440{441}442443fn assert_data<T>()444where445T: #path::query::QueryData,446{447}448449// We generate a readonly assertion for every struct member.450fn assert_all #user_impl_generics_with_world () #user_where_clauses_with_world {451#read_only_asserts452#data_asserts453}454};455456// The original struct will most likely be left unused. As we don't want our users having457// to specify `#[allow(dead_code)]` for their custom queries, we are using this cursed458// workaround.459#[allow(dead_code)]460const _: () = {461fn dead_code_workaround #user_impl_generics (462q: #struct_name #user_ty_generics,463q2: #read_only_struct_name #user_ty_generics464) #user_where_clauses {465#(q.#field_idents;)*466#(q2.#field_idents;)*467}468};469})470}471472struct QueryDataFieldInfo {473/// All field attributes except for `query_data` ones.474attrs: Vec<Attribute>,475}476477fn read_world_query_field_info(field: &Field) -> syn::Result<QueryDataFieldInfo> {478let mut attrs = Vec::new();479for attr in &field.attrs {480if attr481.path()482.get_ident()483.is_some_and(|ident| ident == QUERY_DATA_ATTRIBUTE_NAME)484{485return Err(syn::Error::new_spanned(486attr,487"#[derive(QueryData)] does not support field attributes.",488));489}490attrs.push(attr.clone());491}492493Ok(QueryDataFieldInfo { attrs })494}495496497