Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Unify RenderLayers and TargetCamera (Part 1: RenderGroups) [ADOPT ME] #12502

Open
wants to merge 54 commits into
base: main
Choose a base branch
from
Open
Changes from 1 commit
Commits
Show all changes
54 commits
Select commit Hold shift + click to select a range
faba920
add RenderGroups (WIP)
UkoeHB Mar 15, 2024
756191b
typo
UkoeHB Mar 15, 2024
0d302b1
integration WIP
UkoeHB Mar 16, 2024
ac31198
propagation algorithm WIP
UkoeHB Mar 17, 2024
1b7c495
propagation algorithm WIP
UkoeHB Mar 17, 2024
bd468dc
propagation algorithm WIP
UkoeHB Mar 18, 2024
481a801
propagation algorithm early draft
UkoeHB Mar 18, 2024
7702347
render groups compiles
UkoeHB Mar 18, 2024
646c53b
propagate render groups compiles
UkoeHB Mar 18, 2024
f07b7a5
fmt
UkoeHB Mar 18, 2024
36d7aeb
integration (WIP) compiles
UkoeHB Mar 18, 2024
a4563ac
fixes for render groups usage
UkoeHB Mar 18, 2024
d622489
rearrange
UkoeHB Mar 18, 2024
4b4bed3
cleanup
UkoeHB Mar 18, 2024
829be8a
fmt
UkoeHB Mar 18, 2024
064d276
eliminate spurious allocations
UkoeHB Mar 18, 2024
a70bd8c
finish removing old RenderLayers
UkoeHB Mar 18, 2024
13994a3
cleanup
UkoeHB Mar 18, 2024
bc289e1
docs
UkoeHB Mar 18, 2024
7e8f073
fmt
UkoeHB Mar 18, 2024
ab44847
cleanup
UkoeHB Mar 19, 2024
1a7a4a2
clean up rendering pt.1
UkoeHB Mar 19, 2024
cc7ac69
add comment
UkoeHB Mar 19, 2024
dc8c7ba
refactor PBR lighting so light filtering occurs on the CPU
UkoeHB Mar 19, 2024
187b8c0
fmt
UkoeHB Mar 19, 2024
3a5ce6f
review comments
UkoeHB Mar 19, 2024
1b6e31b
lints
UkoeHB Mar 19, 2024
dab2d88
lints
UkoeHB Mar 19, 2024
a07d7dc
cleanup
UkoeHB Mar 20, 2024
98d6883
fix query access error
UkoeHB Mar 20, 2024
57a7702
add upper bounds on RenderLayer
UkoeHB Mar 20, 2024
6f90f28
add docs
UkoeHB Mar 20, 2024
4614a5d
rebase and fix debug_overlay
UkoeHB Mar 20, 2024
7ac9b04
CI fixes
UkoeHB Mar 20, 2024
d8a63c1
CI complaints
UkoeHB Mar 20, 2024
c82492b
more CI
UkoeHB Mar 20, 2024
35b8dca
CI again
UkoeHB Mar 20, 2024
75b3be2
don't use default RenderGroups when propagating
UkoeHB Mar 20, 2024
5c40a7f
API cleanup
UkoeHB Mar 20, 2024
64ce22e
fix directional lights not being applied properly
UkoeHB Mar 21, 2024
a646c14
Render groups example (#2)
viridia Mar 21, 2024
52870ee
update directional light intersections with entities so if a camera s…
UkoeHB Mar 21, 2024
6d9a32d
update examples template
UkoeHB Mar 21, 2024
b82a084
clippy
UkoeHB Mar 21, 2024
858b88f
add floor to example:
UkoeHB Mar 21, 2024
962d722
fix shadows appearing erroneously for point and spot lights when ligh…
UkoeHB Mar 22, 2024
266b048
cleanup example
UkoeHB Mar 22, 2024
62c04aa
fmt
UkoeHB Mar 22, 2024
6b4a2d3
add second camera to render groups example
UkoeHB Mar 23, 2024
19bacda
simplify: RenderGroups -> RenderLayers, CameraView -> CameraLayer, li…
UkoeHB Mar 26, 2024
48dc2ee
fix examples
UkoeHB Mar 26, 2024
fb9d417
try to fix render_to_texture example
UkoeHB Mar 26, 2024
9c826d8
fmt
UkoeHB Mar 26, 2024
6f3909c
remove unsafe blocker from bevy_pbr
UkoeHB Apr 1, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
fix shadows appearing erroneously for point and spot lights when ligh…
…ts can see a mesh but the camera can't see the lights
  • Loading branch information
UkoeHB committed Apr 1, 2024

Verified

This commit was signed with the committer’s verified signature.
Robbepop Robin Freyler
commit 962d722a9935cdb934e46ce9a47b231466c20f86
88 changes: 54 additions & 34 deletions crates/bevy_pbr/src/light/mod.rs
Original file line number Diff line number Diff line change
@@ -15,7 +15,7 @@ use bevy_render::{
renderer::RenderDevice,
view::{
derive_render_groups, derive_render_groups_ptr, extract_camera_view, CameraView,
InheritedRenderGroups, InheritedVisibility, RenderGroups, RenderGroupsPtr, RenderLayers,
InheritedRenderGroups, InheritedVisibility, RenderGroups, RenderGroupsPtr, RenderGroupsRef, RenderLayers,
ViewVisibility, VisibleEntities,
},
};
@@ -1835,8 +1835,8 @@ pub fn update_spot_light_frusta(

#[allow(clippy::too_many_arguments)]
pub fn check_light_mesh_visibility(
mut aggregate_view_entities: Local<EntityHashSet>,
mut aggregate_view_layers: Local<RenderLayers>,
mut aggregate_view_entities: Local<EntityHashSet>,
visible_point_lights: Query<&VisiblePointLights>,
views: Query<(Entity, Option<&CameraView>), With<Camera>>,
mut point_lights: Query<(
@@ -1879,6 +1879,47 @@ pub fn check_light_mesh_visibility(
(Without<NotShadowCaster>, Without<DirectionalLight>),
>,
) {
// We use camera-aggregates to determine cascade visibility because if any camera can
// see both a light and an entity, then it will use the light to illuminate the entity.
// We want illuminated entities to cast shadows as expected, even if an entity and a light technically
// don't 'see' each other. Only what the camera sees matters.
let set_aggregate_view = |
aggregate_view_layers: &mut RenderLayers,
aggregate_view_entities: &mut EntityHashSet,
light_groups: RenderGroupsRef<'_>,
| {
aggregate_view_layers.clear();
aggregate_view_entities.clear();
for (camera_entity, maybe_view) in views.iter() {
let default_layers = RenderLayers::default();
let view_layers = maybe_view.map(|v| v.layers()).unwrap_or(&default_layers);

if !light_groups.intersects_parts(Some(camera_entity), view_layers) {
continue;
}

aggregate_view_layers.merge(view_layers);
aggregate_view_entities.insert(camera_entity);
}
};

fn aggregate_view_intersects(
aggregate_view_layers: &RenderLayers,
aggregate_view_entities: &EntityHashSet,
entity_groups: RenderGroupsRef<'_>,
) -> bool {
if aggregate_view_layers.intersects(entity_groups.get().layers()) {
return true;
}

let Some(camera_affiliation) = entity_groups.camera() else {
return false;
};

// Only do a hashmap lookup if the entity has a camera affiliation.
aggregate_view_entities.contains(&camera_affiliation)
}

fn shrink_entities(visible_entities: &mut VisibleEntities) {
// Check that visible entities capacity() is no more than two times greater than len()
let capacity = visible_entities.entities.capacity();
@@ -1934,25 +1975,8 @@ pub fn check_light_mesh_visibility(
}

// Get the aggregate render groups for all cameras that can see this directional light.
// - We use the camera-aggregate to determine cascade visibility because if any camera can
// see both a directional light and an entity, then it will use the light to illuminate the entity.
// We want illuminated entities to cast shadows as expected, even if an entity and a light technically
// don't 'see' each other - only what the camera sees matters.
aggregate_view_entities.clear();
aggregate_view_layers.clear();
let view_mask = derive_render_groups(maybe_vm_inherited, maybe_view_mask);

for (camera_entity, maybe_view) in views.iter() {
let default_layers = RenderLayers::default();
let view_layers = maybe_view.map(|v| v.layers()).unwrap_or(&default_layers);

if !view_mask.intersects_parts(Some(camera_entity), view_layers) {
continue;
}

aggregate_view_entities.insert(camera_entity);
aggregate_view_layers.merge(view_layers);
}
set_aggregate_view(&mut aggregate_view_layers, &mut aggregate_view_entities, view_mask);

for (
entity,
@@ -1972,14 +1996,8 @@ pub fn check_light_mesh_visibility(
// treat it as a normal entity. The CameraView component controls what the camera can see, while
// RenderGroups on the camera entity controls who can see the camera entity.
let derived_mask = derive_render_groups(maybe_em_inherited, maybe_entity_mask);
if !aggregate_view_layers.intersects(derived_mask.get().layers()) {
// Only do a hashmap lookup if the entity has a camera affiliation.
if !derived_mask
.camera()
.map_or(false, |c| aggregate_view_entities.contains(&c))
{
continue;
}
if !aggregate_view_intersects(&aggregate_view_layers, &aggregate_view_entities, derived_mask) {
continue;
}

// If we have an aabb and transform, do frustum culling
@@ -2044,6 +2062,8 @@ pub fn check_light_mesh_visibility(
}

let view_mask = derive_render_groups(maybe_vm_inherited, maybe_view_mask);
set_aggregate_view(&mut aggregate_view_layers, &mut aggregate_view_entities, view_mask);

let light_sphere = Sphere {
center: Vec3A::from(transform.translation()),
radius: point_light.range,
@@ -2063,9 +2083,8 @@ pub fn check_light_mesh_visibility(
continue;
}

if !view_mask
.intersects(&derive_render_groups(maybe_em_inherited, maybe_entity_mask))
{
let derived_mask = derive_render_groups(maybe_em_inherited, maybe_entity_mask);
if !aggregate_view_intersects(&aggregate_view_layers, &aggregate_view_entities, derived_mask) {
continue;
}

@@ -2117,6 +2136,8 @@ pub fn check_light_mesh_visibility(
}

let view_mask = derive_render_groups(maybe_vm_inherited, maybe_view_mask);
set_aggregate_view(&mut aggregate_view_layers, &mut aggregate_view_entities, view_mask);

let light_sphere = Sphere {
center: Vec3A::from(transform.translation()),
radius: point_light.range,
@@ -2136,9 +2157,8 @@ pub fn check_light_mesh_visibility(
continue;
}

if !view_mask
.intersects(&derive_render_groups(maybe_em_inherited, maybe_entity_mask))
{
let derived_mask = derive_render_groups(maybe_em_inherited, maybe_entity_mask);
if !aggregate_view_intersects(&aggregate_view_layers, &aggregate_view_entities, derived_mask) {
continue;
}

26 changes: 23 additions & 3 deletions examples/3d/render_groups.rs
Original file line number Diff line number Diff line change
@@ -37,7 +37,7 @@ fn setup(
specular_map: asset_server.load("environment_maps/pisa_specular_rgb9e5_zstd.ktx2"),
intensity: 1500.0,
},
CameraView::from_layers(&[0, 1, 2, 3, 4, 5, 6]),
CameraView::from_layers(&[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]),
));

// Plane
@@ -51,7 +51,8 @@ fn setup(
commands.spawn((
TextBundle::from_section(
"Press '1..3' to toggle mesh render layers\n\
Press '4..6' to toggle directional light render layers",
Press '4..6' to toggle directional light render layers\n\
Press '1 and 7' to toggle the spot light render layers",
TextStyle {
font_size: 20.,
..default()
@@ -91,7 +92,7 @@ fn setup(
transform: Transform::from_xyz(4.0, 25.0, 8.0).looking_at(Vec3::ZERO, Vec3::Y),
directional_light: DirectionalLight {
shadows_enabled: true,
illuminance: 100000.0,
illuminance: 100_000.0,
color: (*color).into(),
..default()
},
@@ -100,6 +101,22 @@ fn setup(
RenderGroups::from_layer(i + 4),
));
}

// Spawn a spot light that is in the same layer as mesh 1.
// - Notice that the mesh does not cast a shadow when the camera can see the light but not the mesh.
commands.spawn((
SpotLightBundle {
transform: Transform::from_xyz(- 3.0, 2.0, 2.0).looking_at(Vec3::ZERO, Vec3::Y),
spot_light: SpotLight {
shadows_enabled: true,
intensity: 10_000_000.0,
color: palettes::basic::LIME.into(),
..default()
},
..default()
},
RenderGroups::from_layers(&[1, 7]),
));
}

fn toggle_layers(mut query_camera: Query<&mut CameraView>, keyboard: Res<ButtonInput<KeyCode>>) {
@@ -125,6 +142,9 @@ fn toggle_layers(mut query_camera: Query<&mut CameraView>, keyboard: Res<ButtonI
if keyboard.just_pressed(KeyCode::Digit6) {
toggle_camera_layer(&mut camera_view, 6);
}
if keyboard.just_pressed(KeyCode::Digit7) {
toggle_camera_layer(&mut camera_view, 7);
}
}

fn toggle_camera_layer(camera_view: &mut CameraView, layer: usize) {