Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
bevyengine
GitHub Repository: bevyengine/bevy
Path: blob/main/crates/bevy_render/src/extract_param.rs
6595 views
1
use crate::MainWorld;
2
use bevy_ecs::{
3
component::Tick,
4
prelude::*,
5
query::FilteredAccessSet,
6
system::{
7
ReadOnlySystemParam, SystemMeta, SystemParam, SystemParamItem, SystemParamValidationError,
8
SystemState,
9
},
10
world::unsafe_world_cell::UnsafeWorldCell,
11
};
12
use core::ops::{Deref, DerefMut};
13
14
/// A helper for accessing [`MainWorld`] content using a system parameter.
15
///
16
/// A [`SystemParam`] adapter which applies the contained `SystemParam` to the [`World`]
17
/// contained in [`MainWorld`]. This parameter only works for systems run
18
/// during the [`ExtractSchedule`](crate::ExtractSchedule).
19
///
20
/// This requires that the contained [`SystemParam`] does not mutate the world, as it
21
/// uses a read-only reference to [`MainWorld`] internally.
22
///
23
/// ## Context
24
///
25
/// [`ExtractSchedule`] is used to extract (move) data from the simulation world ([`MainWorld`]) to the
26
/// render world. The render world drives rendering each frame (generally to a `Window`).
27
/// This design is used to allow performing calculations related to rendering a prior frame at the same
28
/// time as the next frame is simulated, which increases throughput (FPS).
29
///
30
/// [`Extract`] is used to get data from the main world during [`ExtractSchedule`].
31
///
32
/// ## Examples
33
///
34
/// ```
35
/// use bevy_ecs::prelude::*;
36
/// use bevy_render::Extract;
37
/// use bevy_render::sync_world::RenderEntity;
38
/// # #[derive(Component)]
39
/// // Do make sure to sync the cloud entities before extracting them.
40
/// # struct Cloud;
41
/// fn extract_clouds(mut commands: Commands, clouds: Extract<Query<RenderEntity, With<Cloud>>>) {
42
/// for cloud in &clouds {
43
/// commands.entity(cloud).insert(Cloud);
44
/// }
45
/// }
46
/// ```
47
///
48
/// [`ExtractSchedule`]: crate::ExtractSchedule
49
/// [Window]: bevy_window::Window
50
pub struct Extract<'w, 's, P>
51
where
52
P: ReadOnlySystemParam + 'static,
53
{
54
item: SystemParamItem<'w, 's, P>,
55
}
56
57
#[doc(hidden)]
58
pub struct ExtractState<P: SystemParam + 'static> {
59
state: SystemState<P>,
60
main_world_state: <Res<'static, MainWorld> as SystemParam>::State,
61
}
62
63
// SAFETY: The only `World` access (`Res<MainWorld>`) is read-only.
64
unsafe impl<P> ReadOnlySystemParam for Extract<'_, '_, P> where P: ReadOnlySystemParam {}
65
66
// SAFETY: The only `World` access is properly registered by `Res<MainWorld>::init_state`.
67
// This call will also ensure that there are no conflicts with prior params.
68
unsafe impl<P> SystemParam for Extract<'_, '_, P>
69
where
70
P: ReadOnlySystemParam,
71
{
72
type State = ExtractState<P>;
73
type Item<'w, 's> = Extract<'w, 's, P>;
74
75
fn init_state(world: &mut World) -> Self::State {
76
let mut main_world = world.resource_mut::<MainWorld>();
77
ExtractState {
78
state: SystemState::new(&mut main_world),
79
main_world_state: Res::<MainWorld>::init_state(world),
80
}
81
}
82
83
fn init_access(
84
state: &Self::State,
85
system_meta: &mut SystemMeta,
86
component_access_set: &mut FilteredAccessSet,
87
world: &mut World,
88
) {
89
Res::<MainWorld>::init_access(
90
&state.main_world_state,
91
system_meta,
92
component_access_set,
93
world,
94
);
95
}
96
97
#[inline]
98
unsafe fn validate_param(
99
state: &mut Self::State,
100
_system_meta: &SystemMeta,
101
world: UnsafeWorldCell,
102
) -> Result<(), SystemParamValidationError> {
103
// SAFETY: Read-only access to world data registered in `init_state`.
104
let result = unsafe { world.get_resource_by_id(state.main_world_state) };
105
let Some(main_world) = result else {
106
return Err(SystemParamValidationError::invalid::<Self>(
107
"`MainWorld` resource does not exist",
108
));
109
};
110
// SAFETY: Type is guaranteed by `SystemState`.
111
let main_world: &World = unsafe { main_world.deref() };
112
// SAFETY: We provide the main world on which this system state was initialized on.
113
unsafe {
114
SystemState::<P>::validate_param(
115
&mut state.state,
116
main_world.as_unsafe_world_cell_readonly(),
117
)
118
}
119
}
120
121
#[inline]
122
unsafe fn get_param<'w, 's>(
123
state: &'s mut Self::State,
124
system_meta: &SystemMeta,
125
world: UnsafeWorldCell<'w>,
126
change_tick: Tick,
127
) -> Self::Item<'w, 's> {
128
// SAFETY:
129
// - The caller ensures that `world` is the same one that `init_state` was called with.
130
// - The caller ensures that no other `SystemParam`s will conflict with the accesses we have registered.
131
let main_world = unsafe {
132
Res::<MainWorld>::get_param(
133
&mut state.main_world_state,
134
system_meta,
135
world,
136
change_tick,
137
)
138
};
139
let item = state.state.get(main_world.into_inner());
140
Extract { item }
141
}
142
}
143
144
impl<'w, 's, P> Deref for Extract<'w, 's, P>
145
where
146
P: ReadOnlySystemParam,
147
{
148
type Target = SystemParamItem<'w, 's, P>;
149
150
#[inline]
151
fn deref(&self) -> &Self::Target {
152
&self.item
153
}
154
}
155
156
impl<'w, 's, P> DerefMut for Extract<'w, 's, P>
157
where
158
P: ReadOnlySystemParam,
159
{
160
#[inline]
161
fn deref_mut(&mut self) -> &mut Self::Target {
162
&mut self.item
163
}
164
}
165
166
impl<'a, 'w, 's, P> IntoIterator for &'a Extract<'w, 's, P>
167
where
168
P: ReadOnlySystemParam,
169
&'a SystemParamItem<'w, 's, P>: IntoIterator,
170
{
171
type Item = <&'a SystemParamItem<'w, 's, P> as IntoIterator>::Item;
172
type IntoIter = <&'a SystemParamItem<'w, 's, P> as IntoIterator>::IntoIter;
173
174
fn into_iter(self) -> Self::IntoIter {
175
(&self.item).into_iter()
176
}
177
}
178
179