From d3125662bd55f97cdc10f35857c77d08dc916825 Mon Sep 17 00:00:00 2001 From: Daniel McNab <36049421+DJMcNab@users.noreply.github.com> Date: Sun, 10 Jul 2022 21:04:35 +0000 Subject: [PATCH] Extract resources into their target location (#5271) # Objective - Extracting resources currently always uses commands, which requires *at least* one additional move of the extracted value, as well as dynamic dispatch. - Addresses https://github.com/bevyengine/bevy/pull/4402#discussion_r911634931 ## Solution - Write the resource into a `ResMut` directly. - Fall-back to commands if the resource hasn't been added yet. --- crates/bevy_render/Cargo.toml | 1 + crates/bevy_render/src/extract_resource.rs | 24 +++++++++++++++++----- 2 files changed, 20 insertions(+), 5 deletions(-) diff --git a/crates/bevy_render/Cargo.toml b/crates/bevy_render/Cargo.toml index a1643eafd58950..8d646a6d6f64a6 100644 --- a/crates/bevy_render/Cargo.toml +++ b/crates/bevy_render/Cargo.toml @@ -34,6 +34,7 @@ bevy_core = { path = "../bevy_core", version = "0.8.0-dev" } bevy_derive = { path = "../bevy_derive", version = "0.8.0-dev" } bevy_ecs = { path = "../bevy_ecs", version = "0.8.0-dev" } bevy_encase_derive = { path = "../bevy_encase_derive", version = "0.8.0-dev" } +bevy_log = { path = "../bevy_log", version = "0.8.0-dev" } bevy_math = { path = "../bevy_math", version = "0.8.0-dev" } bevy_mikktspace = { path = "../bevy_mikktspace", version = "0.8.0-dev" } bevy_reflect = { path = "../bevy_reflect", version = "0.8.0-dev", features = ["bevy"] } diff --git a/crates/bevy_render/src/extract_resource.rs b/crates/bevy_render/src/extract_resource.rs index 57be9ab7e1c824..0e57d1d1233072 100644 --- a/crates/bevy_render/src/extract_resource.rs +++ b/crates/bevy_render/src/extract_resource.rs @@ -1,7 +1,7 @@ use std::marker::PhantomData; use bevy_app::{App, Plugin}; -use bevy_ecs::system::{Commands, Res, Resource}; +use bevy_ecs::system::{Commands, Local, Res, ResMut, Resource}; pub use bevy_render_macros::ExtractResource; use crate::{Extract, RenderApp, RenderStage}; @@ -38,12 +38,26 @@ impl Plugin for ExtractResourcePlugin { } /// This system extracts the resource of the corresponding [`Resource`] type -/// by cloning it. pub fn extract_resource( mut commands: Commands, - resource: Extract>, + main_resource: Extract>, + target_resource: Option>, + #[cfg(debug_assertions)] mut has_warned_on_remove: Local, ) { - if resource.is_changed() { - commands.insert_resource(R::extract_resource(&*resource)); + if let Some(mut target_resource) = target_resource { + if main_resource.is_changed() { + *target_resource = R::extract_resource(&*main_resource); + } + } else { + #[cfg(debug_assertions)] + if !main_resource.is_added() && !*has_warned_on_remove { + *has_warned_on_remove = true; + bevy_log::warn!( + "Removing resource {} from render world not expected, adding using `Commands`. + This may decrease performance", + std::any::type_name::() + ); + } + commands.insert_resource(R::extract_resource(&*main_resource)); } }