diff --git a/client/src/graphics/voxels/mod.rs b/client/src/graphics/voxels/mod.rs index a76a45fb..eebf58ba 100644 --- a/client/src/graphics/voxels/mod.rs +++ b/client/src/graphics/voxels/mod.rs @@ -16,8 +16,8 @@ use crate::{ Config, Loader, Sim, }; use common::{ - dodeca, dodeca::Vertex, + dodeca::gridding::BOUNDING_SPHERE_RADIUS, graph::NodeId, lru_slab::SlotId, math, @@ -142,7 +142,7 @@ impl Voxels { for &(node, ref node_transform) in &nodes { let node_to_view = local_to_view * node_transform; let origin = node_to_view * math::origin(); - if !frustum_planes.contain(&origin, dodeca::BOUNDING_SPHERE_RADIUS) { + if !frustum_planes.contain(&origin, BOUNDING_SPHERE_RADIUS) { // Don't bother generating or drawing chunks from nodes that are wholly outside the // frustum. continue; diff --git a/common/src/cursor.rs b/common/src/cursor.rs index 61c8c753..6ddf392c 100644 --- a/common/src/cursor.rs +++ b/common/src/cursor.rs @@ -1,6 +1,7 @@ use std::sync::OnceLock; -use crate::dodeca::{Side, Vertex, SIDE_COUNT}; +use crate::dodeca::{Side, Vertex}; +use crate::dodeca::gridding::SIDE_COUNT; use crate::graph::{Graph, NodeId}; use crate::node::ChunkId; diff --git a/common/src/dodeca.rs b/common/src/dodeca.rs index 3c82f712..0178b9b1 100644 --- a/common/src/dodeca.rs +++ b/common/src/dodeca.rs @@ -1,9 +1,7 @@ //! Tools for processing the geometry of a right dodecahedron -use std::sync::OnceLock; - use serde::{Deserialize, Serialize}; -use crate::math; +use crate::dodeca::gridding::*; /// Sides of a right dodecahedron #[derive(Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Hash, Serialize, Deserialize)] @@ -223,231 +221,239 @@ impl Vertex { } } -pub const VERTEX_COUNT: usize = 20; -pub const SIDE_COUNT: usize = 12; -pub const BOUNDING_SPHERE_RADIUS_F64: f64 = 1.2264568712514068; -pub const BOUNDING_SPHERE_RADIUS: f32 = BOUNDING_SPHERE_RADIUS_F64 as f32; - -/// Whether two sides share an edge -fn adjacent() -> &'static [[bool; SIDE_COUNT]; SIDE_COUNT] { - static LOCK: OnceLock<[[bool; SIDE_COUNT]; SIDE_COUNT]> = OnceLock::new(); - LOCK.get_or_init(|| { - let mut result = [[false; SIDE_COUNT]; SIDE_COUNT]; - for (i, side) in result.iter_mut().enumerate() { - for (j, is_adjacent) in side.iter_mut().enumerate() { - let cosh_distance = (reflections_f64()[i] * reflections_f64()[j])[(3, 3)]; - // Possile cosh_distances: 1, 4.23606 = 2+sqrt(5), 9.47213 = 5+2*sqrt(5), 12.70820 = 6+3*sqrt(5); - // < 2.0 indicates identical faces; < 5.0 indicates adjacent faces; > 5.0 indicates non-adjacent faces - *is_adjacent = (2.0..5.0).contains(&cosh_distance); +pub mod gridding { + use std::sync::OnceLock; + + use crate::math; + use crate::dodeca::{Side, Vertex}; + + pub const VERTEX_COUNT: usize = 20; + pub const SIDE_COUNT: usize = 12; + pub const BOUNDING_SPHERE_RADIUS_F64: f64 = 1.2264568712514068; + pub const BOUNDING_SPHERE_RADIUS: f32 = BOUNDING_SPHERE_RADIUS_F64 as f32; + + /// Whether two sides share an edge + pub fn adjacent() -> &'static [[bool; SIDE_COUNT]; SIDE_COUNT] { + static LOCK: OnceLock<[[bool; SIDE_COUNT]; SIDE_COUNT]> = OnceLock::new(); + LOCK.get_or_init(|| { + let mut result = [[false; SIDE_COUNT]; SIDE_COUNT]; + for (i, side) in result.iter_mut().enumerate() { + for (j, is_adjacent) in side.iter_mut().enumerate() { + let cosh_distance = (reflections_f64()[i] * reflections_f64()[j])[(3, 3)]; + // Possile cosh_distances: 1, 4.23606 = 2+sqrt(5), 9.47213 = 5+2*sqrt(5), 12.70820 = 6+3*sqrt(5); + // < 2.0 indicates identical faces; < 5.0 indicates adjacent faces; > 5.0 indicates non-adjacent faces + *is_adjacent = (2.0..5.0).contains(&cosh_distance); + } } - } - result - }) -} + result + }) + } -/// Vector corresponding to the outer normal of each side -fn side_normals_f64() -> &'static [na::Vector4; SIDE_COUNT] { - static LOCK: OnceLock<[na::Vector4; SIDE_COUNT]> = OnceLock::new(); - LOCK.get_or_init(|| { - let phi = libm::sqrt(1.25) + 0.5; // golden ratio - let f = math::lorentz_normalize(&na::Vector4::new(1.0, phi, 0.0, libm::sqrt(phi))); - - let mut result: [na::Vector4; SIDE_COUNT] = [na::zero(); SIDE_COUNT]; - let mut i = 0; - for (x, y, z, w) in [ - (f.x, f.y, f.z, f.w), - (-f.x, f.y, -f.z, f.w), - (f.x, -f.y, -f.z, f.w), - (-f.x, -f.y, f.z, f.w), - ] { - for (x, y, z, w) in [(x, y, z, w), (y, z, x, w), (z, x, y, w)] { - result[i] = na::Vector4::new(x, y, z, w); - i += 1; + /// Vector corresponding to the outer normal of each side + pub fn side_normals_f64() -> &'static [na::Vector4; SIDE_COUNT] { + static LOCK: OnceLock<[na::Vector4; SIDE_COUNT]> = OnceLock::new(); + LOCK.get_or_init(|| { + let phi = libm::sqrt(1.25) + 0.5; // golden ratio + let f = math::lorentz_normalize(&na::Vector4::new(1.0, phi, 0.0, libm::sqrt(phi))); + + let mut result: [na::Vector4; SIDE_COUNT] = [na::zero(); SIDE_COUNT]; + let mut i = 0; + for (x, y, z, w) in [ + (f.x, f.y, f.z, f.w), + (-f.x, f.y, -f.z, f.w), + (f.x, -f.y, -f.z, f.w), + (-f.x, -f.y, f.z, f.w), + ] { + for (x, y, z, w) in [(x, y, z, w), (y, z, x, w), (z, x, y, w)] { + result[i] = na::Vector4::new(x, y, z, w); + i += 1; + } } - } - result - }) -} - -/// Transform that moves from a neighbor to a reference node, for each side -fn reflections_f64() -> &'static [na::Matrix4; SIDE_COUNT] { - static LOCK: OnceLock<[na::Matrix4; SIDE_COUNT]> = OnceLock::new(); - LOCK.get_or_init(|| side_normals_f64().map(|r| math::reflect(&r))) -} + result + }) + } -/// Sides incident to a vertex, in canonical order -fn vertex_sides() -> &'static [[Side; 3]; VERTEX_COUNT] { - static LOCK: OnceLock<[[Side; 3]; VERTEX_COUNT]> = OnceLock::new(); - LOCK.get_or_init(|| { - let mut result = [[Side::A; 3]; VERTEX_COUNT]; - let mut vertex = 0; - // Kind of a hack, but working this out by hand isn't any fun. - for a in 0..SIDE_COUNT { - for b in (a + 1)..SIDE_COUNT { - for c in (b + 1)..SIDE_COUNT { - if !adjacent()[a][b] || !adjacent()[b][c] || !adjacent()[c][a] { - continue; + /// Transform that moves from a neighbor to a reference node, for each side + pub fn reflections_f64() -> &'static [na::Matrix4; SIDE_COUNT] { + static LOCK: OnceLock<[na::Matrix4; SIDE_COUNT]> = OnceLock::new(); + LOCK.get_or_init(|| side_normals_f64().map(|r| math::reflect(&r))) + } + + /// Sides incident to a vertex, in canonical order + pub fn vertex_sides() -> &'static [[Side; 3]; VERTEX_COUNT] { + static LOCK: OnceLock<[[Side; 3]; VERTEX_COUNT]> = OnceLock::new(); + LOCK.get_or_init(|| { + let mut result = [[Side::A; 3]; VERTEX_COUNT]; + let mut vertex = 0; + // Kind of a hack, but working this out by hand isn't any fun. + for a in 0..SIDE_COUNT { + for b in (a + 1)..SIDE_COUNT { + for c in (b + 1)..SIDE_COUNT { + if !adjacent()[a][b] || !adjacent()[b][c] || !adjacent()[c][a] { + continue; + } + result[vertex] = [ + Side::from_index(a), + Side::from_index(b), + Side::from_index(c), + ]; + vertex += 1; } - result[vertex] = [ - Side::from_index(a), - Side::from_index(b), - Side::from_index(c), - ]; - vertex += 1; } } - } - assert_eq!(vertex, 20); - result - }) -} + assert_eq!(vertex, 20); + result + }) + } -// Which vertices are adjacent to other vertices and opposite the canonical sides -fn adjacent_vertices() -> &'static [[Vertex; 3]; VERTEX_COUNT] { - static LOCK: OnceLock<[[Vertex; 3]; VERTEX_COUNT]> = OnceLock::new(); - LOCK.get_or_init(|| { - let mut result = [[Vertex::A; 3]; VERTEX_COUNT]; - - for (i, triple) in result.iter_mut().enumerate() { - for result_index in 0..3 { - let mut test_sides = vertex_sides()[i]; - // Keep modifying the result_index'th element of test_sides until its three elements are all - // adjacent to a single vertex. That vertex is the vertex we're looking for. - for side in Side::iter() { - if side == vertex_sides()[i][result_index] { - continue; - } - test_sides[result_index] = side; - if let Some(adjacent_vertex) = - Vertex::from_sides(test_sides[0], test_sides[1], test_sides[2]) - { - triple[result_index] = adjacent_vertex; + // Which vertices are adjacent to other vertices and opposite the canonical sides + pub fn adjacent_vertices() -> &'static [[Vertex; 3]; VERTEX_COUNT] { + static LOCK: OnceLock<[[Vertex; 3]; VERTEX_COUNT]> = OnceLock::new(); + LOCK.get_or_init(|| { + let mut result = [[Vertex::A; 3]; VERTEX_COUNT]; + + for (i, triple) in result.iter_mut().enumerate() { + for result_index in 0..3 { + let mut test_sides = vertex_sides()[i]; + // Keep modifying the result_index'th element of test_sides until its three elements are all + // adjacent to a single vertex. That vertex is the vertex we're looking for. + for side in Side::iter() { + if side == vertex_sides()[i][result_index] { + continue; + } + test_sides[result_index] = side; + if let Some(adjacent_vertex) = + Vertex::from_sides(test_sides[0], test_sides[1], test_sides[2]) + { + triple[result_index] = adjacent_vertex; + } } } } - } - result - }) -} - -/// Transform that converts from cube-centric coordinates to dodeca-centric coordinates -fn dual_to_node_f64() -> &'static [na::Matrix4; VERTEX_COUNT] { - static LOCK: OnceLock<[na::Matrix4; VERTEX_COUNT]> = OnceLock::new(); - LOCK.get_or_init(|| { - let mip_origin_normal = math::mip(&math::origin(), &side_normals_f64()[0]); // This value is the same for every side - let mut result = [na::zero(); VERTEX_COUNT]; - for (i, map) in result.iter_mut().enumerate() { - let [a, b, c] = vertex_sides()[i]; - let vertex_position = math::lorentz_normalize( - &(math::origin() - - (a.normal_f64() + b.normal_f64() + c.normal_f64()) * mip_origin_normal), - ); - *map = na::Matrix4::from_columns(&[ - -a.normal_f64(), - -b.normal_f64(), - -c.normal_f64(), - vertex_position, - ]); - } - result - }) -} + result + }) + } -/// Transform that converts from dodeca-centric coordinates to cube-centric coordinates -fn node_to_dual_f64() -> &'static [na::Matrix4; VERTEX_COUNT] { - static LOCK: OnceLock<[na::Matrix4; VERTEX_COUNT]> = OnceLock::new(); - LOCK.get_or_init(|| dual_to_node_f64().map(|m| math::mtranspose(&m))) -} + /// Transform that converts from cube-centric coordinates to dodeca-centric coordinates + pub fn dual_to_node_f64() -> &'static [na::Matrix4; VERTEX_COUNT] { + static LOCK: OnceLock<[na::Matrix4; VERTEX_COUNT]> = OnceLock::new(); + LOCK.get_or_init(|| { + let mip_origin_normal = math::mip(&math::origin(), &side_normals_f64()[0]); // This value is the same for every side + let mut result = [na::zero(); VERTEX_COUNT]; + for (i, map) in result.iter_mut().enumerate() { + let [a, b, c] = vertex_sides()[i]; + let vertex_position = math::lorentz_normalize( + &(math::origin() + - (a.normal_f64() + b.normal_f64() + c.normal_f64()) * mip_origin_normal), + ); + *map = na::Matrix4::from_columns(&[ + -a.normal_f64(), + -b.normal_f64(), + -c.normal_f64(), + vertex_position, + ]); + } + result + }) + } -fn dual_to_chunk_factor_f64() -> f64 { - static LOCK: OnceLock = OnceLock::new(); - *LOCK.get_or_init(|| (2.0 + 5.0f64.sqrt()).sqrt()) -} + /// Transform that converts from dodeca-centric coordinates to cube-centric coordinates + pub fn node_to_dual_f64() -> &'static [na::Matrix4; VERTEX_COUNT] { + static LOCK: OnceLock<[na::Matrix4; VERTEX_COUNT]> = OnceLock::new(); + LOCK.get_or_init(|| dual_to_node_f64().map(|m| math::mtranspose(&m))) + } -fn chunk_to_dual_factor_f64() -> f64 { - static LOCK: OnceLock = OnceLock::new(); - *LOCK.get_or_init(|| 1.0 / dual_to_chunk_factor_f64()) -} + pub fn dual_to_chunk_factor_f64() -> f64 { + static LOCK: OnceLock = OnceLock::new(); + *LOCK.get_or_init(|| (2.0 + 5.0f64.sqrt()).sqrt()) + } -/// Vertex shared by 3 sides -fn sides_to_vertex() -> &'static [[[Option; SIDE_COUNT]; SIDE_COUNT]; SIDE_COUNT] { - static LOCK: OnceLock<[[[Option; SIDE_COUNT]; SIDE_COUNT]; SIDE_COUNT]> = - OnceLock::new(); - LOCK.get_or_init(|| { - let mut result = [[[None; SIDE_COUNT]; SIDE_COUNT]; SIDE_COUNT]; - let mut vertex = Vertex::iter(); - // Kind of a hack, but working this out by hand isn't any fun. - for a in 0..SIDE_COUNT { - for b in (a + 1)..SIDE_COUNT { - for c in (b + 1)..SIDE_COUNT { - if !Side::from_index(a).adjacent_to(Side::from_index(b)) - || !Side::from_index(b).adjacent_to(Side::from_index(c)) - || !Side::from_index(c).adjacent_to(Side::from_index(a)) - { - continue; + pub fn chunk_to_dual_factor_f64() -> f64 { + static LOCK: OnceLock = OnceLock::new(); + *LOCK.get_or_init(|| 1.0 / dual_to_chunk_factor_f64()) + } + + /// Vertex shared by 3 sides + pub fn sides_to_vertex() -> &'static [[[Option; SIDE_COUNT]; SIDE_COUNT]; SIDE_COUNT] { + static LOCK: OnceLock<[[[Option; SIDE_COUNT]; SIDE_COUNT]; SIDE_COUNT]> = + OnceLock::new(); + LOCK.get_or_init(|| { + let mut result = [[[None; SIDE_COUNT]; SIDE_COUNT]; SIDE_COUNT]; + let mut vertex = Vertex::iter(); + // Kind of a hack, but working this out by hand isn't any fun. + for a in 0..SIDE_COUNT { + for b in (a + 1)..SIDE_COUNT { + for c in (b + 1)..SIDE_COUNT { + if !Side::from_index(a).adjacent_to(Side::from_index(b)) + || !Side::from_index(b).adjacent_to(Side::from_index(c)) + || !Side::from_index(c).adjacent_to(Side::from_index(a)) + { + continue; + } + let v = Some(vertex.next().unwrap()); + result[a][b][c] = v; + result[a][c][b] = v; + result[b][a][c] = v; + result[b][c][a] = v; + result[c][a][b] = v; + result[c][b][a] = v; } - let v = Some(vertex.next().unwrap()); - result[a][b][c] = v; - result[a][c][b] = v; - result[b][a][c] = v; - result[b][c][a] = v; - result[c][a][b] = v; - result[c][b][a] = v; } } - } - assert_eq!(vertex.next(), None); - result - }) -} + assert_eq!(vertex.next(), None); + result + }) + } -/// Whether the determinant of the cube-to-node transform is negative -fn chunk_to_node_parity() -> &'static [bool; VERTEX_COUNT] { - static LOCK: OnceLock<[bool; VERTEX_COUNT]> = OnceLock::new(); - LOCK.get_or_init(|| { - let mut result = [false; VERTEX_COUNT]; + /// Whether the determinant of the cube-to-node transform is negative + pub fn chunk_to_node_parity() -> &'static [bool; VERTEX_COUNT] { + static LOCK: OnceLock<[bool; VERTEX_COUNT]> = OnceLock::new(); + LOCK.get_or_init(|| { + let mut result = [false; VERTEX_COUNT]; - for v in Vertex::iter() { - result[v as usize] = math::parity(&v.chunk_to_node_f64()); - } + for v in Vertex::iter() { + result[v as usize] = math::parity(&v.chunk_to_node_f64()); + } - result - }) -} + result + }) + } -fn side_normals_f32() -> &'static [na::Vector4; SIDE_COUNT] { - static LOCK: OnceLock<[na::Vector4; SIDE_COUNT]> = OnceLock::new(); - LOCK.get_or_init(|| side_normals_f64().map(|n| n.cast())) -} + pub fn side_normals_f32() -> &'static [na::Vector4; SIDE_COUNT] { + static LOCK: OnceLock<[na::Vector4; SIDE_COUNT]> = OnceLock::new(); + LOCK.get_or_init(|| side_normals_f64().map(|n| n.cast())) + } -fn reflections_f32() -> &'static [na::Matrix4; SIDE_COUNT] { - static LOCK: OnceLock<[na::Matrix4; SIDE_COUNT]> = OnceLock::new(); - LOCK.get_or_init(|| reflections_f64().map(|n| n.cast())) -} + pub fn reflections_f32() -> &'static [na::Matrix4; SIDE_COUNT] { + static LOCK: OnceLock<[na::Matrix4; SIDE_COUNT]> = OnceLock::new(); + LOCK.get_or_init(|| reflections_f64().map(|n| n.cast())) + } -fn dual_to_node_f32() -> &'static [na::Matrix4; VERTEX_COUNT] { - static LOCK: OnceLock<[na::Matrix4; VERTEX_COUNT]> = OnceLock::new(); - LOCK.get_or_init(|| dual_to_node_f64().map(|n| n.cast())) -} + pub fn dual_to_node_f32() -> &'static [na::Matrix4; VERTEX_COUNT] { + static LOCK: OnceLock<[na::Matrix4; VERTEX_COUNT]> = OnceLock::new(); + LOCK.get_or_init(|| dual_to_node_f64().map(|n| n.cast())) + } -fn node_to_dual_f32() -> &'static [na::Matrix4; VERTEX_COUNT] { - static LOCK: OnceLock<[na::Matrix4; VERTEX_COUNT]> = OnceLock::new(); - LOCK.get_or_init(|| node_to_dual_f64().map(|n| n.cast())) -} + pub fn node_to_dual_f32() -> &'static [na::Matrix4; VERTEX_COUNT] { + static LOCK: OnceLock<[na::Matrix4; VERTEX_COUNT]> = OnceLock::new(); + LOCK.get_or_init(|| node_to_dual_f64().map(|n| n.cast())) + } -fn dual_to_chunk_factor_f32() -> f32 { - static LOCK: OnceLock = OnceLock::new(); - *LOCK.get_or_init(|| dual_to_chunk_factor_f64() as f32) -} + pub fn dual_to_chunk_factor_f32() -> f32 { + static LOCK: OnceLock = OnceLock::new(); + *LOCK.get_or_init(|| dual_to_chunk_factor_f64() as f32) + } -fn chunk_to_dual_factor_f32() -> f32 { - static LOCK: OnceLock = OnceLock::new(); - *LOCK.get_or_init(|| chunk_to_dual_factor_f64() as f32) + pub fn chunk_to_dual_factor_f32() -> f32 { + static LOCK: OnceLock = OnceLock::new(); + *LOCK.get_or_init(|| chunk_to_dual_factor_f64() as f32) + } } #[cfg(test)] mod tests { use super::*; + use crate::math; use approx::*; #[test] diff --git a/common/src/graph.rs b/common/src/graph.rs index 7fcc628d..e1ff7b37 100644 --- a/common/src/graph.rs +++ b/common/src/graph.rs @@ -7,9 +7,10 @@ use fxhash::{FxHashMap, FxHashSet}; use serde::{Deserialize, Serialize}; use crate::{ - dodeca::{Side, SIDE_COUNT}, + dodeca::Side, math, node::{ChunkId, ChunkLayout, Node}, + dodeca::gridding::SIDE_COUNT }; /// Graph of the right dodecahedral tiling of H^3 diff --git a/common/src/traversal.rs b/common/src/traversal.rs index cbd26da9..9cf40343 100644 --- a/common/src/traversal.rs +++ b/common/src/traversal.rs @@ -4,7 +4,8 @@ use fxhash::FxHashSet; use crate::{ collision_math::Ray, - dodeca::{self, Side, Vertex}, + dodeca::{Side, Vertex}, + dodeca::gridding::BOUNDING_SPHERE_RADIUS, graph::{Graph, NodeId}, math, node::ChunkId, @@ -185,7 +186,7 @@ impl<'a> RayTraverser<'a> { // because the AABB check can have false positives. let ray_node_distance = (next_node_transform * self.ray.position).w.acosh(); let ray_length = tanh_distance.atanh(); - if ray_node_distance - ray_length - self.radius > dodeca::BOUNDING_SPHERE_RADIUS + if ray_node_distance - ray_length - self.radius > BOUNDING_SPHERE_RADIUS { // Ray cannot intersect node continue; diff --git a/server/src/sim.rs b/server/src/sim.rs index 3f4f3b99..e84dc2ad 100644 --- a/server/src/sim.rs +++ b/server/src/sim.rs @@ -296,7 +296,7 @@ impl Sim { // We want to load all chunks that a player can interact with in a single step, so chunk_generation_distance // is set up to cover that distance. - let chunk_generation_distance = dodeca::BOUNDING_SPHERE_RADIUS + let chunk_generation_distance = common::dodeca::gridding::BOUNDING_SPHERE_RADIUS + self.cfg.character.character_radius + self.cfg.character.speed_cap * self.cfg.step_interval.as_secs_f32() + self.cfg.character.ground_distance_tolerance