use crate::{ vol::{BaseVol, ReadVol, SampleVol, VolSize, WriteVol}, volumes::dyna::DynaErr, }; use hashbrown::{hash_map, HashMap}; use std::{fmt::Debug, marker::PhantomData, sync::Arc}; use vek::*; #[derive(Debug)] pub enum VolMap3dErr { NoSuchChunk, ChunkErr(V::Err), DynaErr(DynaErr), InvalidChunkSize, } // V = Voxel // S = Size (replace with a const when const generics is a thing) // M = Chunk metadata #[derive(Clone)] pub struct VolMap3d { chunks: HashMap, Arc>, phantom: PhantomData, } impl VolMap3d { #[inline(always)] pub fn chunk_key(pos: Vec3) -> Vec3 { pos.map2(S::SIZE, |e, sz| { // Horrid, but it's faster than a cheetah with a red bull blood transfusion let log2 = (sz - 1).count_ones(); ((((i64::from(e) + (1 << 32)) as u64) >> log2) - (1 << (32 - log2))) as i32 }) } #[inline(always)] pub fn chunk_offs(pos: Vec3) -> Vec3 { pos.map2(S::SIZE, |e, sz| { // Horrid, but it's even faster than the aforementioned cheetah (((i64::from(e) + (1 << 32)) as u64) & u64::from(sz - 1)) as i32 }) } } impl BaseVol for VolMap3d { type Vox = V::Vox; type Err = VolMap3dErr; } impl ReadVol for VolMap3d { #[inline(always)] fn get(&self, pos: Vec3) -> Result<&V::Vox, VolMap3dErr> { let ck = Self::chunk_key(pos); self.chunks .get(&ck) .ok_or(VolMap3dErr::NoSuchChunk) .and_then(|chunk| { let co = Self::chunk_offs(pos); chunk.get(co).map_err(VolMap3dErr::ChunkErr) }) } } // TODO: This actually breaks the API: samples are supposed to have an offset of zero! // TODO: Should this be changed, perhaps? impl>, V: BaseVol + ReadVol + Debug, S: VolSize> SampleVol for VolMap3d { type Sample = VolMap3d; /// Take a sample of the terrain by cloning the voxels within the provided range. /// /// Note that the resultant volume does not carry forward metadata from the original chunks. fn sample(&self, range: I) -> Result> { let range = range.into(); let mut sample = VolMap3d::new()?; let chunk_min = Self::chunk_key(range.min); let chunk_max = Self::chunk_key(range.max); for x in chunk_min.x..=chunk_max.x { for y in chunk_min.y..=chunk_max.y { for z in chunk_min.z..=chunk_max.z { let chunk_key = Vec3::new(x, y, z); let chunk = self.get_key_arc(chunk_key).cloned(); if let Some(chunk) = chunk { sample.insert(chunk_key, chunk); } } } } Ok(sample) } } impl WriteVol for VolMap3d { #[inline(always)] fn set(&mut self, pos: Vec3, vox: V::Vox) -> Result<(), VolMap3dErr> { let ck = Self::chunk_key(pos); self.chunks .get_mut(&ck) .ok_or(VolMap3dErr::NoSuchChunk) .and_then(|chunk| { let co = Self::chunk_offs(pos); Arc::make_mut(chunk) .set(co, vox) .map_err(VolMap3dErr::ChunkErr) }) } } impl VolMap3d { pub fn new() -> Result> { if Self::chunk_size() .map(|e| e.is_power_of_two() && e > 0) .reduce_and() { Ok(Self { chunks: HashMap::new(), phantom: PhantomData, }) } else { Err(VolMap3dErr::InvalidChunkSize) } } pub fn chunk_size() -> Vec3 { S::SIZE } pub fn insert(&mut self, key: Vec3, chunk: Arc) -> Option> { self.chunks.insert(key, chunk) } pub fn get_key(&self, key: Vec3) -> Option<&V> { match self.chunks.get(&key) { Some(arc_chunk) => Some(arc_chunk.as_ref()), None => None, } } pub fn get_key_arc(&self, key: Vec3) -> Option<&Arc> { self.chunks.get(&key) } pub fn remove(&mut self, key: Vec3) -> Option> { self.chunks.remove(&key) } pub fn key_pos(&self, key: Vec3) -> Vec3 { key * S::SIZE.map(|e| e as i32) } pub fn pos_key(&self, pos: Vec3) -> Vec3 { Self::chunk_key(pos) } pub fn iter(&self) -> ChunkIter { ChunkIter { iter: self.chunks.iter(), } } } pub struct ChunkIter<'a, V: BaseVol> { iter: hash_map::Iter<'a, Vec3, Arc>, } impl<'a, V: BaseVol> Iterator for ChunkIter<'a, V> { type Item = (Vec3, &'a Arc); fn next(&mut self) -> Option { self.iter.next().map(|(k, c)| (*k, c)) } }