From 84fd7540a9585caa17b7e43e4ed3171349517eaa Mon Sep 17 00:00:00 2001 From: Robert Swain Date: Thu, 24 Aug 2023 00:26:46 +0200 Subject: [PATCH 01/14] Use PassHashMap for render world entity storage for better performance --- crates/bevy_asset/src/id.rs | 3 +- crates/bevy_pbr/src/material.rs | 95 ++++---- crates/bevy_pbr/src/prepass/mod.rs | 25 ++- crates/bevy_pbr/src/render/light.rs | 21 +- crates/bevy_pbr/src/render/mesh.rs | 231 ++++++++++++-------- crates/bevy_pbr/src/render/morph.rs | 18 +- crates/bevy_pbr/src/wireframe.rs | 61 ++++-- crates/bevy_render/src/batching/mod.rs | 25 ++- crates/bevy_render/src/extract_component.rs | 6 +- crates/bevy_sprite/src/mesh2d/material.rs | 94 ++++---- crates/bevy_sprite/src/mesh2d/mesh.rs | 157 ++++++++----- crates/bevy_sprite/src/render/mod.rs | 7 +- crates/bevy_ui/src/render/mod.rs | 7 +- crates/bevy_utils/src/lib.rs | 11 + examples/shader/shader_instancing.rs | 58 ++--- examples/stress_tests/bevymark.rs | 3 +- examples/stress_tests/many_cubes.rs | 2 + 17 files changed, 495 insertions(+), 329 deletions(-) diff --git a/crates/bevy_asset/src/id.rs b/crates/bevy_asset/src/id.rs index 428b992ca0742..e499057bdee34 100644 --- a/crates/bevy_asset/src/id.rs +++ b/crates/bevy_asset/src/id.rs @@ -1,4 +1,5 @@ use crate::{Asset, AssetIndex, Handle, UntypedHandle}; +use bevy_ecs::component::Component; use bevy_reflect::{Reflect, Uuid}; use std::{ any::TypeId, @@ -13,7 +14,7 @@ use std::{ /// For an identifier tied to the lifetime of an asset, see [`Handle`]. /// /// For an "untyped" / "generic-less" id, see [`UntypedAssetId`]. -#[derive(Reflect)] +#[derive(Component, Reflect)] pub enum AssetId { /// A small / efficient runtime identifier that can be used to efficiently look up an asset stored in [`Assets`]. This is /// the "default" identifier used for assets. The alternative(s) (ex: [`AssetId::Uuid`]) will only be used if assets are diff --git a/crates/bevy_pbr/src/material.rs b/crates/bevy_pbr/src/material.rs index d9c835abcaffa..3def1cfe463d8 100644 --- a/crates/bevy_pbr/src/material.rs +++ b/crates/bevy_pbr/src/material.rs @@ -1,6 +1,6 @@ use crate::{ render, AlphaMode, DrawMesh, DrawPrepass, EnvironmentMapLight, MeshPipeline, MeshPipelineKey, - MeshTransforms, PrepassPipelinePlugin, PrepassPlugin, ScreenSpaceAmbientOcclusionSettings, + PrepassPipelinePlugin, PrepassPlugin, RenderMeshInstances, ScreenSpaceAmbientOcclusionSettings, SetMeshBindGroup, SetMeshViewBindGroup, Shadow, }; use bevy_app::{App, Plugin}; @@ -14,10 +14,7 @@ use bevy_core_pipeline::{ use bevy_derive::{Deref, DerefMut}; use bevy_ecs::{ prelude::*, - system::{ - lifetimeless::{Read, SRes}, - SystemParamItem, - }, + system::{lifetimeless::SRes, SystemParamItem}, }; use bevy_render::{ mesh::{Mesh, MeshVertexBufferLayout}, @@ -37,7 +34,7 @@ use bevy_render::{ view::{ExtractedView, Msaa, ViewVisibility, VisibleEntities}, Extract, ExtractSchedule, Render, RenderApp, RenderSet, }; -use bevy_utils::{tracing::error, HashMap, HashSet}; +use bevy_utils::{tracing::error, HashMap, HashSet, PassHashMap}; use std::hash::Hash; use std::marker::PhantomData; @@ -190,6 +187,7 @@ where .add_render_command::>() .init_resource::>() .init_resource::>() + .init_resource::>() .init_resource::>>() .add_systems( ExtractSchedule, @@ -226,26 +224,6 @@ where } } -fn extract_material_meshes( - mut commands: Commands, - mut previous_len: Local, - query: Extract)>>, -) { - let mut values = Vec::with_capacity(*previous_len); - for (entity, view_visibility, material) in &query { - if view_visibility.get() { - // NOTE: MaterialBindGroupId is inserted here to avoid a table move. Upcoming changes - // to use SparseSet for render world entity storage will do this automatically. - values.push(( - entity, - (material.clone_weak(), MaterialBindGroupId::default()), - )); - } - } - *previous_len = values.len(); - commands.insert_or_spawn_batch(values); -} - /// A key uniquely identifying a specialized [`MaterialPipeline`]. pub struct MaterialPipelineKey { pub mesh_key: MeshPipelineKey, @@ -368,24 +346,53 @@ type DrawMaterial = ( /// Sets the bind group for a given [`Material`] at the configured `I` index. pub struct SetMaterialBindGroup(PhantomData); impl RenderCommand

for SetMaterialBindGroup { - type Param = SRes>; + type Param = (SRes>, SRes>); type ViewWorldQuery = (); - type ItemWorldQuery = Read>; + type ItemWorldQuery = (); #[inline] fn render<'w>( - _item: &P, + item: &P, _view: (), - material_handle: &'_ Handle, - materials: SystemParamItem<'w, '_, Self::Param>, + _item_query: (), + (materials, material_instances): SystemParamItem<'w, '_, Self::Param>, pass: &mut TrackedRenderPass<'w>, ) -> RenderCommandResult { - let material = materials.into_inner().get(&material_handle.id()).unwrap(); + let materials = materials.into_inner(); + let material_instances = material_instances.into_inner(); + + let Some(material_asset_id) = material_instances.get(&item.entity()) else { + return RenderCommandResult::Failure; + }; + let Some(material) = materials.get(material_asset_id) else { + return RenderCommandResult::Failure; + }; pass.set_bind_group(I, &material.bind_group, &[]); RenderCommandResult::Success } } +#[derive(Resource, Deref, DerefMut)] +pub struct RenderMaterialInstances(PassHashMap>); + +impl Default for RenderMaterialInstances { + fn default() -> Self { + Self(Default::default()) + } +} + +fn extract_material_meshes( + mut material_instances: ResMut>, + query: Extract)>>, +) { + material_instances.clear(); + for (entity, view_visibility, handle) in &query { + if view_visibility.get() { + material_instances.insert(entity, handle.id()); + } + } +} + const fn alpha_mode_pipeline_key(alpha_mode: AlphaMode) -> MeshPipelineKey { match alpha_mode { // Premultiplied and Add share the same pipeline key @@ -424,12 +431,8 @@ pub fn queue_material_meshes( msaa: Res, render_meshes: Res>, render_materials: Res>, - mut material_meshes: Query<( - &Handle, - &mut MaterialBindGroupId, - &Handle, - &MeshTransforms, - )>, + mut render_mesh_instances: ResMut, + render_material_instances: Res>, images: Res>, mut views: Query<( &ExtractedView, @@ -493,15 +496,16 @@ pub fn queue_material_meshes( } let rangefinder = view.rangefinder3d(); for visible_entity in &visible_entities.entities { - let Ok((material_handle, mut material_bind_group_id, mesh_handle, mesh_transforms)) = - material_meshes.get_mut(*visible_entity) - else { + let Some(material_asset_id) = render_material_instances.get(visible_entity) else { + continue; + }; + let Some(mesh_instance) = render_mesh_instances.get_mut(visible_entity) else { continue; }; - let Some(mesh) = render_meshes.get(mesh_handle) else { + let Some(mesh) = render_meshes.get(mesh_instance.mesh_asset_id) else { continue; }; - let Some(material) = render_materials.get(&material_handle.id()) else { + let Some(material) = render_materials.get(material_asset_id) else { continue; }; let mut mesh_key = view_key; @@ -530,9 +534,10 @@ pub fn queue_material_meshes( } }; - *material_bind_group_id = material.get_bind_group_id(); + mesh_instance.material_bind_group_id = material.get_bind_group_id(); - let distance = rangefinder.distance_translation(&mesh_transforms.transform.translation) + let distance = rangefinder + .distance_translation(&mesh_instance.transforms.transform.translation) + material.properties.depth_bias; match material.properties.alpha_mode { AlphaMode::Opaque => { diff --git a/crates/bevy_pbr/src/prepass/mod.rs b/crates/bevy_pbr/src/prepass/mod.rs index ca1caf5c44720..e11209f1e8eba 100644 --- a/crates/bevy_pbr/src/prepass/mod.rs +++ b/crates/bevy_pbr/src/prepass/mod.rs @@ -47,7 +47,8 @@ use bevy_utils::tracing::error; use crate::{ prepare_materials, setup_morph_and_skinning_defs, AlphaMode, DrawMesh, Material, MaterialPipeline, MaterialPipelineKey, MeshLayouts, MeshPipeline, MeshPipelineKey, - MeshTransforms, RenderMaterials, SetMaterialBindGroup, SetMeshBindGroup, + RenderMaterialInstances, RenderMaterials, RenderMeshInstances, SetMaterialBindGroup, + SetMeshBindGroup, }; use std::{hash::Hash, marker::PhantomData}; @@ -755,8 +756,9 @@ pub fn queue_prepass_material_meshes( pipeline_cache: Res, msaa: Res, render_meshes: Res>, + render_mesh_instances: Res, render_materials: Res>, - material_meshes: Query<(&Handle, &Handle, &MeshTransforms)>, + render_material_instances: Res>, mut views: Query<( &ExtractedView, &VisibleEntities, @@ -801,16 +803,16 @@ pub fn queue_prepass_material_meshes( let rangefinder = view.rangefinder3d(); for visible_entity in &visible_entities.entities { - let Ok((material_handle, mesh_handle, mesh_transforms)) = - material_meshes.get(*visible_entity) - else { + let Some(material_asset_id) = render_material_instances.get(visible_entity) else { continue; }; - - let (Some(material), Some(mesh)) = ( - render_materials.get(&material_handle.id()), - render_meshes.get(mesh_handle), - ) else { + let Some(mesh_instance) = render_mesh_instances.get(visible_entity) else { + continue; + }; + let Some(material) = render_materials.get(material_asset_id) else { + continue; + }; + let Some(mesh) = render_meshes.get(mesh_instance.mesh_asset_id) else { continue; }; @@ -846,7 +848,8 @@ pub fn queue_prepass_material_meshes( } }; - let distance = rangefinder.distance_translation(&mesh_transforms.transform.translation) + let distance = rangefinder + .distance_translation(&mesh_instance.transforms.transform.translation) + material.properties.depth_bias; match alpha_mode { AlphaMode::Opaque => { diff --git a/crates/bevy_pbr/src/render/light.rs b/crates/bevy_pbr/src/render/light.rs index bad686b92f2f9..c488ca23a16d1 100644 --- a/crates/bevy_pbr/src/render/light.rs +++ b/crates/bevy_pbr/src/render/light.rs @@ -3,10 +3,9 @@ use crate::{ CascadeShadowConfig, Cascades, CascadesVisibleEntities, Clusters, CubemapVisibleEntities, DirectionalLight, DirectionalLightShadowMap, DrawPrepass, EnvironmentMapLight, GlobalVisiblePointLights, Material, MaterialPipelineKey, MeshPipeline, MeshPipelineKey, - NotShadowCaster, PointLight, PointLightShadowMap, PrepassPipeline, RenderMaterials, SpotLight, - VisiblePointLights, + PointLight, PointLightShadowMap, PrepassPipeline, RenderMaterialInstances, RenderMaterials, + RenderMeshInstances, SpotLight, VisiblePointLights, }; -use bevy_asset::Handle; use bevy_core_pipeline::core_3d::Transparent3d; use bevy_ecs::prelude::*; use bevy_math::{Mat4, UVec3, UVec4, Vec2, Vec3, Vec3Swizzles, Vec4, Vec4Swizzles}; @@ -1549,9 +1548,10 @@ pub fn prepare_clusters( pub fn queue_shadows( shadow_draw_functions: Res>, prepass_pipeline: Res>, - casting_meshes: Query<(&Handle, &Handle), Without>, render_meshes: Res>, + render_mesh_instances: Res, render_materials: Res>, + render_material_instances: Res>, mut pipelines: ResMut>>, pipeline_cache: Res, view_lights: Query<(Entity, &ViewLightEntities)>, @@ -1594,15 +1594,22 @@ pub fn queue_shadows( // NOTE: Lights with shadow mapping disabled will have no visible entities // so no meshes will be queued for entity in visible_entities.iter().copied() { - let Ok((mesh_handle, material_handle)) = casting_meshes.get(entity) else { + let Some(mesh_instance) = render_mesh_instances.get(&entity) else { continue; }; - let Some(mesh) = render_meshes.get(mesh_handle) else { + if !mesh_instance.shadow_caster { + continue; + } + let Some(material_asset_id) = render_material_instances.get(&entity) else { continue; }; - let Some(material) = render_materials.get(&material_handle.id()) else { + let Some(material) = render_materials.get(material_asset_id) else { continue; }; + let Some(mesh) = render_meshes.get(mesh_instance.mesh_asset_id) else { + continue; + }; + let mut mesh_key = MeshPipelineKey::from_primitive_topology(mesh.primitive_topology) | MeshPipelineKey::DEPTH_PREPASS; diff --git a/crates/bevy_pbr/src/render/mesh.rs b/crates/bevy_pbr/src/render/mesh.rs index 995c8bfa59f2c..312f6419d33f5 100644 --- a/crates/bevy_pbr/src/render/mesh.rs +++ b/crates/bevy_pbr/src/render/mesh.rs @@ -1,8 +1,9 @@ use crate::{ - environment_map, prepass, EnvironmentMapLight, FogMeta, GlobalLightMeta, GpuFog, GpuLights, - GpuPointLights, LightMeta, MaterialBindGroupId, NotShadowCaster, NotShadowReceiver, - PreviousGlobalTransform, ScreenSpaceAmbientOcclusionTextures, Shadow, ShadowSamplers, - ViewClusterBindings, ViewFogUniformOffset, ViewLightsUniformOffset, ViewShadowBindings, + environment_map, prepass, render::morph::MorphInstances, EnvironmentMapLight, FogMeta, + GlobalLightMeta, GpuFog, GpuLights, GpuPointLights, LightMeta, MaterialBindGroupId, + NotShadowCaster, NotShadowReceiver, PreviousGlobalTransform, + ScreenSpaceAmbientOcclusionTextures, Shadow, ShadowSamplers, ViewClusterBindings, + ViewFogUniformOffset, ViewLightsUniformOffset, ViewShadowBindings, CLUSTERED_FORWARD_STORAGE_BUFFER_COUNT, MAX_CASCADES_PER_LIGHT, MAX_DIRECTIONAL_LIGHTS, }; use bevy_app::Plugin; @@ -14,6 +15,7 @@ use bevy_core_pipeline::{ get_lut_bind_group_layout_entries, get_lut_bindings, Tonemapping, TonemappingLuts, }, }; +use bevy_derive::{Deref, DerefMut}; use bevy_ecs::{ prelude::*, query::{QueryItem, ROQueryItem}, @@ -44,10 +46,10 @@ use bevy_render::{ Extract, ExtractSchedule, Render, RenderApp, RenderSet, }; use bevy_transform::components::GlobalTransform; -use bevy_utils::{tracing::error, HashMap, Hashed}; +use bevy_utils::{tracing::error, HashMap, Hashed, PassHashMap}; use crate::render::{ - morph::{extract_morphs, prepare_morphs, MorphIndex, MorphUniform}, + morph::{extract_morphs, prepare_morphs, MorphUniform}, MeshLayouts, }; @@ -112,6 +114,9 @@ impl Plugin for MeshRenderPlugin { if let Ok(render_app) = app.get_sub_app_mut(RenderApp) { render_app + .init_resource::() + .init_resource::() + .init_resource::() .init_resource::() .init_resource::() .init_resource::() @@ -220,10 +225,23 @@ bitflags::bitflags! { } } +pub struct RenderMeshInstance { + pub transforms: MeshTransforms, + pub mesh_asset_id: AssetId, + pub material_bind_group_id: MaterialBindGroupId, + pub shadow_caster: bool, + pub automatic_batching: bool, +} + +#[derive(Default, Resource, Deref, DerefMut)] +pub struct RenderMeshInstances(PassHashMap); + +#[derive(Component)] +pub struct Mesh3d; + pub fn extract_meshes( mut commands: Commands, - mut prev_caster_commands_len: Local, - mut prev_not_caster_commands_len: Local, + mut render_mesh_instances: ResMut, meshes_query: Extract< Query<( Entity, @@ -233,15 +251,26 @@ pub fn extract_meshes( &Handle, Option>, Option>, + Has, )>, >, ) { - let mut caster_commands = Vec::with_capacity(*prev_caster_commands_len); - let mut not_caster_commands = Vec::with_capacity(*prev_not_caster_commands_len); + let capacity = meshes_query.iter().len(); + render_mesh_instances.clear(); + let mut entities = Vec::with_capacity(capacity); + let visible_meshes = meshes_query.iter().filter(|(_, vis, ..)| vis.get()); - for (entity, _, transform, previous_transform, handle, not_receiver, not_caster) in - visible_meshes + for ( + entity, + _, + transform, + previous_transform, + handle, + not_receiver, + not_caster, + no_automatic_batching, + ) in visible_meshes { let transform = transform.affine(); let previous_transform = previous_transform.map(|t| t.0).unwrap_or(transform); @@ -258,16 +287,21 @@ pub fn extract_meshes( previous_transform: (&previous_transform).into(), flags: flags.bits(), }; - if not_caster.is_some() { - not_caster_commands.push((entity, (handle.clone_weak(), transforms, NotShadowCaster))); - } else { - caster_commands.push((entity, (handle.clone_weak(), transforms))); - } + // FIXME: Remove this - it is just a workaround to enable rendering to work as + // render commands require an entity to exist at the moment. + entities.push((entity, Mesh3d)); + render_mesh_instances.insert( + entity, + RenderMeshInstance { + mesh_asset_id: handle.id(), + transforms, + shadow_caster: not_caster.is_none(), + material_bind_group_id: MaterialBindGroupId::default(), + automatic_batching: !no_automatic_batching, + }, + ); } - *prev_caster_commands_len = caster_commands.len(); - *prev_not_caster_commands_len = not_caster_commands.len(); - commands.insert_or_spawn_batch(caster_commands); - commands.insert_or_spawn_batch(not_caster_commands); + commands.insert_or_spawn_batch(entities); } #[derive(Component)] @@ -316,16 +350,19 @@ impl SkinnedMeshJoints { } } +#[derive(Default, Resource, Deref, DerefMut)] +pub struct SkinnedMeshJointsInstances(PassHashMap); + pub fn extract_skinned_meshes( - mut commands: Commands, - mut previous_len: Local, + mut skinned_mesh_joints_instances: ResMut, mut uniform: ResMut, query: Extract>, inverse_bindposes: Extract>>, joint_query: Extract>, ) { + skinned_mesh_joints_instances.clear(); uniform.buffer.clear(); - let mut values = Vec::with_capacity(*previous_len); + let mut last_start = 0; for (entity, view_visibility, skin) in &query { @@ -337,12 +374,7 @@ pub fn extract_skinned_meshes( SkinnedMeshJoints::build(skin, &inverse_bindposes, &joint_query, &mut uniform.buffer) { last_start = last_start.max(skinned_joints.index as usize); - // NOTE: The skinned joints uniform buffer has to be bound at a dynamic offset per - // entity and so cannot currently be batched. - values.push(( - entity, - (skinned_joints.to_buffer_index(), NoAutomaticBatching), - )); + skinned_mesh_joints_instances.insert(entity, skinned_joints.to_buffer_index()); } } @@ -350,9 +382,6 @@ pub fn extract_skinned_meshes( while uniform.buffer.len() - last_start < MAX_JOINTS { uniform.buffer.push(Mat4::ZERO); } - - *previous_len = values.len(); - commands.insert_or_spawn_batch(values); } #[derive(Resource, Clone)] @@ -638,22 +667,28 @@ impl MeshPipeline { } impl GetBatchData for MeshPipeline { - type Query = ( - Option<&'static MaterialBindGroupId>, - &'static Handle, - &'static MeshTransforms, - ); - type CompareData = (Option, AssetId); + type Param = SRes; + type Query = Entity; + type QueryFilter = With; + type CompareData = (MaterialBindGroupId, AssetId); type BufferData = MeshUniform; - fn get_buffer_data(&(.., mesh_transforms): &QueryItem) -> Self::BufferData { - mesh_transforms.into() - } - - fn get_compare_data( - &(material_bind_group_id, mesh_handle, ..): &QueryItem, - ) -> Self::CompareData { - (material_bind_group_id.copied(), mesh_handle.id()) + fn get_batch_data( + mesh_instances: &SystemParamItem, + entity: &QueryItem, + ) -> (Self::BufferData, Option) { + let mesh_instance = mesh_instances + .get(entity) + .expect("Failed to find render mesh instance"); + ( + (&mesh_instance.transforms).into(), + mesh_instance.automatic_batching.then(|| { + ( + mesh_instance.material_bind_group_id, + mesh_instance.mesh_asset_id, + ) + }), + ) } } @@ -1025,12 +1060,12 @@ impl MeshBindGroups { /// Get the `BindGroup` for `GpuMesh` with given `handle_id`. pub fn get( &self, - handle_id: AssetId, + asset_id: AssetId, is_skinned: bool, morph: bool, ) -> Option<&BindGroup> { match (is_skinned, morph) { - (_, true) => self.morph_targets.get(&handle_id), + (_, true) => self.morph_targets.get(&asset_id), (true, false) => self.skinned.as_ref(), (false, false) => self.model_only.as_ref(), } @@ -1304,27 +1339,44 @@ impl RenderCommand

for SetMeshViewBindGroup pub struct SetMeshBindGroup; impl RenderCommand

for SetMeshBindGroup { - type Param = SRes; - type ViewWorldQuery = (); - type ItemWorldQuery = ( - Read>, - Option>, - Option>, + type Param = ( + SRes, + SRes, + SRes, + SRes, ); + type ViewWorldQuery = (); + type ItemWorldQuery = (); #[inline] fn render<'w>( item: &P, _view: (), - (mesh, skin_index, morph_index): ROQueryItem, - bind_groups: SystemParamItem<'w, '_, Self::Param>, + _item_query: (), + (bind_groups, mesh_instances, skin_instances, morph_instances): SystemParamItem< + 'w, + '_, + Self::Param, + >, pass: &mut TrackedRenderPass<'w>, ) -> RenderCommandResult { let bind_groups = bind_groups.into_inner(); + let mesh_instances = mesh_instances.into_inner(); + let skin_instances = skin_instances.into_inner(); + let morph_instances = morph_instances.into_inner(); + + let entity = &item.entity(); + + let Some(mesh) = mesh_instances.get(entity) else { + return RenderCommandResult::Success; + }; + let skin_index = skin_instances.get(entity); + let morph_index = morph_instances.get(entity); + let is_skinned = skin_index.is_some(); let is_morphed = morph_index.is_some(); - let Some(bind_group) = bind_groups.get(mesh.id(), is_skinned, is_morphed) else { + let Some(bind_group) = bind_groups.get(mesh.mesh_asset_id, is_skinned, is_morphed) else { error!( "The MeshBindGroups resource wasn't set in the render phase. \ It should be set by the queue_mesh_bind_group system.\n\ @@ -1355,43 +1407,50 @@ impl RenderCommand

for SetMeshBindGroup { pub struct DrawMesh; impl RenderCommand

for DrawMesh { - type Param = SRes>; + type Param = (SRes>, SRes); type ViewWorldQuery = (); - type ItemWorldQuery = Read>; + type ItemWorldQuery = (); #[inline] fn render<'w>( item: &P, _view: (), - mesh_handle: ROQueryItem<'_, Self::ItemWorldQuery>, - meshes: SystemParamItem<'w, '_, Self::Param>, + _item_query: (), + (meshes, mesh_instances): SystemParamItem<'w, '_, Self::Param>, pass: &mut TrackedRenderPass<'w>, ) -> RenderCommandResult { - if let Some(gpu_mesh) = meshes.into_inner().get(mesh_handle) { - let batch_range = item.batch_range(); - pass.set_vertex_buffer(0, gpu_mesh.vertex_buffer.slice(..)); - #[cfg(all(feature = "webgl", target_arch = "wasm32"))] - pass.set_push_constants( - ShaderStages::VERTEX, - 0, - &(batch_range.start as i32).to_le_bytes(), - ); - match &gpu_mesh.buffer_info { - GpuBufferInfo::Indexed { - buffer, - index_format, - count, - } => { - pass.set_index_buffer(buffer.slice(..), 0, *index_format); - pass.draw_indexed(0..*count, 0, batch_range.clone()); - } - GpuBufferInfo::NonIndexed => { - pass.draw(0..gpu_mesh.vertex_count, batch_range.clone()); - } + let meshes = meshes.into_inner(); + let mesh_instances = mesh_instances.into_inner(); + + let Some(mesh_instance) = mesh_instances.get(&item.entity()) else { + return RenderCommandResult::Failure; + }; + let Some(gpu_mesh) = meshes.get(mesh_instance.mesh_asset_id) else { + return RenderCommandResult::Failure; + }; + + pass.set_vertex_buffer(0, gpu_mesh.vertex_buffer.slice(..)); + + let batch_range = item.batch_range(); + #[cfg(all(feature = "webgl", target_arch = "wasm32"))] + pass.set_push_constants( + ShaderStages::VERTEX, + 0, + &(batch_range.start as i32).to_le_bytes(), + ); + match &gpu_mesh.buffer_info { + GpuBufferInfo::Indexed { + buffer, + index_format, + count, + } => { + pass.set_index_buffer(buffer.slice(..), 0, *index_format); + pass.draw_indexed(0..*count, 0, batch_range.clone()); + } + GpuBufferInfo::NonIndexed => { + pass.draw(0..gpu_mesh.vertex_count, batch_range.clone()); } - RenderCommandResult::Success - } else { - RenderCommandResult::Failure } + RenderCommandResult::Success } } diff --git a/crates/bevy_pbr/src/render/morph.rs b/crates/bevy_pbr/src/render/morph.rs index 5b98de2ad84d9..9fed9440b3333 100644 --- a/crates/bevy_pbr/src/render/morph.rs +++ b/crates/bevy_pbr/src/render/morph.rs @@ -1,14 +1,15 @@ use std::{iter, mem}; +use bevy_derive::{Deref, DerefMut}; use bevy_ecs::prelude::*; use bevy_render::{ - batching::NoAutomaticBatching, mesh::morph::{MeshMorphWeights, MAX_MORPH_WEIGHTS}, render_resource::{BufferUsages, BufferVec}, renderer::{RenderDevice, RenderQueue}, view::ViewVisibility, Extract, }; +use bevy_utils::PassHashMap; use bytemuck::Pod; #[derive(Component)] @@ -69,16 +70,17 @@ fn add_to_alignment(buffer: &mut BufferVec) { buffer.extend(iter::repeat_with(T::default).take(ts_to_add)); } +#[derive(Default, Resource, Deref, DerefMut)] +pub struct MorphInstances(PassHashMap); + pub fn extract_morphs( - mut commands: Commands, - mut previous_len: Local, + mut morph_instances: ResMut, mut uniform: ResMut, query: Extract>, ) { + morph_instances.clear(); uniform.buffer.clear(); - let mut values = Vec::with_capacity(*previous_len); - for (entity, view_visibility, morph_weights) in &query { if !view_visibility.get() { continue; @@ -90,10 +92,6 @@ pub fn extract_morphs( add_to_alignment::(&mut uniform.buffer); let index = (start * mem::size_of::()) as u32; - // NOTE: Because morph targets require per-morph target texture bindings, they cannot - // currently be batched. - values.push((entity, (MorphIndex { index }, NoAutomaticBatching))); + morph_instances.insert(entity, MorphIndex { index }); } - *previous_len = values.len(); - commands.insert_or_spawn_batch(values); } diff --git a/crates/bevy_pbr/src/wireframe.rs b/crates/bevy_pbr/src/wireframe.rs index b1be7a2ef5ccb..08a20433eb936 100644 --- a/crates/bevy_pbr/src/wireframe.rs +++ b/crates/bevy_pbr/src/wireframe.rs @@ -1,13 +1,15 @@ -use crate::{DrawMesh, MeshPipelineKey, SetMeshBindGroup, SetMeshViewBindGroup}; -use crate::{MeshPipeline, MeshTransforms}; +use crate::MeshPipeline; +use crate::{ + DrawMesh, MeshPipelineKey, RenderMeshInstance, RenderMeshInstances, SetMeshBindGroup, + SetMeshViewBindGroup, +}; use bevy_app::Plugin; use bevy_asset::{load_internal_asset, Handle}; use bevy_core_pipeline::core_3d::Opaque3d; +use bevy_derive::{Deref, DerefMut}; use bevy_ecs::{prelude::*, reflect::ReflectComponent}; use bevy_reflect::std_traits::ReflectDefault; use bevy_reflect::Reflect; -use bevy_render::extract_component::{ExtractComponent, ExtractComponentPlugin}; -use bevy_render::Render; use bevy_render::{ extract_resource::{ExtractResource, ExtractResourcePlugin}, mesh::{Mesh, MeshVertexBufferLayout}, @@ -20,7 +22,9 @@ use bevy_render::{ view::{ExtractedView, Msaa, VisibleEntities}, RenderApp, RenderSet, }; +use bevy_render::{ExtractSchedule, Render}; use bevy_utils::tracing::error; +use bevy_utils::PassHashSet; pub const WIREFRAME_SHADER_HANDLE: Handle = Handle::weak_from_u128(192598014480025766); @@ -39,15 +43,13 @@ impl Plugin for WireframePlugin { app.register_type::() .register_type::() .init_resource::() - .add_plugins(( - ExtractResourcePlugin::::default(), - ExtractComponentPlugin::::default(), - )); + .add_plugins((ExtractResourcePlugin::::default(),)); if let Ok(render_app) = app.get_sub_app_mut(RenderApp) { render_app .add_render_command::() .init_resource::>() + .add_systems(ExtractSchedule, extract_wireframes) .add_systems(Render, queue_wireframes.in_set(RenderSet::QueueMeshes)); } } @@ -60,7 +62,7 @@ impl Plugin for WireframePlugin { } /// Controls whether an entity should rendered in wireframe-mode if the [`WireframePlugin`] is enabled -#[derive(Component, Debug, Clone, Default, ExtractComponent, Reflect)] +#[derive(Component, Debug, Clone, Default, Reflect)] #[reflect(Component, Default)] pub struct Wireframe; @@ -71,6 +73,14 @@ pub struct WireframeConfig { pub global: bool, } +#[derive(Resource, Deref, DerefMut)] +pub struct Wireframes(PassHashSet); + +fn extract_wireframes(mut wireframes: ResMut, query: Query>) { + wireframes.clear(); + wireframes.extend(query.into_iter()); +} + #[derive(Resource, Clone)] pub struct WireframePipeline { mesh_pipeline: MeshPipeline, @@ -110,15 +120,13 @@ impl SpecializedMeshPipeline for WireframePipeline { fn queue_wireframes( opaque_3d_draw_functions: Res>, render_meshes: Res>, + render_mesh_instances: Res, + wireframes: Res, wireframe_config: Res, wireframe_pipeline: Res, mut pipelines: ResMut>, pipeline_cache: Res, msaa: Res, - mut material_meshes: ParamSet<( - Query<(Entity, &Handle, &MeshTransforms)>, - Query<(Entity, &Handle, &MeshTransforms), With>, - )>, mut views: Query<(&ExtractedView, &VisibleEntities, &mut RenderPhase)>, ) { let draw_custom = opaque_3d_draw_functions.read().id::(); @@ -127,10 +135,10 @@ fn queue_wireframes( let rangefinder = view.rangefinder3d(); let view_key = msaa_key | MeshPipelineKey::from_hdr(view.hdr); - let add_render_phase = |phase_item: (Entity, &Handle, &MeshTransforms)| { - let (entity, mesh_handle, mesh_transforms) = phase_item; + let add_render_phase = |phase_item: (Entity, &RenderMeshInstance)| { + let (entity, mesh_instance) = phase_item; - let Some(mesh) = render_meshes.get(mesh_handle) else { + let Some(mesh) = render_meshes.get(mesh_instance.mesh_asset_id) else { return; }; let mut key = @@ -151,25 +159,36 @@ fn queue_wireframes( entity, pipeline: pipeline_id, draw_function: draw_custom, - distance: rangefinder.distance_translation(&mesh_transforms.transform.translation), + distance: rangefinder + .distance_translation(&mesh_instance.transforms.transform.translation), batch_range: 0..1, dynamic_offset: None, }); }; if wireframe_config.global { - let query = material_meshes.p0(); visible_entities .entities .iter() - .filter_map(|visible_entity| query.get(*visible_entity).ok()) + .filter_map(|visible_entity| { + render_mesh_instances + .get(visible_entity) + .map(|mesh_instance| (*visible_entity, mesh_instance)) + }) .for_each(add_render_phase); } else { - let query = material_meshes.p1(); visible_entities .entities .iter() - .filter_map(|visible_entity| query.get(*visible_entity).ok()) + .filter_map(|visible_entity| { + if wireframes.contains(visible_entity) { + render_mesh_instances + .get(visible_entity) + .map(|mesh_instance| (*visible_entity, mesh_instance)) + } else { + None + } + }) .for_each(add_render_phase); } } diff --git a/crates/bevy_render/src/batching/mod.rs b/crates/bevy_render/src/batching/mod.rs index 715402b2b4b16..6fbb2b46510d5 100644 --- a/crates/bevy_render/src/batching/mod.rs +++ b/crates/bevy_render/src/batching/mod.rs @@ -1,8 +1,8 @@ use bevy_ecs::{ component::Component, prelude::Res, - query::{Has, QueryItem, ReadOnlyWorldQuery}, - system::{Query, ResMut}, + query::{QueryItem, ReadOnlyWorldQuery}, + system::{Query, ResMut, StaticSystemParam, SystemParam, SystemParamItem}, }; use bevy_utils::nonmax::NonMaxU32; @@ -56,7 +56,9 @@ impl BatchMeta { /// A trait to support getting data used for batching draw commands via phase /// items. pub trait GetBatchData { + type Param: SystemParam + 'static; type Query: ReadOnlyWorldQuery; + type QueryFilter: ReadOnlyWorldQuery; /// Data used for comparison between phase items. If the pipeline id, draw /// function id, per-instance data buffer dynamic offset and this data /// matches, the draws can be batched. @@ -65,10 +67,12 @@ pub trait GetBatchData { /// containing these data for all instances. type BufferData: GpuArrayBufferable + Sync + Send + 'static; /// Get the per-instance data to be inserted into the [`GpuArrayBuffer`]. - fn get_buffer_data(query_item: &QueryItem) -> Self::BufferData; /// Get the data used for comparison when deciding whether draws can be /// batched. - fn get_compare_data(query_item: &QueryItem) -> Self::CompareData; + fn get_batch_data( + param: &SystemParamItem, + query_item: &QueryItem, + ) -> (Self::BufferData, Option); } /// Batch the items in a render phase. This means comparing metadata needed to draw each phase item @@ -76,24 +80,23 @@ pub trait GetBatchData { pub fn batch_and_prepare_render_phase( gpu_array_buffer: ResMut>, mut views: Query<&mut RenderPhase>, - query: Query<(Has, F::Query)>, + query: Query, + param: StaticSystemParam, ) { let gpu_array_buffer = gpu_array_buffer.into_inner(); + let system_param_item = param.into_inner(); let mut process_item = |item: &mut I| { - let (no_auto_batching, batch_query_item) = query.get(item.entity()).ok()?; + let batch_query_item = query.get(item.entity()).ok()?; - let buffer_data = F::get_buffer_data(&batch_query_item); + let (buffer_data, compare_data) = F::get_batch_data(&system_param_item, &batch_query_item); let buffer_index = gpu_array_buffer.push(buffer_data); let index = buffer_index.index.get(); *item.batch_range_mut() = index..index + 1; *item.dynamic_offset_mut() = buffer_index.dynamic_offset; - (!no_auto_batching).then(|| { - let compare_data = F::get_compare_data(&batch_query_item); - BatchMeta::new(item, compare_data) - }) + compare_data.map(|compare_data| BatchMeta::new(item, compare_data)) }; for mut phase in &mut views { diff --git a/crates/bevy_render/src/extract_component.rs b/crates/bevy_render/src/extract_component.rs index df555d392a96b..059843871c1b4 100644 --- a/crates/bevy_render/src/extract_component.rs +++ b/crates/bevy_render/src/extract_component.rs @@ -5,7 +5,7 @@ use crate::{ Extract, ExtractSchedule, Render, RenderApp, RenderSet, }; use bevy_app::{App, Plugin}; -use bevy_asset::{Asset, Handle}; +use bevy_asset::{Asset, AssetId, Handle}; use bevy_ecs::{ component::Component, prelude::*, @@ -194,11 +194,11 @@ impl Plugin for ExtractComponentPlugin { impl ExtractComponent for Handle { type Query = Read>; type Filter = (); - type Out = Handle; + type Out = AssetId; #[inline] fn extract_component(handle: QueryItem<'_, Self::Query>) -> Option { - Some(handle.clone_weak()) + Some(handle.id()) } } diff --git a/crates/bevy_sprite/src/mesh2d/material.rs b/crates/bevy_sprite/src/mesh2d/material.rs index 4b496c7242ac4..1398c8ef61348 100644 --- a/crates/bevy_sprite/src/mesh2d/material.rs +++ b/crates/bevy_sprite/src/mesh2d/material.rs @@ -7,11 +7,7 @@ use bevy_core_pipeline::{ use bevy_derive::{Deref, DerefMut}; use bevy_ecs::{ prelude::*, - query::ROQueryItem, - system::{ - lifetimeless::{Read, SRes}, - SystemParamItem, - }, + system::{lifetimeless::SRes, SystemParamItem}, }; use bevy_log::error; use bevy_render::{ @@ -33,12 +29,12 @@ use bevy_render::{ Extract, ExtractSchedule, Render, RenderApp, RenderSet, }; use bevy_transform::components::{GlobalTransform, Transform}; -use bevy_utils::{FloatOrd, HashMap, HashSet}; +use bevy_utils::{FloatOrd, HashMap, HashSet, PassHashMap}; use std::hash::Hash; use std::marker::PhantomData; use crate::{ - DrawMesh2d, Mesh2dHandle, Mesh2dPipeline, Mesh2dPipelineKey, Mesh2dTransforms, + DrawMesh2d, Mesh2dHandle, Mesh2dPipeline, Mesh2dPipelineKey, RenderMesh2dInstances, SetMesh2dBindGroup, SetMesh2dViewBindGroup, }; @@ -150,6 +146,7 @@ where .add_render_command::>() .init_resource::>() .init_resource::>() + .init_resource::>() .init_resource::>>() .add_systems( ExtractSchedule, @@ -176,24 +173,25 @@ where } } +#[derive(Resource, Deref, DerefMut)] +pub struct RenderMaterial2dInstances(PassHashMap>); + +impl Default for RenderMaterial2dInstances { + fn default() -> Self { + Self(Default::default()) + } +} + fn extract_material_meshes_2d( - mut commands: Commands, - mut previous_len: Local, + mut material_instances: ResMut>, query: Extract)>>, ) { - let mut values = Vec::with_capacity(*previous_len); - for (entity, view_visibility, material) in &query { + material_instances.clear(); + for (entity, view_visibility, handle) in &query { if view_visibility.get() { - // NOTE: Material2dBindGroupId is inserted here to avoid a table move. Upcoming changes - // to use SparseSet for render world entity storage will do this automatically. - values.push(( - entity, - (material.clone_weak(), Material2dBindGroupId::default()), - )); + material_instances.insert(entity, handle.id()); } } - *previous_len = values.len(); - commands.insert_or_spawn_batch(values); } /// Render pipeline data for a given [`Material2d`] @@ -322,19 +320,29 @@ pub struct SetMaterial2dBindGroup(PhantomData) impl RenderCommand

for SetMaterial2dBindGroup { - type Param = SRes>; + type Param = ( + SRes>, + SRes>, + ); type ViewWorldQuery = (); - type ItemWorldQuery = Read>; + type ItemWorldQuery = (); #[inline] fn render<'w>( - _item: &P, + item: &P, _view: (), - material2d_handle: ROQueryItem<'_, Self::ItemWorldQuery>, - materials: SystemParamItem<'w, '_, Self::Param>, + _item_query: (), + (materials, material_instances): SystemParamItem<'w, '_, Self::Param>, pass: &mut TrackedRenderPass<'w>, ) -> RenderCommandResult { - let material2d = materials.into_inner().get(&material2d_handle.id()).unwrap(); + let materials = materials.into_inner(); + let material_instances = material_instances.into_inner(); + let Some(material_instance) = material_instances.get(&item.entity()) else { + return RenderCommandResult::Failure; + }; + let Some(material2d) = materials.get(material_instance) else { + return RenderCommandResult::Failure; + }; pass.set_bind_group(I, &material2d.bind_group, &[]); RenderCommandResult::Success } @@ -364,12 +372,8 @@ pub fn queue_material2d_meshes( msaa: Res, render_meshes: Res>, render_materials: Res>, - mut material2d_meshes: Query<( - &Handle, - &mut Material2dBindGroupId, - &Mesh2dHandle, - &Mesh2dTransforms, - )>, + mut render_mesh_instances: ResMut, + render_material_instances: Res>, mut views: Query<( &ExtractedView, &VisibleEntities, @@ -380,7 +384,7 @@ pub fn queue_material2d_meshes( ) where M::Data: PartialEq + Eq + Hash + Clone, { - if material2d_meshes.is_empty() { + if render_material_instances.is_empty() { return; } @@ -400,19 +404,16 @@ pub fn queue_material2d_meshes( } } for visible_entity in &visible_entities.entities { - let Ok(( - material2d_handle, - mut material2d_bind_group_id, - mesh2d_handle, - mesh2d_uniform, - )) = material2d_meshes.get_mut(*visible_entity) - else { + let Some(material_asset_id) = render_material_instances.get(visible_entity) else { continue; }; - let Some(material2d) = render_materials.get(&material2d_handle.id()) else { + let Some(mesh_instance) = render_mesh_instances.get_mut(visible_entity) else { continue; }; - let Some(mesh) = render_meshes.get(&mesh2d_handle.0) else { + let Some(material2d) = render_materials.get(material_asset_id) else { + continue; + }; + let Some(mesh) = render_meshes.get(mesh_instance.mesh_asset_id) else { continue; }; let mesh_key = @@ -436,8 +437,9 @@ pub fn queue_material2d_meshes( } }; - *material2d_bind_group_id = material2d.get_bind_group_id(); - let mesh_z = mesh2d_uniform.transform.translation.z; + mesh_instance.material_bind_group_id = material2d.get_bind_group_id(); + + let mesh_z = mesh_instance.transforms.transform.translation.z; transparent_phase.add(Transparent2d { entity: *visible_entity, draw_function: draw_transparent_pbr, @@ -580,7 +582,7 @@ pub fn prepare_materials_2d( render_materials.remove(&removed); } - for (handle, material) in std::mem::take(&mut extracted_assets.extracted) { + for (asset_id, material) in std::mem::take(&mut extracted_assets.extracted) { match prepare_material2d( &material, &render_device, @@ -589,10 +591,10 @@ pub fn prepare_materials_2d( &pipeline, ) { Ok(prepared_asset) => { - render_materials.insert(handle, prepared_asset); + render_materials.insert(asset_id, prepared_asset); } Err(AsBindGroupError::RetryNextUpdate) => { - prepare_next_frame.assets.push((handle, material)); + prepare_next_frame.assets.push((asset_id, material)); } } } diff --git a/crates/bevy_sprite/src/mesh2d/mesh.rs b/crates/bevy_sprite/src/mesh2d/mesh.rs index 2717acd394d4e..ef08ba15892a7 100644 --- a/crates/bevy_sprite/src/mesh2d/mesh.rs +++ b/crates/bevy_sprite/src/mesh2d/mesh.rs @@ -2,6 +2,7 @@ use bevy_app::Plugin; use bevy_asset::{load_internal_asset, AssetId, Handle}; use bevy_core_pipeline::core_2d::Transparent2d; +use bevy_derive::{Deref, DerefMut}; use bevy_ecs::{ prelude::*, query::{QueryItem, ROQueryItem}, @@ -10,7 +11,10 @@ use bevy_ecs::{ use bevy_math::{Affine3, Vec2, Vec4}; use bevy_reflect::Reflect; use bevy_render::{ - batching::{batch_and_prepare_render_phase, write_batched_instance_buffer, GetBatchData}, + batching::{ + batch_and_prepare_render_phase, write_batched_instance_buffer, GetBatchData, + NoAutomaticBatching, + }, globals::{GlobalsBuffer, GlobalsUniform}, mesh::{GpuBufferInfo, Mesh, MeshVertexBufferLayout}, render_asset::RenderAssets, @@ -26,6 +30,7 @@ use bevy_render::{ Extract, ExtractSchedule, Render, RenderApp, RenderSet, }; use bevy_transform::components::GlobalTransform; +use bevy_utils::PassHashMap; use crate::Material2dBindGroupId; @@ -89,6 +94,7 @@ impl Plugin for Mesh2dRenderPlugin { if let Ok(render_app) = app.get_sub_app_mut(RenderApp) { render_app + .init_resource::() .init_resource::>() .add_systems(ExtractSchedule, extract_mesh2d) .add_systems( @@ -178,29 +184,57 @@ bitflags::bitflags! { } } +pub struct RenderMesh2dInstance { + pub transforms: Mesh2dTransforms, + pub mesh_asset_id: AssetId, + pub material_bind_group_id: Material2dBindGroupId, + pub automatic_batching: bool, +} + +#[derive(Default, Resource, Deref, DerefMut)] +pub struct RenderMesh2dInstances(PassHashMap); + +#[derive(Component)] +pub struct Mesh2d; + pub fn extract_mesh2d( mut commands: Commands, - mut previous_len: Local, - query: Extract>, + mut render_mesh_instances: ResMut, + query: Extract< + Query<( + Entity, + &ViewVisibility, + &GlobalTransform, + &Mesh2dHandle, + Has, + )>, + >, ) { - let mut values = Vec::with_capacity(*previous_len); - for (entity, view_visibility, transform, handle) in &query { + let capacity = query.iter().len(); + render_mesh_instances.clear(); + let mut entities = Vec::with_capacity(capacity); + + for (entity, view_visibility, transform, handle, no_automatic_batching) in &query { if !view_visibility.get() { continue; } - values.push(( + // FIXME: Remove this - it is just a workaround to enable rendering to work as + // render commands require an entity to exist at the moment. + entities.push((entity, Mesh2d)); + render_mesh_instances.insert( entity, - ( - Mesh2dHandle(handle.0.clone_weak()), - Mesh2dTransforms { + RenderMesh2dInstance { + transforms: Mesh2dTransforms { transform: (&transform.affine()).into(), flags: MeshFlags::empty().bits(), }, - ), - )); + mesh_asset_id: handle.0.id(), + material_bind_group_id: Material2dBindGroupId::default(), + automatic_batching: !no_automatic_batching, + }, + ); } - *previous_len = values.len(); - commands.insert_or_spawn_batch(values); + commands.insert_or_spawn_batch(entities); } #[derive(Resource, Clone)] @@ -325,22 +359,28 @@ impl Mesh2dPipeline { } impl GetBatchData for Mesh2dPipeline { - type Query = ( - Option<&'static Material2dBindGroupId>, - &'static Mesh2dHandle, - &'static Mesh2dTransforms, - ); - type CompareData = (Option, AssetId); + type Param = SRes; + type Query = Entity; + type QueryFilter = With; + type CompareData = (Material2dBindGroupId, AssetId); type BufferData = Mesh2dUniform; - fn get_buffer_data(&(.., mesh_transforms): &QueryItem) -> Self::BufferData { - mesh_transforms.into() - } - - fn get_compare_data( - &(material_bind_group_id, mesh_handle, ..): &QueryItem, - ) -> Self::CompareData { - (material_bind_group_id.copied(), mesh_handle.0.id()) + fn get_batch_data( + mesh_instances: &SystemParamItem, + entity: &QueryItem, + ) -> (Self::BufferData, Option) { + let mesh_instance = mesh_instances + .get(entity) + .expect("Failed to find render mesh2d instance"); + ( + (&mesh_instance.transforms).into(), + mesh_instance.automatic_batching.then(|| { + ( + mesh_instance.material_bind_group_id, + mesh_instance.mesh_asset_id, + ) + }), + ) } } @@ -653,43 +693,52 @@ impl RenderCommand

for SetMesh2dBindGroup { pub struct DrawMesh2d; impl RenderCommand

for DrawMesh2d { - type Param = SRes>; + type Param = (SRes>, SRes); type ViewWorldQuery = (); - type ItemWorldQuery = Read; + type ItemWorldQuery = (); #[inline] fn render<'w>( item: &P, _view: (), - mesh_handle: ROQueryItem<'w, Self::ItemWorldQuery>, - meshes: SystemParamItem<'w, '_, Self::Param>, + _item_query: (), + (meshes, render_mesh2d_instances): SystemParamItem<'w, '_, Self::Param>, pass: &mut TrackedRenderPass<'w>, ) -> RenderCommandResult { + let meshes = meshes.into_inner(); + let render_mesh2d_instances = render_mesh2d_instances.into_inner(); + + let Some(RenderMesh2dInstance { mesh_asset_id, .. }) = + render_mesh2d_instances.get(&item.entity()) + else { + return RenderCommandResult::Failure; + }; + let Some(gpu_mesh) = meshes.get(*mesh_asset_id) else { + return RenderCommandResult::Failure; + }; + + pass.set_vertex_buffer(0, gpu_mesh.vertex_buffer.slice(..)); + let batch_range = item.batch_range(); - if let Some(gpu_mesh) = meshes.into_inner().get(&mesh_handle.0) { - pass.set_vertex_buffer(0, gpu_mesh.vertex_buffer.slice(..)); - #[cfg(all(feature = "webgl", target_arch = "wasm32"))] - pass.set_push_constants( - ShaderStages::VERTEX, - 0, - &(batch_range.start as i32).to_le_bytes(), - ); - match &gpu_mesh.buffer_info { - GpuBufferInfo::Indexed { - buffer, - index_format, - count, - } => { - pass.set_index_buffer(buffer.slice(..), 0, *index_format); - pass.draw_indexed(0..*count, 0, batch_range.clone()); - } - GpuBufferInfo::NonIndexed => { - pass.draw(0..gpu_mesh.vertex_count, batch_range.clone()); - } + #[cfg(all(feature = "webgl", target_arch = "wasm32"))] + pass.set_push_constants( + ShaderStages::VERTEX, + 0, + &(batch_range.start as i32).to_le_bytes(), + ); + match &gpu_mesh.buffer_info { + GpuBufferInfo::Indexed { + buffer, + index_format, + count, + } => { + pass.set_index_buffer(buffer.slice(..), 0, *index_format); + pass.draw_indexed(0..*count, 0, batch_range.clone()); + } + GpuBufferInfo::NonIndexed => { + pass.draw(0..gpu_mesh.vertex_count, batch_range.clone()); } - RenderCommandResult::Success - } else { - RenderCommandResult::Failure } + RenderCommandResult::Success } } diff --git a/crates/bevy_sprite/src/render/mod.rs b/crates/bevy_sprite/src/render/mod.rs index 2d5343a867adc..a035b0a7b9efa 100644 --- a/crates/bevy_sprite/src/render/mod.rs +++ b/crates/bevy_sprite/src/render/mod.rs @@ -11,7 +11,6 @@ use bevy_core_pipeline::{ }; use bevy_ecs::{ prelude::*, - storage::SparseSet, system::{lifetimeless::*, SystemParamItem, SystemState}, }; use bevy_math::{Affine3A, Quat, Rect, Vec2, Vec4}; @@ -34,7 +33,7 @@ use bevy_render::{ Extract, }; use bevy_transform::components::GlobalTransform; -use bevy_utils::{FloatOrd, HashMap}; +use bevy_utils::{FloatOrd, HashMap, PassHashMap}; use bytemuck::{Pod, Zeroable}; use fixedbitset::FixedBitSet; @@ -330,7 +329,7 @@ pub struct ExtractedSprite { #[derive(Resource, Default)] pub struct ExtractedSprites { - pub sprites: SparseSet, + pub sprites: PassHashMap, } #[derive(Resource, Default)] @@ -641,7 +640,7 @@ pub fn prepare_sprites( // Compatible items share the same entity. for item_index in 0..transparent_phase.items.len() { let item = &transparent_phase.items[item_index]; - let Some(extracted_sprite) = extracted_sprites.sprites.get(item.entity) else { + let Some(extracted_sprite) = extracted_sprites.sprites.get(&item.entity) else { // If there is a phase item that is not a sprite, then we must start a new // batch to draw the other phase item(s) and to respect draw order. This can be // done by invalidating the batch_image_handle diff --git a/crates/bevy_ui/src/render/mod.rs b/crates/bevy_ui/src/render/mod.rs index d146e53beb79d..32af20d704b37 100644 --- a/crates/bevy_ui/src/render/mod.rs +++ b/crates/bevy_ui/src/render/mod.rs @@ -2,7 +2,6 @@ mod pipeline; mod render_pass; use bevy_core_pipeline::{core_2d::Camera2d, core_3d::Camera3d}; -use bevy_ecs::storage::SparseSet; use bevy_hierarchy::Parent; use bevy_render::render_phase::PhaseItem; use bevy_render::view::ViewVisibility; @@ -36,7 +35,7 @@ use bevy_sprite::{SpriteAssetEvents, TextureAtlas}; #[cfg(feature = "bevy_text")] use bevy_text::{PositionedGlyph, Text, TextLayoutInfo}; use bevy_transform::components::GlobalTransform; -use bevy_utils::{FloatOrd, HashMap}; +use bevy_utils::{FloatOrd, HashMap, PassHashMap}; use bytemuck::{Pod, Zeroable}; use std::ops::Range; @@ -164,7 +163,7 @@ pub struct ExtractedUiNode { #[derive(Resource, Default)] pub struct ExtractedUiNodes { - pub uinodes: SparseSet, + pub uinodes: PassHashMap, } pub fn extract_atlas_uinodes( @@ -733,7 +732,7 @@ pub fn prepare_uinodes( for item_index in 0..ui_phase.items.len() { let item = &mut ui_phase.items[item_index]; - if let Some(extracted_uinode) = extracted_uinodes.uinodes.get(item.entity) { + if let Some(extracted_uinode) = extracted_uinodes.uinodes.get(&item.entity) { let mut existing_batch = batches .last_mut() .filter(|_| batch_image_handle == extracted_uinode.image); diff --git a/crates/bevy_utils/src/lib.rs b/crates/bevy_utils/src/lib.rs index 52f33f31d7dc6..714c3dd44b13a 100644 --- a/crates/bevy_utils/src/lib.rs +++ b/crates/bevy_utils/src/lib.rs @@ -212,6 +212,11 @@ impl Hasher for PassHasher { panic!("can only hash u64 using PassHasher"); } + #[inline] + fn write_u32(&mut self, i: u32) { + self.hash = i as u64; + } + #[inline] fn write_u64(&mut self, i: u64) { self.hash = i; @@ -223,6 +228,12 @@ impl Hasher for PassHasher { } } +/// A [`HashMap`] pre-configured to use [`PassHash`] passthrough hashing. +pub type PassHashMap = hashbrown::HashMap; + +/// A [`HashSet`] pre-configured to use [`PassHash`] passthrough hashing. +pub type PassHashSet = hashbrown::HashSet; + /// A [`HashMap`] pre-configured to use [`Hashed`] keys and [`PassHash`] passthrough hashing. pub type PreHashMap = hashbrown::HashMap, V, PassHash>; diff --git a/examples/shader/shader_instancing.rs b/examples/shader/shader_instancing.rs index d5e751ae0fa1d..6d68f15ff2590 100644 --- a/examples/shader/shader_instancing.rs +++ b/examples/shader/shader_instancing.rs @@ -6,7 +6,7 @@ use bevy::{ query::QueryItem, system::{lifetimeless::*, SystemParamItem}, }, - pbr::{MeshPipeline, MeshPipelineKey, MeshTransforms, SetMeshBindGroup, SetMeshViewBindGroup}, + pbr::{MeshPipeline, MeshPipelineKey, SetMeshBindGroup, SetMeshViewBindGroup}, prelude::*, render::{ extract_component::{ExtractComponent, ExtractComponentPlugin}, @@ -22,6 +22,7 @@ use bevy::{ Render, RenderApp, RenderSet, }, }; +use bevy_internal::pbr::RenderMeshInstances; use bytemuck::{Pod, Zeroable}; fn main() { @@ -113,7 +114,8 @@ fn queue_custom( mut pipelines: ResMut>, pipeline_cache: Res, meshes: Res>, - material_meshes: Query<(Entity, &MeshTransforms, &Handle), With>, + render_mesh_instances: Res, + material_meshes: Query>, mut views: Query<(&ExtractedView, &mut RenderPhase)>, ) { let draw_custom = transparent_3d_draw_functions.read().id::(); @@ -123,23 +125,26 @@ fn queue_custom( for (view, mut transparent_phase) in &mut views { let view_key = msaa_key | MeshPipelineKey::from_hdr(view.hdr); let rangefinder = view.rangefinder3d(); - for (entity, mesh_transforms, mesh_handle) in &material_meshes { - if let Some(mesh) = meshes.get(mesh_handle) { - let key = - view_key | MeshPipelineKey::from_primitive_topology(mesh.primitive_topology); - let pipeline = pipelines - .specialize(&pipeline_cache, &custom_pipeline, key, &mesh.layout) - .unwrap(); - transparent_phase.add(Transparent3d { - entity, - pipeline, - draw_function: draw_custom, - distance: rangefinder - .distance_translation(&mesh_transforms.transform.translation), - batch_range: 0..1, - dynamic_offset: None, - }); - } + for entity in &material_meshes { + let Some(mesh_instance) = render_mesh_instances.get(&entity) else { + continue; + }; + let Some(mesh) = meshes.get(mesh_instance.mesh_asset_id) else { + continue; + }; + let key = view_key | MeshPipelineKey::from_primitive_topology(mesh.primitive_topology); + let pipeline = pipelines + .specialize(&pipeline_cache, &custom_pipeline, key, &mesh.layout) + .unwrap(); + transparent_phase.add(Transparent3d { + entity, + pipeline, + draw_function: draw_custom, + distance: rangefinder + .distance_translation(&mesh_instance.transforms.transform.translation), + batch_range: 0..1, + dynamic_offset: None, + }); } } } @@ -238,19 +243,22 @@ type DrawCustom = ( pub struct DrawMeshInstanced; impl RenderCommand

for DrawMeshInstanced { - type Param = SRes>; + type Param = (SRes>, SRes); type ViewWorldQuery = (); - type ItemWorldQuery = (Read>, Read); + type ItemWorldQuery = Read; #[inline] fn render<'w>( - _item: &P, + item: &P, _view: (), - (mesh_handle, instance_buffer): (&'w Handle, &'w InstanceBuffer), - meshes: SystemParamItem<'w, '_, Self::Param>, + instance_buffer: &'w InstanceBuffer, + (meshes, render_mesh_instances): SystemParamItem<'w, '_, Self::Param>, pass: &mut TrackedRenderPass<'w>, ) -> RenderCommandResult { - let gpu_mesh = match meshes.into_inner().get(mesh_handle) { + let Some(mesh_instance) = render_mesh_instances.get(&item.entity()) else { + return RenderCommandResult::Failure; + }; + let gpu_mesh = match meshes.into_inner().get(mesh_instance.mesh_asset_id) { Some(gpu_mesh) => gpu_mesh, None => return RenderCommandResult::Failure, }; diff --git a/examples/stress_tests/bevymark.rs b/examples/stress_tests/bevymark.rs index cd8c1ad77f3c2..34112ebd7e0c9 100644 --- a/examples/stress_tests/bevymark.rs +++ b/examples/stress_tests/bevymark.rs @@ -97,7 +97,8 @@ fn main() { DefaultPlugins.set(WindowPlugin { primary_window: Some(Window { title: "BevyMark".into(), - resolution: (800., 600.).into(), + resolution: WindowResolution::new(1920.0, 1080.0) + .with_scale_factor_override(1.0), present_mode: PresentMode::AutoNoVsync, ..default() }), diff --git a/examples/stress_tests/many_cubes.rs b/examples/stress_tests/many_cubes.rs index 218757e471c79..a14ad02ff45f6 100644 --- a/examples/stress_tests/many_cubes.rs +++ b/examples/stress_tests/many_cubes.rs @@ -18,6 +18,7 @@ use bevy::{ render::render_resource::{Extent3d, TextureDimension, TextureFormat}, window::{PresentMode, WindowPlugin}, }; +use bevy_internal::window::WindowResolution; use rand::{rngs::StdRng, seq::SliceRandom, Rng, SeedableRng}; #[derive(FromArgs, Resource)] @@ -70,6 +71,7 @@ fn main() { DefaultPlugins.set(WindowPlugin { primary_window: Some(Window { present_mode: PresentMode::AutoNoVsync, + resolution: WindowResolution::default().with_scale_factor_override(1.0), ..default() }), ..default() From c3ef067fe48b15a52d712443bc7e323e0fc400e0 Mon Sep 17 00:00:00 2001 From: Robert Swain Date: Sat, 23 Sep 2023 00:41:33 +0200 Subject: [PATCH 02/14] Fix wireframe and clippy --- crates/bevy_pbr/src/render/mesh.rs | 10 ++++------ crates/bevy_pbr/src/wireframe.rs | 12 ++++++++---- crates/bevy_sprite/src/mesh2d/mesh.rs | 10 ++++------ 3 files changed, 16 insertions(+), 16 deletions(-) diff --git a/crates/bevy_pbr/src/render/mesh.rs b/crates/bevy_pbr/src/render/mesh.rs index 312f6419d33f5..cedc0c8ba3fd3 100644 --- a/crates/bevy_pbr/src/render/mesh.rs +++ b/crates/bevy_pbr/src/render/mesh.rs @@ -682,12 +682,10 @@ impl GetBatchData for MeshPipeline { .expect("Failed to find render mesh instance"); ( (&mesh_instance.transforms).into(), - mesh_instance.automatic_batching.then(|| { - ( - mesh_instance.material_bind_group_id, - mesh_instance.mesh_asset_id, - ) - }), + mesh_instance.automatic_batching.then_some(( + mesh_instance.material_bind_group_id, + mesh_instance.mesh_asset_id, + )), ) } } diff --git a/crates/bevy_pbr/src/wireframe.rs b/crates/bevy_pbr/src/wireframe.rs index 08a20433eb936..009e1b2d7163e 100644 --- a/crates/bevy_pbr/src/wireframe.rs +++ b/crates/bevy_pbr/src/wireframe.rs @@ -22,7 +22,7 @@ use bevy_render::{ view::{ExtractedView, Msaa, VisibleEntities}, RenderApp, RenderSet, }; -use bevy_render::{ExtractSchedule, Render}; +use bevy_render::{Extract, ExtractSchedule, Render}; use bevy_utils::tracing::error; use bevy_utils::PassHashSet; @@ -49,6 +49,7 @@ impl Plugin for WireframePlugin { render_app .add_render_command::() .init_resource::>() + .init_resource::() .add_systems(ExtractSchedule, extract_wireframes) .add_systems(Render, queue_wireframes.in_set(RenderSet::QueueMeshes)); } @@ -73,12 +74,15 @@ pub struct WireframeConfig { pub global: bool, } -#[derive(Resource, Deref, DerefMut)] +#[derive(Resource, Default, Deref, DerefMut)] pub struct Wireframes(PassHashSet); -fn extract_wireframes(mut wireframes: ResMut, query: Query>) { +fn extract_wireframes( + mut wireframes: ResMut, + query: Extract>>, +) { wireframes.clear(); - wireframes.extend(query.into_iter()); + wireframes.extend(&query); } #[derive(Resource, Clone)] diff --git a/crates/bevy_sprite/src/mesh2d/mesh.rs b/crates/bevy_sprite/src/mesh2d/mesh.rs index ef08ba15892a7..8abb2d76fa610 100644 --- a/crates/bevy_sprite/src/mesh2d/mesh.rs +++ b/crates/bevy_sprite/src/mesh2d/mesh.rs @@ -374,12 +374,10 @@ impl GetBatchData for Mesh2dPipeline { .expect("Failed to find render mesh2d instance"); ( (&mesh_instance.transforms).into(), - mesh_instance.automatic_batching.then(|| { - ( - mesh_instance.material_bind_group_id, - mesh_instance.mesh_asset_id, - ) - }), + mesh_instance.automatic_batching.then_some(( + mesh_instance.material_bind_group_id, + mesh_instance.mesh_asset_id, + )), ) } } From ad94257fd7b7ba018b16f5fbbc7fb15127a4460a Mon Sep 17 00:00:00 2001 From: Robert Swain Date: Mon, 25 Sep 2023 22:26:26 +0200 Subject: [PATCH 03/14] Use all 64-bits of the Entity for a hash all the time. Fixes PassHash only using Entity.index. --- crates/bevy_ecs/src/entity/mod.rs | 10 ++++++++-- crates/bevy_utils/src/lib.rs | 5 ----- 2 files changed, 8 insertions(+), 7 deletions(-) diff --git a/crates/bevy_ecs/src/entity/mod.rs b/crates/bevy_ecs/src/entity/mod.rs index 3a49c99aedcf6..c7ec7b8211fe2 100644 --- a/crates/bevy_ecs/src/entity/mod.rs +++ b/crates/bevy_ecs/src/entity/mod.rs @@ -44,7 +44,7 @@ use crate::{ storage::{SparseSetIndex, TableId, TableRow}, }; use serde::{Deserialize, Serialize}; -use std::{convert::TryFrom, fmt, mem, sync::atomic::Ordering}; +use std::{convert::TryFrom, fmt, hash::Hash, mem, sync::atomic::Ordering}; #[cfg(target_has_atomic = "64")] use std::sync::atomic::AtomicI64 as AtomicIdCursor; @@ -115,12 +115,18 @@ type IdCursor = isize; /// [`EntityCommands`]: crate::system::EntityCommands /// [`Query::get`]: crate::system::Query::get /// [`World`]: crate::world::World -#[derive(Clone, Copy, Eq, Hash, Ord, PartialEq, PartialOrd)] +#[derive(Clone, Copy, Eq, Ord, PartialEq, PartialOrd)] pub struct Entity { generation: u32, index: u32, } +impl Hash for Entity { + fn hash(&self, state: &mut H) { + self.to_bits().hash(state); + } +} + pub(crate) enum AllocAtWithoutReplacement { Exists(EntityLocation), DidNotExist, diff --git a/crates/bevy_utils/src/lib.rs b/crates/bevy_utils/src/lib.rs index 714c3dd44b13a..775a10f52f063 100644 --- a/crates/bevy_utils/src/lib.rs +++ b/crates/bevy_utils/src/lib.rs @@ -212,11 +212,6 @@ impl Hasher for PassHasher { panic!("can only hash u64 using PassHasher"); } - #[inline] - fn write_u32(&mut self, i: u32) { - self.hash = i as u64; - } - #[inline] fn write_u64(&mut self, i: u64) { self.hash = i; From 11a16120b77615c756ddec3626838c2e29b4a746 Mon Sep 17 00:00:00 2001 From: Robert Swain Date: Mon, 25 Sep 2023 23:34:35 +0200 Subject: [PATCH 04/14] Move imports --- crates/bevy_pbr/src/render/mesh.rs | 19 ++++++++----------- 1 file changed, 8 insertions(+), 11 deletions(-) diff --git a/crates/bevy_pbr/src/render/mesh.rs b/crates/bevy_pbr/src/render/mesh.rs index bb9eb2343e565..d1d026de977c9 100644 --- a/crates/bevy_pbr/src/render/mesh.rs +++ b/crates/bevy_pbr/src/render/mesh.rs @@ -1,13 +1,8 @@ use crate::{ - environment_map, prepass, - render::{ - morph::{no_automatic_morph_batching, MorphIndices}, - skin::no_automatic_skin_batching, - }, - EnvironmentMapLight, FogMeta, GlobalLightMeta, GpuFog, GpuLights, GpuPointLights, LightMeta, - MaterialBindGroupId, NotShadowCaster, NotShadowReceiver, PreviousGlobalTransform, - ScreenSpaceAmbientOcclusionTextures, Shadow, ShadowSamplers, ViewClusterBindings, - ViewFogUniformOffset, ViewLightsUniformOffset, ViewShadowBindings, + environment_map, prepass, EnvironmentMapLight, FogMeta, GlobalLightMeta, GpuFog, GpuLights, + GpuPointLights, LightMeta, MaterialBindGroupId, NotShadowCaster, NotShadowReceiver, + PreviousGlobalTransform, ScreenSpaceAmbientOcclusionTextures, Shadow, ShadowSamplers, + ViewClusterBindings, ViewFogUniformOffset, ViewLightsUniformOffset, ViewShadowBindings, CLUSTERED_FORWARD_STORAGE_BUFFER_COUNT, MAX_CASCADES_PER_LIGHT, MAX_DIRECTIONAL_LIGHTS, }; use bevy_app::{Plugin, PostUpdate}; @@ -52,8 +47,10 @@ use bevy_transform::components::GlobalTransform; use bevy_utils::{tracing::error, HashMap, Hashed, PassHashMap}; use crate::render::{ - morph::{extract_morphs, prepare_morphs, MorphUniform}, - skin::{extract_skins, prepare_skins, SkinUniform}, + morph::{ + extract_morphs, no_automatic_morph_batching, prepare_morphs, MorphIndices, MorphUniform, + }, + skin::{extract_skins, no_automatic_skin_batching, prepare_skins, SkinUniform}, MeshLayouts, }; From 249b6a3c93685f181bf68eac09caaa28fa5353be Mon Sep 17 00:00:00 2001 From: Robert Swain Date: Mon, 25 Sep 2023 23:44:18 +0200 Subject: [PATCH 05/14] Improve docs --- crates/bevy_render/src/batching/mod.rs | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/crates/bevy_render/src/batching/mod.rs b/crates/bevy_render/src/batching/mod.rs index 6fbb2b46510d5..f7cdf545e61a6 100644 --- a/crates/bevy_render/src/batching/mod.rs +++ b/crates/bevy_render/src/batching/mod.rs @@ -67,8 +67,9 @@ pub trait GetBatchData { /// containing these data for all instances. type BufferData: GpuArrayBufferable + Sync + Send + 'static; /// Get the per-instance data to be inserted into the [`GpuArrayBuffer`]. - /// Get the data used for comparison when deciding whether draws can be - /// batched. + /// If the instance can be batched, also return the data used for + /// comparison when deciding whether draws can be batched, else return None + /// for the CompareData. fn get_batch_data( param: &SystemParamItem, query_item: &QueryItem, From 98b34f6b3f82c9014ccab6af6849ed0bd2ecb8fa Mon Sep 17 00:00:00 2001 From: Robert Swain Date: Tue, 26 Sep 2023 01:29:29 +0200 Subject: [PATCH 06/14] Use Skifire's suggested hash function and rename to EntityHashMap --- crates/bevy_pbr/src/material.rs | 4 +- crates/bevy_pbr/src/render/mesh.rs | 4 +- crates/bevy_pbr/src/render/morph.rs | 4 +- crates/bevy_pbr/src/render/skin.rs | 4 +- crates/bevy_pbr/src/wireframe.rs | 4 +- crates/bevy_sprite/src/mesh2d/material.rs | 4 +- crates/bevy_sprite/src/mesh2d/mesh.rs | 4 +- crates/bevy_sprite/src/render/mod.rs | 4 +- crates/bevy_ui/src/render/mod.rs | 4 +- crates/bevy_utils/src/lib.rs | 48 ++++++++++++++++++++--- 10 files changed, 60 insertions(+), 24 deletions(-) diff --git a/crates/bevy_pbr/src/material.rs b/crates/bevy_pbr/src/material.rs index 3def1cfe463d8..c3287af0d9cbf 100644 --- a/crates/bevy_pbr/src/material.rs +++ b/crates/bevy_pbr/src/material.rs @@ -34,7 +34,7 @@ use bevy_render::{ view::{ExtractedView, Msaa, ViewVisibility, VisibleEntities}, Extract, ExtractSchedule, Render, RenderApp, RenderSet, }; -use bevy_utils::{tracing::error, HashMap, HashSet, PassHashMap}; +use bevy_utils::{tracing::error, EntityHashMap, HashMap, HashSet}; use std::hash::Hash; use std::marker::PhantomData; @@ -373,7 +373,7 @@ impl RenderCommand

for SetMaterial } #[derive(Resource, Deref, DerefMut)] -pub struct RenderMaterialInstances(PassHashMap>); +pub struct RenderMaterialInstances(EntityHashMap>); impl Default for RenderMaterialInstances { fn default() -> Self { diff --git a/crates/bevy_pbr/src/render/mesh.rs b/crates/bevy_pbr/src/render/mesh.rs index d1d026de977c9..83334b68f456e 100644 --- a/crates/bevy_pbr/src/render/mesh.rs +++ b/crates/bevy_pbr/src/render/mesh.rs @@ -44,7 +44,7 @@ use bevy_render::{ Extract, ExtractSchedule, Render, RenderApp, RenderSet, }; use bevy_transform::components::GlobalTransform; -use bevy_utils::{tracing::error, HashMap, Hashed, PassHashMap}; +use bevy_utils::{tracing::error, EntityHashMap, HashMap, Hashed}; use crate::render::{ morph::{ @@ -237,7 +237,7 @@ pub struct RenderMeshInstance { } #[derive(Default, Resource, Deref, DerefMut)] -pub struct RenderMeshInstances(PassHashMap); +pub struct RenderMeshInstances(EntityHashMap); #[derive(Component)] pub struct Mesh3d; diff --git a/crates/bevy_pbr/src/render/morph.rs b/crates/bevy_pbr/src/render/morph.rs index 107e8b5630034..61dfef75d5280 100644 --- a/crates/bevy_pbr/src/render/morph.rs +++ b/crates/bevy_pbr/src/render/morph.rs @@ -10,7 +10,7 @@ use bevy_render::{ view::ViewVisibility, Extract, }; -use bevy_utils::PassHashMap; +use bevy_utils::EntityHashMap; use bytemuck::Pod; #[derive(Component)] @@ -19,7 +19,7 @@ pub struct MorphIndex { } #[derive(Default, Resource, Deref, DerefMut)] -pub struct MorphIndices(PassHashMap); +pub struct MorphIndices(EntityHashMap); #[derive(Resource)] pub struct MorphUniform { diff --git a/crates/bevy_pbr/src/render/skin.rs b/crates/bevy_pbr/src/render/skin.rs index aa3bd028e1ace..bd4e687a8c31f 100644 --- a/crates/bevy_pbr/src/render/skin.rs +++ b/crates/bevy_pbr/src/render/skin.rs @@ -11,7 +11,7 @@ use bevy_render::{ Extract, }; use bevy_transform::prelude::GlobalTransform; -use bevy_utils::PassHashMap; +use bevy_utils::EntityHashMap; /// Maximum number of joints supported for skinned meshes. pub const MAX_JOINTS: usize = 256; @@ -31,7 +31,7 @@ impl SkinIndex { } #[derive(Default, Resource, Deref, DerefMut)] -pub struct SkinIndices(PassHashMap); +pub struct SkinIndices(EntityHashMap); // Notes on implementation: see comment on top of the `extract_skins` system. #[derive(Resource)] diff --git a/crates/bevy_pbr/src/wireframe.rs b/crates/bevy_pbr/src/wireframe.rs index 009e1b2d7163e..9566afffb9242 100644 --- a/crates/bevy_pbr/src/wireframe.rs +++ b/crates/bevy_pbr/src/wireframe.rs @@ -24,7 +24,7 @@ use bevy_render::{ }; use bevy_render::{Extract, ExtractSchedule, Render}; use bevy_utils::tracing::error; -use bevy_utils::PassHashSet; +use bevy_utils::EntityHashSet; pub const WIREFRAME_SHADER_HANDLE: Handle = Handle::weak_from_u128(192598014480025766); @@ -75,7 +75,7 @@ pub struct WireframeConfig { } #[derive(Resource, Default, Deref, DerefMut)] -pub struct Wireframes(PassHashSet); +pub struct Wireframes(EntityHashSet); fn extract_wireframes( mut wireframes: ResMut, diff --git a/crates/bevy_sprite/src/mesh2d/material.rs b/crates/bevy_sprite/src/mesh2d/material.rs index 1398c8ef61348..74f6073066b31 100644 --- a/crates/bevy_sprite/src/mesh2d/material.rs +++ b/crates/bevy_sprite/src/mesh2d/material.rs @@ -29,7 +29,7 @@ use bevy_render::{ Extract, ExtractSchedule, Render, RenderApp, RenderSet, }; use bevy_transform::components::{GlobalTransform, Transform}; -use bevy_utils::{FloatOrd, HashMap, HashSet, PassHashMap}; +use bevy_utils::{EntityHashMap, FloatOrd, HashMap, HashSet}; use std::hash::Hash; use std::marker::PhantomData; @@ -174,7 +174,7 @@ where } #[derive(Resource, Deref, DerefMut)] -pub struct RenderMaterial2dInstances(PassHashMap>); +pub struct RenderMaterial2dInstances(EntityHashMap>); impl Default for RenderMaterial2dInstances { fn default() -> Self { diff --git a/crates/bevy_sprite/src/mesh2d/mesh.rs b/crates/bevy_sprite/src/mesh2d/mesh.rs index 8abb2d76fa610..b5b0fc7d946ba 100644 --- a/crates/bevy_sprite/src/mesh2d/mesh.rs +++ b/crates/bevy_sprite/src/mesh2d/mesh.rs @@ -30,7 +30,7 @@ use bevy_render::{ Extract, ExtractSchedule, Render, RenderApp, RenderSet, }; use bevy_transform::components::GlobalTransform; -use bevy_utils::PassHashMap; +use bevy_utils::EntityHashMap; use crate::Material2dBindGroupId; @@ -192,7 +192,7 @@ pub struct RenderMesh2dInstance { } #[derive(Default, Resource, Deref, DerefMut)] -pub struct RenderMesh2dInstances(PassHashMap); +pub struct RenderMesh2dInstances(EntityHashMap); #[derive(Component)] pub struct Mesh2d; diff --git a/crates/bevy_sprite/src/render/mod.rs b/crates/bevy_sprite/src/render/mod.rs index a035b0a7b9efa..c7d79dcc088e8 100644 --- a/crates/bevy_sprite/src/render/mod.rs +++ b/crates/bevy_sprite/src/render/mod.rs @@ -33,7 +33,7 @@ use bevy_render::{ Extract, }; use bevy_transform::components::GlobalTransform; -use bevy_utils::{FloatOrd, HashMap, PassHashMap}; +use bevy_utils::{EntityHashMap, FloatOrd, HashMap}; use bytemuck::{Pod, Zeroable}; use fixedbitset::FixedBitSet; @@ -329,7 +329,7 @@ pub struct ExtractedSprite { #[derive(Resource, Default)] pub struct ExtractedSprites { - pub sprites: PassHashMap, + pub sprites: EntityHashMap, } #[derive(Resource, Default)] diff --git a/crates/bevy_ui/src/render/mod.rs b/crates/bevy_ui/src/render/mod.rs index 32af20d704b37..f962298b9b1d9 100644 --- a/crates/bevy_ui/src/render/mod.rs +++ b/crates/bevy_ui/src/render/mod.rs @@ -35,7 +35,7 @@ use bevy_sprite::{SpriteAssetEvents, TextureAtlas}; #[cfg(feature = "bevy_text")] use bevy_text::{PositionedGlyph, Text, TextLayoutInfo}; use bevy_transform::components::GlobalTransform; -use bevy_utils::{FloatOrd, HashMap, PassHashMap}; +use bevy_utils::{EntityHashMap, FloatOrd, HashMap}; use bytemuck::{Pod, Zeroable}; use std::ops::Range; @@ -163,7 +163,7 @@ pub struct ExtractedUiNode { #[derive(Resource, Default)] pub struct ExtractedUiNodes { - pub uinodes: PassHashMap, + pub uinodes: EntityHashMap, } pub fn extract_atlas_uinodes( diff --git a/crates/bevy_utils/src/lib.rs b/crates/bevy_utils/src/lib.rs index 775a10f52f063..7dc03e39fba88 100644 --- a/crates/bevy_utils/src/lib.rs +++ b/crates/bevy_utils/src/lib.rs @@ -223,12 +223,6 @@ impl Hasher for PassHasher { } } -/// A [`HashMap`] pre-configured to use [`PassHash`] passthrough hashing. -pub type PassHashMap = hashbrown::HashMap; - -/// A [`HashSet`] pre-configured to use [`PassHash`] passthrough hashing. -pub type PassHashSet = hashbrown::HashSet; - /// A [`HashMap`] pre-configured to use [`Hashed`] keys and [`PassHash`] passthrough hashing. pub type PreHashMap = hashbrown::HashMap, V, PassHash>; @@ -256,6 +250,48 @@ impl PreHashMapExt for PreHashMap Self::Hasher { + EntityHasher::default() + } +} + +/// A very fast hash that is only designed to work on generational indices +/// like [`Entity`]. It will panic if attempting to hash a type containing +/// non-u64 fields. +#[derive(Debug, Default)] +pub struct EntityHasher { + hash: u64, +} + +impl Hasher for EntityHasher { + fn write(&mut self, _bytes: &[u8]) { + panic!("can only hash u64 using EntityHasher"); + } + + #[inline] + fn write_u64(&mut self, i: u64) { + self.hash = i | (i.wrapping_mul(0x517cc1b727220a95) << 32); + } + + #[inline] + fn finish(&self) -> u64 { + self.hash + } +} + +/// A [`HashMap`] pre-configured to use [`EntityHash`] hashing. +pub type EntityHashMap = hashbrown::HashMap; + +/// A [`HashSet`] pre-configured to use [`EntityHash`] hashing. +pub type EntityHashSet = hashbrown::HashSet; + /// A type which calls a function when dropped. /// This can be used to ensure that cleanup code is run even in case of a panic. /// From df0502c14dd262fbe4fc9b5da255507125af2ec3 Mon Sep 17 00:00:00 2001 From: Robert Swain Date: Tue, 26 Sep 2023 01:46:42 +0200 Subject: [PATCH 07/14] Use the old capacity method for allocating the vec for mesh extraction --- crates/bevy_pbr/src/render/mesh.rs | 5 +++-- crates/bevy_sprite/src/mesh2d/mesh.rs | 5 +++-- 2 files changed, 6 insertions(+), 4 deletions(-) diff --git a/crates/bevy_pbr/src/render/mesh.rs b/crates/bevy_pbr/src/render/mesh.rs index 83334b68f456e..583475bf9a2e2 100644 --- a/crates/bevy_pbr/src/render/mesh.rs +++ b/crates/bevy_pbr/src/render/mesh.rs @@ -244,6 +244,7 @@ pub struct Mesh3d; pub fn extract_meshes( mut commands: Commands, + mut previous_len: Local, mut render_mesh_instances: ResMut, meshes_query: Extract< Query<( @@ -258,9 +259,8 @@ pub fn extract_meshes( )>, >, ) { - let capacity = meshes_query.iter().len(); render_mesh_instances.clear(); - let mut entities = Vec::with_capacity(capacity); + let mut entities = Vec::with_capacity(*previous_len); let visible_meshes = meshes_query.iter().filter(|(_, vis, ..)| vis.get()); @@ -304,6 +304,7 @@ pub fn extract_meshes( }, ); } + *previous_len = entities.len(); commands.insert_or_spawn_batch(entities); } diff --git a/crates/bevy_sprite/src/mesh2d/mesh.rs b/crates/bevy_sprite/src/mesh2d/mesh.rs index b5b0fc7d946ba..1639b7df97f8d 100644 --- a/crates/bevy_sprite/src/mesh2d/mesh.rs +++ b/crates/bevy_sprite/src/mesh2d/mesh.rs @@ -199,6 +199,7 @@ pub struct Mesh2d; pub fn extract_mesh2d( mut commands: Commands, + mut previous_len: Local, mut render_mesh_instances: ResMut, query: Extract< Query<( @@ -210,9 +211,8 @@ pub fn extract_mesh2d( )>, >, ) { - let capacity = query.iter().len(); render_mesh_instances.clear(); - let mut entities = Vec::with_capacity(capacity); + let mut entities = Vec::with_capacity(*previous_len); for (entity, view_visibility, transform, handle, no_automatic_batching) in &query { if !view_visibility.get() { @@ -234,6 +234,7 @@ pub fn extract_mesh2d( }, ); } + *previous_len = entities.len(); commands.insert_or_spawn_batch(entities); } From 903d5abb7c04e5f594781ffa3a53926b9ca02c8a Mon Sep 17 00:00:00 2001 From: Robert Swain Date: Tue, 26 Sep 2023 20:18:08 +0200 Subject: [PATCH 08/14] Document the magic constant --- crates/bevy_utils/src/lib.rs | 12 +++++++++++- 1 file changed, 11 insertions(+), 1 deletion(-) diff --git a/crates/bevy_utils/src/lib.rs b/crates/bevy_utils/src/lib.rs index 7dc03e39fba88..c954523c68663 100644 --- a/crates/bevy_utils/src/lib.rs +++ b/crates/bevy_utils/src/lib.rs @@ -270,6 +270,12 @@ pub struct EntityHasher { hash: u64, } +// This value comes from rustc-hash (also known as FxHasher) which in turn got +// it from Firefox. It is something like `u64::MAX / N` for an N that gives a +// value close to π and works well for distributing bits for hashing when using +// with a wrapping multiplication. +const FRAC_U64MAX_PI: u64 = 0x517cc1b727220a95; + impl Hasher for EntityHasher { fn write(&mut self, _bytes: &[u8]) { panic!("can only hash u64 using EntityHasher"); @@ -277,7 +283,11 @@ impl Hasher for EntityHasher { #[inline] fn write_u64(&mut self, i: u64) { - self.hash = i | (i.wrapping_mul(0x517cc1b727220a95) << 32); + // Apparently hashbrown's hashmap uses the upper 7 bits for some SIMD + // optimisation that uses those bits for binning. This hash function + // was faster than i | (i << (64 - 7)) in the worst cases, and was + // faster than PassHasher for all cases tested. + self.hash = i | (i.wrapping_mul(FRAC_U64MAX_PI) << 32); } #[inline] From 91afe4d3c44a761769268910e60f48a2fdb0fa28 Mon Sep 17 00:00:00 2001 From: Robert Swain Date: Tue, 26 Sep 2023 20:28:55 +0200 Subject: [PATCH 09/14] CI documentation fix --- crates/bevy_render/src/batching/mod.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/crates/bevy_render/src/batching/mod.rs b/crates/bevy_render/src/batching/mod.rs index f7cdf545e61a6..16633b77b8a4f 100644 --- a/crates/bevy_render/src/batching/mod.rs +++ b/crates/bevy_render/src/batching/mod.rs @@ -69,7 +69,7 @@ pub trait GetBatchData { /// Get the per-instance data to be inserted into the [`GpuArrayBuffer`]. /// If the instance can be batched, also return the data used for /// comparison when deciding whether draws can be batched, else return None - /// for the CompareData. + /// for the `CompareData`. fn get_batch_data( param: &SystemParamItem, query_item: &QueryItem, From 07aae971696d022775108a2b6e36c577098dc762 Mon Sep 17 00:00:00 2001 From: Robert Swain Date: Tue, 26 Sep 2023 20:32:17 +0200 Subject: [PATCH 10/14] CI fixes --- crates/bevy_utils/src/lib.rs | 2 +- examples/shader/shader_instancing.rs | 5 +++-- examples/stress_tests/many_cubes.rs | 6 +++--- 3 files changed, 7 insertions(+), 6 deletions(-) diff --git a/crates/bevy_utils/src/lib.rs b/crates/bevy_utils/src/lib.rs index c954523c68663..85e70e60bc541 100644 --- a/crates/bevy_utils/src/lib.rs +++ b/crates/bevy_utils/src/lib.rs @@ -263,7 +263,7 @@ impl BuildHasher for EntityHash { } /// A very fast hash that is only designed to work on generational indices -/// like [`Entity`]. It will panic if attempting to hash a type containing +/// like `Entity`. It will panic if attempting to hash a type containing /// non-u64 fields. #[derive(Debug, Default)] pub struct EntityHasher { diff --git a/examples/shader/shader_instancing.rs b/examples/shader/shader_instancing.rs index 6d68f15ff2590..fc18bb63c73b7 100644 --- a/examples/shader/shader_instancing.rs +++ b/examples/shader/shader_instancing.rs @@ -6,7 +6,9 @@ use bevy::{ query::QueryItem, system::{lifetimeless::*, SystemParamItem}, }, - pbr::{MeshPipeline, MeshPipelineKey, SetMeshBindGroup, SetMeshViewBindGroup}, + pbr::{ + MeshPipeline, MeshPipelineKey, RenderMeshInstances, SetMeshBindGroup, SetMeshViewBindGroup, + }, prelude::*, render::{ extract_component::{ExtractComponent, ExtractComponentPlugin}, @@ -22,7 +24,6 @@ use bevy::{ Render, RenderApp, RenderSet, }, }; -use bevy_internal::pbr::RenderMeshInstances; use bytemuck::{Pod, Zeroable}; fn main() { diff --git a/examples/stress_tests/many_cubes.rs b/examples/stress_tests/many_cubes.rs index a14ad02ff45f6..740b9bbcc7659 100644 --- a/examples/stress_tests/many_cubes.rs +++ b/examples/stress_tests/many_cubes.rs @@ -16,9 +16,8 @@ use bevy::{ math::{DVec2, DVec3}, prelude::*, render::render_resource::{Extent3d, TextureDimension, TextureFormat}, - window::{PresentMode, WindowPlugin}, + window::{PresentMode, WindowPlugin, WindowResolution}, }; -use bevy_internal::window::WindowResolution; use rand::{rngs::StdRng, seq::SliceRandom, Rng, SeedableRng}; #[derive(FromArgs, Resource)] @@ -71,7 +70,8 @@ fn main() { DefaultPlugins.set(WindowPlugin { primary_window: Some(Window { present_mode: PresentMode::AutoNoVsync, - resolution: WindowResolution::default().with_scale_factor_override(1.0), + resolution: WindowResolution::new(1920.0, 1080.0) + .with_scale_factor_override(1.0), ..default() }), ..default() From eac5b46c7990d10864f77af355842a0eb45b7a57 Mon Sep 17 00:00:00 2001 From: Robert Swain Date: Tue, 26 Sep 2023 20:40:21 +0200 Subject: [PATCH 11/14] Revert ExtractComponent impl for Handle to using clone_weak() instead of id() --- crates/bevy_asset/src/id.rs | 2 +- crates/bevy_render/src/extract_component.rs | 6 +++--- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/crates/bevy_asset/src/id.rs b/crates/bevy_asset/src/id.rs index e499057bdee34..b89a0afcba90f 100644 --- a/crates/bevy_asset/src/id.rs +++ b/crates/bevy_asset/src/id.rs @@ -14,7 +14,7 @@ use std::{ /// For an identifier tied to the lifetime of an asset, see [`Handle`]. /// /// For an "untyped" / "generic-less" id, see [`UntypedAssetId`]. -#[derive(Component, Reflect)] +#[derive(Reflect)] pub enum AssetId { /// A small / efficient runtime identifier that can be used to efficiently look up an asset stored in [`Assets`]. This is /// the "default" identifier used for assets. The alternative(s) (ex: [`AssetId::Uuid`]) will only be used if assets are diff --git a/crates/bevy_render/src/extract_component.rs b/crates/bevy_render/src/extract_component.rs index a855d52862ccc..4c6e4f8553928 100644 --- a/crates/bevy_render/src/extract_component.rs +++ b/crates/bevy_render/src/extract_component.rs @@ -5,7 +5,7 @@ use crate::{ Extract, ExtractSchedule, Render, RenderApp, RenderSet, }; use bevy_app::{App, Plugin}; -use bevy_asset::{Asset, AssetId, Handle}; +use bevy_asset::{Asset, Handle}; use bevy_ecs::{ component::Component, prelude::*, @@ -197,11 +197,11 @@ impl Plugin for ExtractComponentPlugin { impl ExtractComponent for Handle { type Query = Read>; type Filter = (); - type Out = AssetId; + type Out = Handle; #[inline] fn extract_component(handle: QueryItem<'_, Self::Query>) -> Option { - Some(handle.id()) + Some(handle.clone_weak()) } } From 43e4752368a67e06f54cc4f4f148e3d15e261293 Mon Sep 17 00:00:00 2001 From: Robert Swain Date: Tue, 26 Sep 2023 20:43:48 +0200 Subject: [PATCH 12/14] Remove unnecessary import --- crates/bevy_asset/src/id.rs | 1 - 1 file changed, 1 deletion(-) diff --git a/crates/bevy_asset/src/id.rs b/crates/bevy_asset/src/id.rs index b89a0afcba90f..428b992ca0742 100644 --- a/crates/bevy_asset/src/id.rs +++ b/crates/bevy_asset/src/id.rs @@ -1,5 +1,4 @@ use crate::{Asset, AssetIndex, Handle, UntypedHandle}; -use bevy_ecs::component::Component; use bevy_reflect::{Reflect, Uuid}; use std::{ any::TypeId, From ef26d87085a8851ef11e401d8539c9f172af5235 Mon Sep 17 00:00:00 2001 From: Robert Swain Date: Wed, 27 Sep 2023 07:47:24 +0200 Subject: [PATCH 13/14] Clear SkinIndices before extracting Co-authored-by: robtfm <50659922+robtfm@users.noreply.github.com> --- crates/bevy_pbr/src/render/skin.rs | 1 + 1 file changed, 1 insertion(+) diff --git a/crates/bevy_pbr/src/render/skin.rs b/crates/bevy_pbr/src/render/skin.rs index bd4e687a8c31f..bfb12fd794427 100644 --- a/crates/bevy_pbr/src/render/skin.rs +++ b/crates/bevy_pbr/src/render/skin.rs @@ -95,6 +95,7 @@ pub fn extract_skins( joints: Extract>, ) { uniform.buffer.clear(); + skin_indices.clear(); let mut last_start = 0; // PERF: This can be expensive, can we move this to prepare? From 566189573fd1203100f62da843b46c6d516157e4 Mon Sep 17 00:00:00 2001 From: Robert Swain Date: Wed, 27 Sep 2023 07:45:54 +0200 Subject: [PATCH 14/14] Inline Entity hash function --- crates/bevy_ecs/src/entity/mod.rs | 1 + 1 file changed, 1 insertion(+) diff --git a/crates/bevy_ecs/src/entity/mod.rs b/crates/bevy_ecs/src/entity/mod.rs index c7ec7b8211fe2..677477680f10c 100644 --- a/crates/bevy_ecs/src/entity/mod.rs +++ b/crates/bevy_ecs/src/entity/mod.rs @@ -122,6 +122,7 @@ pub struct Entity { } impl Hash for Entity { + #[inline] fn hash(&self, state: &mut H) { self.to_bits().hash(state); }