Instrument all par joins, defer ubiquitous pos/vel writes with a component instead of using vecs

This commit is contained in:
Imbris 2021-03-16 03:40:31 -04:00
parent 767686b579
commit ac03f818d7
5 changed files with 153 additions and 75 deletions

View File

@ -78,8 +78,8 @@ pub use self::{
misc::Object, misc::Object,
ori::Ori, ori::Ori,
phys::{ phys::{
Collider, ForceUpdate, Gravity, Mass, PhysicsState, Pos, PreviousPhysCache, Scale, Sticky, Collider, ForceUpdate, Gravity, Mass, PhysicsState, Pos, PosVelDefer, PreviousPhysCache,
Vel, Scale, Sticky, Vel,
}, },
player::Player, player::Player,
poise::{Poise, PoiseChange, PoiseSource, PoiseState}, poise::{Poise, PoiseChange, PoiseSource, PoiseState},

View File

@ -4,19 +4,34 @@ use specs::{Component, DerefFlaggedStorage, NullStorage};
use specs_idvs::IdvStorage; use specs_idvs::IdvStorage;
use vek::*; use vek::*;
// Position /// Position
#[derive(Copy, Clone, Default, Debug, PartialEq, Serialize, Deserialize)] #[derive(Copy, Clone, Default, Debug, PartialEq, Serialize, Deserialize)]
pub struct Pos(pub Vec3<f32>); pub struct Pos(pub Vec3<f32>);
impl Component for Pos { impl Component for Pos {
// TODO: why not regular vec storage????
// TODO: component occupancy metrics
type Storage = IdvStorage<Self>; type Storage = IdvStorage<Self>;
} }
// Velocity /// Velocity
#[derive(Copy, Clone, Default, Debug, PartialEq, Serialize, Deserialize)] #[derive(Copy, Clone, Default, Debug, PartialEq, Serialize, Deserialize)]
pub struct Vel(pub Vec3<f32>); pub struct Vel(pub Vec3<f32>);
impl Component for Vel { impl Component for Vel {
// TODO: why not regular vec storage????
type Storage = IdvStorage<Self>;
}
/// Used to defer writes to Pos/Vel in nested join loops
#[derive(Copy, Clone, Debug)]
pub struct PosVelDefer {
pub pos: Pos,
pub vel: Vel,
}
impl Component for PosVelDefer {
// TODO: why not regular vec storage????
type Storage = IdvStorage<Self>; type Storage = IdvStorage<Self>;
} }
@ -37,6 +52,7 @@ pub struct PreviousPhysCache {
} }
impl Component for PreviousPhysCache { impl Component for PreviousPhysCache {
// TODO: why not regular vec storage????
type Storage = IdvStorage<Self>; type Storage = IdvStorage<Self>;
} }

View File

@ -2,6 +2,7 @@ use common::{
comp::{Ori, Pos, Vel}, comp::{Ori, Pos, Vel},
resources::{PlayerEntity, Time}, resources::{PlayerEntity, Time},
}; };
use common_base::prof_span;
use common_ecs::{Job, Origin, Phase, System}; use common_ecs::{Job, Origin, Phase, System};
use common_net::sync::InterpolatableComponent; use common_net::sync::InterpolatableComponent;
use specs::{ use specs::{
@ -44,20 +45,38 @@ impl<'a> System<'a> for InterpolationSystem {
) )
.par_join() .par_join()
.filter(|(e, _, _, _)| Some(e) != player.as_ref()) .filter(|(e, _, _, _)| Some(e) != player.as_ref())
.for_each(|(_, pos, interp, vel)| { .for_each_init(
*pos = pos.interpolate(interp, time, vel); || {
}); prof_span!(guard, "interpolate pos rayon job");
guard
},
|_guard, (_, pos, interp, vel)| {
*pos = pos.interpolate(interp, time, vel);
},
);
(&data.entities, &mut data.vel, &data.vel_interpdata) (&data.entities, &mut data.vel, &data.vel_interpdata)
.par_join() .par_join()
.filter(|(e, _, _)| Some(e) != player.as_ref()) .filter(|(e, _, _)| Some(e) != player.as_ref())
.for_each(|(_, vel, interp)| { .for_each_init(
*vel = vel.interpolate(interp, time, &()); || {
}); prof_span!(guard, "interpolate vel rayon job");
guard
},
|_guard, (_, vel, interp)| {
*vel = vel.interpolate(interp, time, &());
},
);
(&data.entities, &mut data.ori, &data.ori_interpdata) (&data.entities, &mut data.ori, &data.ori_interpdata)
.par_join() .par_join()
.filter(|(e, _, _)| Some(e) != player.as_ref()) .filter(|(e, _, _)| Some(e) != player.as_ref())
.for_each(|(_, ori, interp)| { .for_each_init(
*ori = ori.interpolate(interp, time, &()); || {
}); prof_span!(guard, "interpolate ori rayon job");
guard
},
|_guard, (_, ori, interp)| {
*ori = ori.interpolate(interp, time, &());
},
);
} }
} }

View File

@ -5,8 +5,8 @@ use spatial_grid::SpatialGrid;
use common::{ use common::{
comp::{ comp::{
body::ship::figuredata::VOXEL_COLLIDER_MANIFEST, BeamSegment, Body, CharacterState, body::ship::figuredata::VOXEL_COLLIDER_MANIFEST, BeamSegment, Body, CharacterState,
Collider, Gravity, Mass, Mounting, Ori, PhysicsState, Pos, PreviousPhysCache, Projectile, Collider, Gravity, Mass, Mounting, Ori, PhysicsState, Pos, PosVelDefer, PreviousPhysCache,
Scale, Shockwave, Sticky, Vel, Projectile, Scale, Shockwave, Sticky, Vel,
}, },
consts::{FRIC_GROUND, GRAVITY}, consts::{FRIC_GROUND, GRAVITY},
event::{EventBus, ServerEvent}, event::{EventBus, ServerEvent},
@ -17,7 +17,6 @@ use common::{
}; };
use common_base::{prof_span, span}; use common_base::{prof_span, span};
use common_ecs::{Job, Origin, ParMode, Phase, PhysicsMetrics, System}; use common_ecs::{Job, Origin, ParMode, Phase, PhysicsMetrics, System};
use hashbrown::HashMap;
use rayon::iter::ParallelIterator; use rayon::iter::ParallelIterator;
use specs::{ use specs::{
shred::{ResourceId, World}, shred::{ResourceId, World},
@ -99,6 +98,7 @@ pub struct PhysicsWrite<'a> {
physics_states: WriteStorage<'a, PhysicsState>, physics_states: WriteStorage<'a, PhysicsState>,
positions: WriteStorage<'a, Pos>, positions: WriteStorage<'a, Pos>,
velocities: WriteStorage<'a, Vel>, velocities: WriteStorage<'a, Vel>,
pos_vel_defers: WriteStorage<'a, PosVelDefer>,
orientations: WriteStorage<'a, Ori>, orientations: WriteStorage<'a, Ori>,
previous_phys_cache: WriteStorage<'a, PreviousPhysCache>, previous_phys_cache: WriteStorage<'a, PreviousPhysCache>,
} }
@ -238,7 +238,6 @@ impl<'a> PhysicsData<'a> {
} }
fn apply_pushback(&mut self, job: &mut Job<Sys>, spatial_grid: &SpatialGrid) { fn apply_pushback(&mut self, job: &mut Job<Sys>, spatial_grid: &SpatialGrid) {
// TODO: make sure to check git stash show -p to make sure nothing was missed
span!(_guard, "Apply pushback"); span!(_guard, "Apply pushback");
job.cpu_stats.measure(ParMode::Rayon); job.cpu_stats.measure(ParMode::Rayon);
let PhysicsData { let PhysicsData {
@ -357,7 +356,7 @@ impl<'a> PhysicsData<'a> {
let mass_other = mass_other let mass_other = mass_other
.map(|m| m.0) .map(|m| m.0)
.unwrap_or(previous_cache_other.scale); .unwrap_or(previous_cache_other.scale);
//This check after the pos check, as we currently don't have // This check after the pos check, as we currently don't have
// that many // that many
// massless entites [citation needed] // massless entites [citation needed]
if mass_other == 0.0 { if mass_other == 0.0 {
@ -453,6 +452,31 @@ impl<'a> PhysicsData<'a> {
ref read, ref read,
ref mut write, ref mut write,
} = self; } = self;
prof_span!(guard, "insert PosVelDefer");
// NOTE: keep in sync with join below
(
&read.entities,
read.colliders.mask(),
&write.positions,
&write.velocities,
write.orientations.mask(),
write.physics_states.mask(),
!&write.pos_vel_defers, // This is the one we are adding
write.previous_phys_cache.mask(),
!&read.mountings,
)
.join()
.map(|t| (t.0, *t.2, *t.3))
.collect::<Vec<_>>()
.into_iter()
.for_each(|(entity, pos, vel)| {
let _ = write
.pos_vel_defers
.insert(entity, PosVelDefer { pos, vel });
});
drop(guard);
// Apply movement inputs // Apply movement inputs
span!(guard, "Apply movement and terrain collision"); span!(guard, "Apply movement and terrain collision");
let (positions, velocities, previous_phys_cache, orientations) = ( let (positions, velocities, previous_phys_cache, orientations) = (
@ -473,14 +497,19 @@ impl<'a> PhysicsData<'a> {
!&read.mountings, !&read.mountings,
) )
.par_join() .par_join()
.for_each(|(entity, pos, vel, physics_state, _)| { .for_each_init(
let in_loaded_chunk = read || {
.terrain prof_span!(guard, "velocity update rayon job");
.get_key(read.terrain.pos_key(pos.0.map(|e| e.floor() as i32))) guard
.is_some(); },
// Integrate forces |_guard, (entity, pos, vel, physics_state, _)| {
// Friction is assumed to be a constant dependent on location let in_loaded_chunk = read
let friction = if physics_state.on_ground { 0.0 } else { FRIC_AIR } .terrain
.get_key(read.terrain.pos_key(pos.0.map(|e| e.floor() as i32)))
.is_some();
// Integrate forces
// Friction is assumed to be a constant dependent on location
let friction = if physics_state.on_ground { 0.0 } else { FRIC_AIR }
// .max(if physics_state.on_ground { // .max(if physics_state.on_ground {
// FRIC_GROUND // FRIC_GROUND
// } else { // } else {
@ -491,26 +520,27 @@ impl<'a> PhysicsData<'a> {
} else { } else {
0.0 0.0
}); });
let downward_force = let downward_force =
if !in_loaded_chunk { if !in_loaded_chunk {
0.0 // No gravity in unloaded chunks 0.0 // No gravity in unloaded chunks
} else if physics_state } else if physics_state
.in_liquid .in_liquid
.map(|depth| depth > 0.75) .map(|depth| depth > 0.75)
.unwrap_or(false) .unwrap_or(false)
{ {
(1.0 - BOUYANCY) * GRAVITY (1.0 - BOUYANCY) * GRAVITY
} else { } else {
GRAVITY GRAVITY
} * read.gravities.get(entity).map(|g| g.0).unwrap_or_default(); } * read.gravities.get(entity).map(|g| g.0).unwrap_or_default();
vel.0 = integrate_forces(read.dt.0, vel.0, downward_force, friction); vel.0 = integrate_forces(read.dt.0, vel.0, downward_force, friction);
}); },
);
let velocities = &write.velocities; let velocities = &write.velocities;
// Second pass: resolve collisions // Second pass: resolve collisions
let (pos_writes, vel_writes, land_on_grounds) = ( let land_on_grounds = (
&read.entities, &read.entities,
read.scales.maybe(), read.scales.maybe(),
read.stickies.maybe(), read.stickies.maybe(),
@ -521,13 +551,17 @@ impl<'a> PhysicsData<'a> {
read.bodies.maybe(), read.bodies.maybe(),
read.character_states.maybe(), read.character_states.maybe(),
&mut write.physics_states, &mut write.physics_states,
&mut write.pos_vel_defers,
previous_phys_cache, previous_phys_cache,
!&read.mountings, !&read.mountings,
) )
.par_join() .par_join()
.fold( .map_init(
|| (Vec::new(), Vec::new(), Vec::new()), || {
|(mut pos_writes, mut vel_writes, mut land_on_grounds), prof_span!(guard, "physics e<>t rayon job");
guard
},
|_guard,
( (
entity, entity,
scale, scale,
@ -539,16 +573,18 @@ impl<'a> PhysicsData<'a> {
body, body,
character_state, character_state,
mut physics_state, mut physics_state,
pos_vel_defer,
_previous_cache, _previous_cache,
_, _,
)| { )| {
let mut land_on_ground = None;
// Defer the writes of positions and velocities to allow an inner loop over // Defer the writes of positions and velocities to allow an inner loop over
// terrain-like entities // terrain-like entities
let mut vel = *vel; let mut vel = *vel;
let old_vel = vel;
if sticky.is_some() && physics_state.on_surface().is_some() { if sticky.is_some() && physics_state.on_surface().is_some() {
vel.0 = physics_state.ground_vel; vel.0 = physics_state.ground_vel;
return (pos_writes, vel_writes, land_on_grounds); return land_on_ground;
} }
let scale = if let Collider::Voxel { .. } = collider { let scale = if let Collider::Voxel { .. } = collider {
@ -615,7 +651,7 @@ impl<'a> PhysicsData<'a> {
was_on_ground, was_on_ground,
block_snap, block_snap,
climbing, climbing,
|entity, vel| land_on_grounds.push((entity, vel)), |entity, vel| land_on_ground = Some((entity, vel)),
); );
tgt_pos = cpos.0; tgt_pos = cpos.0;
}, },
@ -644,7 +680,7 @@ impl<'a> PhysicsData<'a> {
was_on_ground, was_on_ground,
block_snap, block_snap,
climbing, climbing,
|entity, vel| land_on_grounds.push((entity, vel)), |entity, vel| land_on_ground = Some((entity, vel)),
); );
tgt_pos = cpos.0; tgt_pos = cpos.0;
}, },
@ -806,8 +842,8 @@ impl<'a> PhysicsData<'a> {
block_snap, block_snap,
climbing, climbing,
|entity, vel| { |entity, vel| {
land_on_grounds land_on_ground =
.push((entity, Vel(ori_from.mul_direction(vel.0)))) Some((entity, Vel(ori_from.mul_direction(vel.0))));
}, },
); );
@ -841,48 +877,52 @@ impl<'a> PhysicsData<'a> {
} }
} }
if tgt_pos != pos.0 { *pos_vel_defer = PosVelDefer {
pos_writes.push((entity, Pos(tgt_pos))); pos: Pos(tgt_pos),
} vel,
if vel != old_vel { };
vel_writes.push((entity, vel));
}
(pos_writes, vel_writes, land_on_grounds) land_on_ground
},
)
.fold(
|| Vec::new(),
|mut land_on_grounds, land_on_ground| {
land_on_ground.map(|log| land_on_grounds.push(log));
land_on_grounds
}, },
) )
.reduce( .reduce(
|| (Vec::new(), Vec::new(), Vec::new()), || Vec::new(),
|(mut pos_writes_a, mut vel_writes_a, mut land_on_grounds_a), |mut land_on_grounds_a, mut land_on_grounds_b| {
(mut pos_writes_b, mut vel_writes_b, mut land_on_grounds_b)| {
pos_writes_a.append(&mut pos_writes_b);
vel_writes_a.append(&mut vel_writes_b);
land_on_grounds_a.append(&mut land_on_grounds_b); land_on_grounds_a.append(&mut land_on_grounds_b);
(pos_writes_a, vel_writes_a, land_on_grounds_a) land_on_grounds_a
}, },
); );
drop(guard); drop(guard);
job.cpu_stats.measure(ParMode::Single); job.cpu_stats.measure(ParMode::Single);
let pos_writes: HashMap<Entity, Pos> = pos_writes.into_iter().collect(); prof_span!(guard, "write deferred pos and vel");
let vel_writes: HashMap<Entity, Vel> = vel_writes.into_iter().collect(); for (_, pos, vel, pos_vel_defer) in (
for (entity, pos, vel) in &read.entities,
(&read.entities, &mut write.positions, &mut write.velocities).join() &mut write.positions,
&mut write.velocities,
&write.pos_vel_defers,
)
.join()
{ {
if let Some(new_pos) = pos_writes.get(&entity) { *pos = pos_vel_defer.pos;
*pos = *new_pos; *vel = pos_vel_defer.vel;
}
if let Some(new_vel) = vel_writes.get(&entity) {
*vel = *new_vel;
}
} }
drop(guard);
prof_span!(guard, "record ori into phys_cache");
for (ori, previous_phys_cache) in for (ori, previous_phys_cache) in
(&write.orientations, &mut write.previous_phys_cache).join() (&write.orientations, &mut write.previous_phys_cache).join()
{ {
previous_phys_cache.ori = ori.to_quat(); previous_phys_cache.ori = ori.to_quat();
} }
drop(guard);
let mut event_emitter = read.event_bus.emitter(); let mut event_emitter = read.event_bus.emitter();
land_on_grounds.into_iter().for_each(|(entity, vel)| { land_on_grounds.into_iter().for_each(|(entity, vel)| {

View File

@ -161,6 +161,10 @@ impl State {
ecs.register::<comp::Ori>(); ecs.register::<comp::Ori>();
ecs.register::<comp::Inventory>(); ecs.register::<comp::Inventory>();
// Register common unsynced components
ecs.register::<comp::PreviousPhysCache>();
ecs.register::<comp::PosVelDefer>();
// Register client-local components // Register client-local components
// TODO: only register on the client // TODO: only register on the client
ecs.register::<comp::LightAnimation>(); ecs.register::<comp::LightAnimation>();
@ -188,7 +192,6 @@ impl State {
ecs.register::<comp::invite::Invite>(); ecs.register::<comp::invite::Invite>();
ecs.register::<comp::invite::PendingInvites>(); ecs.register::<comp::invite::PendingInvites>();
ecs.register::<comp::Beam>(); ecs.register::<comp::Beam>();
ecs.register::<comp::PreviousPhysCache>();
// Register synced resources used by the ECS. // Register synced resources used by the ECS.
ecs.insert(TimeOfDay(0.0)); ecs.insert(TimeOfDay(0.0));