use crate::{ vol::{BaseVol, RasterableVol, ReadVol, SampleVol, WriteVol}, volumes::dyna::DynaError, }; use hashbrown::{hash_map, HashMap}; use std::{fmt::Debug, sync::Arc}; use vek::*; #[derive(Debug)] pub enum VolGrid3dError { NoSuchChunk, ChunkErr(V::Error), DynaError(DynaError), InvalidChunkSize, } // V = Voxel // S = Size (replace with a const when const generics is a thing) // M = Chunk metadata #[derive(Clone)] pub struct VolGrid3d { chunks: HashMap, Arc>, } impl VolGrid3d { #[inline(always)] pub fn chunk_key(pos: Vec3) -> Vec3 { pos.map2(V::SIZE, |e, sz: u32| e >> (sz - 1).count_ones()) } #[inline(always)] pub fn chunk_offs(pos: Vec3) -> Vec3 { pos.map2(V::SIZE, |e, sz| e & (sz - 1) as i32) } } impl BaseVol for VolGrid3d { type Error = VolGrid3dError; type Vox = V::Vox; } impl ReadVol for VolGrid3d { #[inline(always)] fn get(&self, pos: Vec3) -> Result<&V::Vox, VolGrid3dError> { let ck = Self::chunk_key(pos); self.chunks .get(&ck) .ok_or(VolGrid3dError::NoSuchChunk) .and_then(|chunk| { let co = Self::chunk_offs(pos); chunk.get(co).map_err(VolGrid3dError::ChunkErr) }) } } // TODO: This actually breaks the API: samples are supposed to have an offset of // zero! TODO: Should this be changed, perhaps? impl>, V: RasterableVol + ReadVol + Debug> SampleVol for VolGrid3d { type Sample = VolGrid3d; /// 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 = VolGrid3d::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 + 1 { for y in chunk_min.y..chunk_max.y + 1 { for z in chunk_min.z..chunk_max.z + 1 { 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 VolGrid3d { #[inline(always)] fn set(&mut self, pos: Vec3, vox: V::Vox) -> Result<(), VolGrid3dError> { let ck = Self::chunk_key(pos); self.chunks .get_mut(&ck) .ok_or(VolGrid3dError::NoSuchChunk) .and_then(|chunk| { let co = Self::chunk_offs(pos); Arc::make_mut(chunk) .set(co, vox) .map_err(VolGrid3dError::ChunkErr) }) } } impl VolGrid3d { pub fn new() -> Result> { if Self::chunk_size() .map(|e| e.is_power_of_two() && e > 0) .reduce_and() { Ok(Self { chunks: HashMap::new(), }) } else { Err(VolGrid3dError::InvalidChunkSize) } } pub fn chunk_size() -> Vec3 { V::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 * V::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: RasterableVol> { iter: hash_map::Iter<'a, Vec3, Arc>, } impl<'a, V: RasterableVol> Iterator for ChunkIter<'a, V> { type Item = (Vec3, &'a Arc); fn next(&mut self) -> Option { self.iter.next().map(|(k, c)| (*k, c)) } }