2022-04-14 14:55:45 -07:00
|
|
|
// TODO: https://github.com/rust-lang/rust/issues/88581 for div_ceil
|
|
|
|
|
2022-05-16 02:36:14 -07:00
|
|
|
use std::collections::HashMap;
|
2022-04-14 14:55:45 -07:00
|
|
|
use std::io::Write;
|
2022-04-29 00:48:41 -07:00
|
|
|
use std::iter::FusedIterator;
|
2022-05-16 02:36:14 -07:00
|
|
|
use std::ops::Deref;
|
2022-04-14 14:55:45 -07:00
|
|
|
|
|
|
|
use bitvec::vec::BitVec;
|
|
|
|
use num::Integer;
|
2022-05-16 02:36:14 -07:00
|
|
|
use rayon::iter::{IntoParallelRefIterator, IntoParallelRefMutIterator, ParallelIterator};
|
2022-04-14 14:55:45 -07:00
|
|
|
|
2022-04-30 19:05:38 -07:00
|
|
|
use crate::block::BlockState;
|
2022-06-09 20:26:21 -07:00
|
|
|
use crate::packets::play::s2c::{
|
|
|
|
BlockChange, ChunkDataAndUpdateLight, ChunkDataHeightmaps, MultiBlockChange, S2cPlayPacket,
|
2022-04-14 14:55:45 -07:00
|
|
|
};
|
|
|
|
use crate::protocol::{Encode, Nbt};
|
|
|
|
use crate::var_int::VarInt;
|
2022-06-22 08:06:54 -07:00
|
|
|
use crate::var_long::VarLong;
|
|
|
|
use crate::{BiomeId, BlockPos, DimensionId, Server, Ticks};
|
2022-04-14 14:55:45 -07:00
|
|
|
|
2022-05-16 02:36:14 -07:00
|
|
|
pub struct Chunks {
|
|
|
|
chunks: HashMap<ChunkPos, Chunk>,
|
|
|
|
server: Server,
|
2022-06-22 08:06:54 -07:00
|
|
|
dimension: DimensionId,
|
2022-04-29 00:48:41 -07:00
|
|
|
}
|
|
|
|
|
2022-05-16 02:36:14 -07:00
|
|
|
impl Chunks {
|
2022-06-22 08:06:54 -07:00
|
|
|
pub(crate) fn new(server: Server, dimension: DimensionId) -> Self {
|
2022-05-16 02:36:14 -07:00
|
|
|
Self {
|
|
|
|
chunks: HashMap::new(),
|
|
|
|
server,
|
2022-06-22 08:06:54 -07:00
|
|
|
dimension,
|
2022-05-16 02:36:14 -07:00
|
|
|
}
|
2022-04-29 00:48:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn count(&self) -> usize {
|
2022-05-16 02:36:14 -07:00
|
|
|
self.chunks.len()
|
2022-04-29 00:48:41 -07:00
|
|
|
}
|
|
|
|
|
2022-06-22 08:06:54 -07:00
|
|
|
pub fn get(&self, pos: impl Into<ChunkPos>) -> Option<&Chunk> {
|
|
|
|
self.chunks.get(&pos.into())
|
2022-04-29 00:48:41 -07:00
|
|
|
}
|
|
|
|
|
2022-05-16 02:36:14 -07:00
|
|
|
pub fn clear(&mut self) {
|
|
|
|
self.chunks.clear();
|
2022-04-29 00:48:41 -07:00
|
|
|
}
|
|
|
|
|
2022-05-16 02:36:14 -07:00
|
|
|
pub fn iter(&self) -> impl FusedIterator<Item = (ChunkPos, &Chunk)> + Clone + '_ {
|
|
|
|
self.chunks.iter().map(|(&pos, chunk)| (pos, chunk))
|
2022-04-29 00:48:41 -07:00
|
|
|
}
|
|
|
|
|
2022-05-16 02:36:14 -07:00
|
|
|
pub fn par_iter(&self) -> impl ParallelIterator<Item = (ChunkPos, &Chunk)> + Clone + '_ {
|
|
|
|
self.chunks.par_iter().map(|(&pos, chunk)| (pos, chunk))
|
2022-04-29 00:48:41 -07:00
|
|
|
}
|
2022-06-22 08:06:54 -07:00
|
|
|
|
|
|
|
pub fn get_block_state(&self, pos: impl Into<BlockPos>) -> Option<BlockState> {
|
|
|
|
let pos = pos.into();
|
2022-06-24 16:11:15 -07:00
|
|
|
let chunk_pos = ChunkPos::from(pos);
|
2022-06-22 08:06:54 -07:00
|
|
|
|
|
|
|
let chunk = self.get(chunk_pos)?;
|
|
|
|
|
|
|
|
let min_y = self.server.dimension(self.dimension).min_y;
|
|
|
|
|
|
|
|
let y = pos.y.checked_sub(min_y)?.try_into().ok()?;
|
|
|
|
|
|
|
|
if y < chunk.height() {
|
|
|
|
Some(chunk.get_block_state(
|
|
|
|
pos.x.rem_euclid(16) as usize,
|
|
|
|
y,
|
|
|
|
pos.z.rem_euclid(16) as usize,
|
|
|
|
))
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
2022-05-16 02:36:14 -07:00
|
|
|
}
|
2022-04-29 00:48:41 -07:00
|
|
|
|
2022-05-16 02:36:14 -07:00
|
|
|
impl<'a> ChunksMut<'a> {
|
|
|
|
pub(crate) fn new(chunks: &'a mut Chunks) -> Self {
|
|
|
|
Self(chunks)
|
2022-04-29 00:48:41 -07:00
|
|
|
}
|
|
|
|
|
2022-06-24 16:11:15 -07:00
|
|
|
pub fn reborrow(&mut self) -> ChunksMut {
|
|
|
|
ChunksMut(self.0)
|
|
|
|
}
|
|
|
|
|
2022-05-16 11:53:21 -07:00
|
|
|
pub fn create(&mut self, pos: impl Into<ChunkPos>) -> bool {
|
2022-06-22 08:06:54 -07:00
|
|
|
let section_count = (self.server.dimension(self.dimension).height / 16) as u32;
|
|
|
|
let chunk = Chunk::new(section_count, self.server.current_tick());
|
2022-05-16 11:53:21 -07:00
|
|
|
self.0.chunks.insert(pos.into(), chunk).is_none()
|
2022-04-29 00:48:41 -07:00
|
|
|
}
|
|
|
|
|
2022-05-16 02:36:14 -07:00
|
|
|
pub fn delete(&mut self, pos: ChunkPos) -> bool {
|
|
|
|
self.0.chunks.remove(&pos).is_some()
|
2022-04-29 00:48:41 -07:00
|
|
|
}
|
|
|
|
|
2022-06-22 08:06:54 -07:00
|
|
|
pub fn get_mut(&mut self, pos: impl Into<ChunkPos>) -> Option<ChunkMut> {
|
|
|
|
self.0.chunks.get_mut(&pos.into()).map(ChunkMut)
|
2022-04-29 00:48:41 -07:00
|
|
|
}
|
|
|
|
|
2022-05-16 02:36:14 -07:00
|
|
|
pub fn iter_mut(&mut self) -> impl FusedIterator<Item = (ChunkPos, ChunkMut)> + '_ {
|
|
|
|
self.0
|
|
|
|
.chunks
|
|
|
|
.iter_mut()
|
|
|
|
.map(|(&pos, chunk)| (pos, ChunkMut(chunk)))
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn par_iter_mut(&mut self) -> impl ParallelIterator<Item = (ChunkPos, ChunkMut)> + '_ {
|
|
|
|
self.0
|
|
|
|
.chunks
|
|
|
|
.par_iter_mut()
|
|
|
|
.map(|(&pos, chunk)| (pos, ChunkMut(chunk)))
|
2022-04-29 00:48:41 -07:00
|
|
|
}
|
2022-06-22 08:06:54 -07:00
|
|
|
|
|
|
|
pub fn set_block_state(&mut self, pos: impl Into<BlockPos>, block: BlockState) -> bool {
|
|
|
|
let pos = pos.into();
|
2022-06-24 16:11:15 -07:00
|
|
|
let chunk_pos = ChunkPos::from(pos);
|
2022-06-22 08:06:54 -07:00
|
|
|
|
|
|
|
if let Some(chunk) = self.0.chunks.get_mut(&chunk_pos) {
|
|
|
|
let min_y = self.0.server.dimension(self.0.dimension).min_y;
|
|
|
|
|
|
|
|
if let Some(y) = pos.y.checked_sub(min_y).and_then(|y| y.try_into().ok()) {
|
|
|
|
if y < chunk.height() {
|
|
|
|
ChunkMut(chunk).set_block_state(
|
|
|
|
pos.x.rem_euclid(16) as usize,
|
|
|
|
y,
|
|
|
|
pos.z.rem_euclid(16) as usize,
|
|
|
|
block,
|
|
|
|
);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
false
|
|
|
|
}
|
2022-04-29 00:48:41 -07:00
|
|
|
}
|
|
|
|
|
2022-05-16 02:36:14 -07:00
|
|
|
pub struct ChunksMut<'a>(&'a mut Chunks);
|
|
|
|
|
|
|
|
impl<'a> Deref for ChunksMut<'a> {
|
|
|
|
type Target = Chunks;
|
|
|
|
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
|
|
self.0
|
|
|
|
}
|
|
|
|
}
|
2022-04-29 00:48:41 -07:00
|
|
|
|
2022-04-14 14:55:45 -07:00
|
|
|
pub struct Chunk {
|
|
|
|
sections: Box<[ChunkSection]>,
|
|
|
|
// TODO block_entities: HashMap<u32, BlockEntity>,
|
2022-05-16 02:36:14 -07:00
|
|
|
/// The MOTION_BLOCKING heightmap
|
2022-04-14 14:55:45 -07:00
|
|
|
heightmap: Vec<i64>,
|
2022-05-16 02:36:14 -07:00
|
|
|
created_tick: Ticks,
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Chunk {
|
2022-05-16 02:36:14 -07:00
|
|
|
pub(crate) fn new(section_count: u32, current_tick: Ticks) -> Self {
|
2022-04-14 14:55:45 -07:00
|
|
|
let sect = ChunkSection {
|
2022-06-22 08:06:54 -07:00
|
|
|
blocks: [BlockState::AIR.to_raw(); 4096],
|
|
|
|
modified_count: 1, // Must be >0 so the chunk is initialized.
|
2022-04-30 19:05:38 -07:00
|
|
|
biomes: [BiomeId::default(); 64],
|
2022-04-14 14:55:45 -07:00
|
|
|
compact_data: Vec::new(),
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut chunk = Self {
|
2022-05-16 02:36:14 -07:00
|
|
|
sections: vec![sect; section_count as usize].into(),
|
2022-04-14 14:55:45 -07:00
|
|
|
heightmap: Vec::new(),
|
2022-05-16 02:36:14 -07:00
|
|
|
created_tick: current_tick,
|
2022-04-14 14:55:45 -07:00
|
|
|
};
|
|
|
|
|
2022-05-16 02:36:14 -07:00
|
|
|
ChunkMut(&mut chunk).apply_modifications();
|
2022-04-14 14:55:45 -07:00
|
|
|
chunk
|
|
|
|
}
|
|
|
|
|
2022-05-16 02:36:14 -07:00
|
|
|
pub fn created_tick(&self) -> Ticks {
|
|
|
|
self.created_tick
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn height(&self) -> usize {
|
|
|
|
self.sections.len() * 16
|
|
|
|
}
|
|
|
|
|
2022-04-30 19:05:38 -07:00
|
|
|
pub fn get_block_state(&self, x: usize, y: usize, z: usize) -> BlockState {
|
2022-04-14 14:55:45 -07:00
|
|
|
if x < 16 && y < self.height() && z < 16 {
|
2022-06-22 08:06:54 -07:00
|
|
|
BlockState::from_raw_unchecked(
|
|
|
|
self.sections[y / 16].blocks[x + z * 16 + y % 16 * 16 * 16] & BLOCK_STATE_MASK,
|
|
|
|
)
|
2022-04-14 14:55:45 -07:00
|
|
|
} else {
|
2022-04-30 19:05:38 -07:00
|
|
|
BlockState::AIR
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_biome(&self, x: usize, y: usize, z: usize) -> BiomeId {
|
|
|
|
if x < 4 && y < self.height() / 4 && z < 4 {
|
2022-04-30 19:05:38 -07:00
|
|
|
self.sections[y / 4].biomes[x + z * 4 + y % 4 * 4 * 4]
|
2022-04-14 14:55:45 -07:00
|
|
|
} else {
|
|
|
|
BiomeId::default()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-16 02:36:14 -07:00
|
|
|
/// Gets the chunk data packet for this chunk with the given position. This
|
|
|
|
/// does not include unapplied changes.
|
|
|
|
pub(crate) fn chunk_data_packet(&self, pos: ChunkPos) -> ChunkDataAndUpdateLight {
|
2022-04-14 14:55:45 -07:00
|
|
|
let mut blocks_and_biomes = Vec::new();
|
|
|
|
|
2022-05-16 02:36:14 -07:00
|
|
|
for sect in self.sections.iter() {
|
|
|
|
blocks_and_biomes.extend_from_slice(§.compact_data);
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ChunkDataAndUpdateLight {
|
|
|
|
chunk_x: pos.x,
|
|
|
|
chunk_z: pos.z,
|
2022-05-16 02:36:14 -07:00
|
|
|
heightmaps: Nbt(ChunkDataHeightmaps {
|
|
|
|
motion_blocking: self.heightmap.clone(),
|
|
|
|
}),
|
2022-04-14 14:55:45 -07:00
|
|
|
blocks_and_biomes,
|
|
|
|
block_entities: Vec::new(), // TODO
|
|
|
|
trust_edges: true,
|
2022-04-30 05:06:20 -07:00
|
|
|
// sky_light_mask: bitvec![u64, _; 1; section_count + 2],
|
|
|
|
sky_light_mask: BitVec::new(),
|
2022-04-14 14:55:45 -07:00
|
|
|
block_light_mask: BitVec::new(),
|
|
|
|
empty_sky_light_mask: BitVec::new(),
|
|
|
|
empty_block_light_mask: BitVec::new(),
|
2022-04-30 05:06:20 -07:00
|
|
|
// sky_light_arrays: vec![[0xff; 2048]; section_count + 2],
|
|
|
|
sky_light_arrays: Vec::new(),
|
2022-04-14 14:55:45 -07:00
|
|
|
block_light_arrays: Vec::new(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-22 08:06:54 -07:00
|
|
|
/// Returns unapplied changes to this chunk as block change packets through
|
|
|
|
/// the provided closure.
|
|
|
|
pub(crate) fn block_change_packets(
|
|
|
|
&self,
|
|
|
|
pos: ChunkPos,
|
|
|
|
min_y: i32,
|
|
|
|
mut packet: impl FnMut(BlockChangePacket),
|
|
|
|
) {
|
|
|
|
for (sect_y, sect) in self.sections.iter().enumerate() {
|
|
|
|
if sect.modified_count == 1 {
|
|
|
|
let (idx, &block) = sect
|
|
|
|
.blocks
|
|
|
|
.iter()
|
|
|
|
.enumerate()
|
|
|
|
.find(|&(_, &b)| b & !BLOCK_STATE_MASK != 0)
|
|
|
|
.expect("invalid modified count");
|
|
|
|
|
|
|
|
let global_x = pos.x * 16 + (idx % 16) as i32;
|
|
|
|
let global_y = sect_y as i32 * 16 + (idx / (16 * 16)) as i32 + min_y;
|
|
|
|
let global_z = pos.z * 16 + (idx / 16 % 16) as i32;
|
|
|
|
|
|
|
|
packet(BlockChangePacket::Single(BlockChange {
|
|
|
|
location: BlockPos::new(global_x, global_y, global_z),
|
|
|
|
block_id: VarInt((block & BLOCK_STATE_MASK).into()),
|
|
|
|
}));
|
|
|
|
} else if sect.modified_count > 1 {
|
|
|
|
let mut blocks = Vec::new();
|
|
|
|
for y in 0..16 {
|
|
|
|
for z in 0..16 {
|
|
|
|
for x in 0..16 {
|
|
|
|
let block =
|
|
|
|
sect.blocks[x as usize + z as usize * 16 + y as usize * 16 * 16];
|
|
|
|
|
|
|
|
if block & !BLOCK_STATE_MASK != 0 {
|
|
|
|
blocks.push(VarLong(
|
|
|
|
((block & BLOCK_STATE_MASK) as i64) << 12
|
|
|
|
| (x << 8 | z << 4 | y),
|
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let chunk_section_position = (pos.x as i64) << 42
|
|
|
|
| (pos.z as i64 & 0x3fffff) << 20
|
2022-06-24 16:11:15 -07:00
|
|
|
| (sect_y as i64 + min_y.div_euclid(16) as i64) & 0xfffff;
|
2022-04-14 14:55:45 -07:00
|
|
|
|
2022-06-22 08:06:54 -07:00
|
|
|
packet(BlockChangePacket::Multi(MultiBlockChange {
|
|
|
|
chunk_section_position,
|
|
|
|
invert_trust_edges: false,
|
|
|
|
blocks,
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
}
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
2022-05-16 02:36:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
pub struct ChunkMut<'a>(&'a mut Chunk);
|
|
|
|
|
|
|
|
impl<'a> Deref for ChunkMut<'a> {
|
|
|
|
type Target = Chunk;
|
|
|
|
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
|
|
self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> ChunkMut<'a> {
|
|
|
|
pub fn set_block_state(&mut self, x: usize, y: usize, z: usize, block: BlockState) {
|
2022-06-22 08:06:54 -07:00
|
|
|
assert!(
|
|
|
|
x < 16 && y < self.height() && z < 16,
|
|
|
|
"the chunk block coordinates must be within bounds"
|
|
|
|
);
|
|
|
|
|
|
|
|
let sect = &mut self.0.sections[y / 16];
|
|
|
|
let idx = x + z * 16 + y % 16 * 16 * 16;
|
|
|
|
|
|
|
|
if block.to_raw() != sect.blocks[idx] & BLOCK_STATE_MASK {
|
|
|
|
if sect.blocks[idx] & !BLOCK_STATE_MASK == 0 {
|
|
|
|
sect.modified_count += 1;
|
2022-05-16 02:36:14 -07:00
|
|
|
}
|
2022-06-22 08:06:54 -07:00
|
|
|
sect.blocks[idx] = block.to_raw() | !BLOCK_STATE_MASK;
|
|
|
|
|
|
|
|
// TODO: if the block type was modified and the old block type
|
|
|
|
// could be a block entity, then the block entity at this
|
|
|
|
// position must be cleared.
|
2022-05-16 02:36:14 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn set_biome(&mut self, x: usize, y: usize, z: usize, b: BiomeId) {
|
2022-06-22 08:06:54 -07:00
|
|
|
assert!(
|
|
|
|
x < 4 && y < self.height() / 4 && z < 4,
|
|
|
|
"the chunk biome coordinates must be within bounds"
|
|
|
|
);
|
|
|
|
|
|
|
|
self.0.sections[y / 4].biomes[x + z * 4 + y % 4 * 4 * 4] = b;
|
2022-05-16 02:36:14 -07:00
|
|
|
}
|
2022-04-14 14:55:45 -07:00
|
|
|
|
|
|
|
pub(crate) fn apply_modifications(&mut self) {
|
2022-06-22 08:06:54 -07:00
|
|
|
let mut any_modified = false;
|
|
|
|
|
|
|
|
for sect in self.0.sections.iter_mut() {
|
|
|
|
if sect.modified_count > 0 {
|
|
|
|
sect.modified_count = 0;
|
|
|
|
any_modified = true;
|
|
|
|
|
|
|
|
sect.compact_data.clear();
|
|
|
|
|
|
|
|
let mut non_air_block_count: i16 = 0;
|
|
|
|
|
|
|
|
for b in &mut sect.blocks {
|
|
|
|
*b &= BLOCK_STATE_MASK;
|
|
|
|
if !BlockState::from_raw_unchecked(*b).is_air() {
|
|
|
|
non_air_block_count += 1;
|
|
|
|
}
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
2022-06-22 08:06:54 -07:00
|
|
|
|
|
|
|
non_air_block_count.encode(&mut sect.compact_data).unwrap();
|
|
|
|
|
|
|
|
encode_paletted_container(
|
|
|
|
sect.blocks.iter().cloned(),
|
|
|
|
4,
|
|
|
|
9,
|
|
|
|
log2_ceil((BlockState::max_raw() + 1) as usize),
|
|
|
|
&mut sect.compact_data,
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
// TODO: The direct bits per idx changes depending on the number of biomes in
|
|
|
|
// the biome registry.
|
|
|
|
encode_paletted_container(
|
|
|
|
sect.biomes.iter().map(|b| b.0),
|
|
|
|
0,
|
|
|
|
4,
|
|
|
|
6,
|
|
|
|
&mut sect.compact_data,
|
|
|
|
)
|
|
|
|
.unwrap();
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
2022-06-22 08:06:54 -07:00
|
|
|
}
|
2022-04-14 14:55:45 -07:00
|
|
|
|
2022-06-22 08:06:54 -07:00
|
|
|
if any_modified {
|
2022-05-16 02:36:14 -07:00
|
|
|
build_heightmap(&self.0.sections, &mut self.0.heightmap);
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone, Debug)]
|
|
|
|
pub(crate) enum BlockChangePacket {
|
|
|
|
Single(BlockChange),
|
|
|
|
Multi(MultiBlockChange),
|
|
|
|
}
|
|
|
|
|
2022-06-09 20:26:21 -07:00
|
|
|
impl From<BlockChangePacket> for S2cPlayPacket {
|
2022-04-14 14:55:45 -07:00
|
|
|
fn from(p: BlockChangePacket) -> Self {
|
|
|
|
match p {
|
|
|
|
BlockChangePacket::Single(p) => p.into(),
|
|
|
|
BlockChangePacket::Multi(p) => p.into(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The X and Z position of a chunk in a world.
|
|
|
|
#[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash, Debug)]
|
|
|
|
pub struct ChunkPos {
|
|
|
|
/// The X position of the chunk.
|
|
|
|
pub x: i32,
|
|
|
|
/// The Z position of the chunk.
|
|
|
|
pub z: i32,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ChunkPos {
|
|
|
|
pub const fn new(x: i32, z: i32) -> Self {
|
|
|
|
Self { x, z }
|
|
|
|
}
|
2022-06-19 08:40:37 -07:00
|
|
|
|
|
|
|
pub fn at(x: f64, z: f64) -> Self {
|
2022-06-24 16:11:15 -07:00
|
|
|
Self::new((x / 16.0).floor() as i32, (z / 16.0).floor() as i32)
|
2022-06-19 08:40:37 -07:00
|
|
|
}
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
impl From<(i32, i32)> for ChunkPos {
|
|
|
|
fn from((x, z): (i32, i32)) -> Self {
|
|
|
|
ChunkPos { x, z }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-16 02:36:14 -07:00
|
|
|
impl Into<(i32, i32)> for ChunkPos {
|
|
|
|
fn into(self) -> (i32, i32) {
|
|
|
|
(self.x, self.z)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<[i32; 2]> for ChunkPos {
|
|
|
|
fn from([x, z]: [i32; 2]) -> Self {
|
|
|
|
(x, z).into()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Into<[i32; 2]> for ChunkPos {
|
|
|
|
fn into(self) -> [i32; 2] {
|
|
|
|
[self.x, self.z]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-24 16:11:15 -07:00
|
|
|
impl From<BlockPos> for ChunkPos {
|
|
|
|
fn from(pos: BlockPos) -> Self {
|
|
|
|
Self::new(pos.x.div_euclid(16), pos.z.div_euclid(16))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-14 14:55:45 -07:00
|
|
|
/// A 16x16x16 section of blocks, biomes, and light in a chunk.
|
|
|
|
#[derive(Clone)]
|
|
|
|
struct ChunkSection {
|
2022-06-22 08:06:54 -07:00
|
|
|
/// The block states in this section, stored in x, z, y order.
|
|
|
|
/// The most significant bit is used to indicate if this block has been
|
|
|
|
/// modified.
|
|
|
|
blocks: [u16; 4096],
|
|
|
|
/// The number of modified blocks
|
|
|
|
modified_count: u16,
|
2022-04-30 19:05:38 -07:00
|
|
|
biomes: [BiomeId; 64],
|
2022-04-14 14:55:45 -07:00
|
|
|
compact_data: Vec<u8>,
|
|
|
|
}
|
|
|
|
|
2022-06-22 08:06:54 -07:00
|
|
|
const BLOCK_STATE_MASK: u16 = 0x7fff;
|
|
|
|
|
|
|
|
const _: () = assert!(
|
|
|
|
BlockState::max_raw() <= BLOCK_STATE_MASK,
|
|
|
|
"There is not enough space in the block state type to store the modified bit. A bit array \
|
|
|
|
separate from the block state array should be created to keep track of modified blocks in \
|
|
|
|
the chunk section."
|
|
|
|
);
|
|
|
|
|
2022-04-14 14:55:45 -07:00
|
|
|
/// Builds the MOTION_BLOCKING heightmap.
|
|
|
|
fn build_heightmap(sections: &[ChunkSection], heightmap: &mut Vec<i64>) {
|
|
|
|
let height = sections.len() * 16;
|
|
|
|
let bits_per_val = log2_ceil(height);
|
|
|
|
let vals_per_u64 = 64 / bits_per_val;
|
2022-04-29 00:48:41 -07:00
|
|
|
let u64_count = Integer::div_ceil(&256, &vals_per_u64);
|
2022-04-14 14:55:45 -07:00
|
|
|
|
|
|
|
heightmap.clear();
|
2022-04-29 00:48:41 -07:00
|
|
|
heightmap.resize(u64_count, 0);
|
2022-04-14 14:55:45 -07:00
|
|
|
|
|
|
|
for x in 0..16 {
|
|
|
|
for z in 0..16 {
|
|
|
|
for y in (0..height).rev() {
|
2022-06-22 08:06:54 -07:00
|
|
|
let block = BlockState::from_raw_unchecked(
|
|
|
|
sections[y / 16].blocks[x + z * 16 + y % 16 * 16 * 16] & BLOCK_STATE_MASK,
|
|
|
|
);
|
|
|
|
|
2022-04-14 14:55:45 -07:00
|
|
|
// TODO: is_solid || is_fluid heuristic for motion blocking.
|
2022-04-30 19:05:38 -07:00
|
|
|
if !block.is_air() {
|
2022-04-14 14:55:45 -07:00
|
|
|
let column_height = y as u64;
|
|
|
|
|
|
|
|
let i = x * 16 + z; // TODO: X or Z major?
|
|
|
|
heightmap[i / vals_per_u64] |=
|
|
|
|
(column_height << (i % vals_per_u64 * bits_per_val)) as i64;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn encode_paletted_container(
|
2022-05-16 11:53:21 -07:00
|
|
|
mut entries: impl ExactSizeIterator<Item = u16> + Clone,
|
2022-04-14 14:55:45 -07:00
|
|
|
min_bits_per_idx: usize,
|
|
|
|
direct_threshold: usize,
|
|
|
|
direct_bits_per_idx: usize,
|
|
|
|
w: &mut impl Write,
|
|
|
|
) -> anyhow::Result<()> {
|
|
|
|
let mut palette = Vec::new();
|
|
|
|
|
2022-04-30 19:05:38 -07:00
|
|
|
for entry in entries.clone() {
|
2022-04-14 14:55:45 -07:00
|
|
|
if !palette.contains(&entry) {
|
|
|
|
palette.push(entry);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let bits_per_idx = log2_ceil(palette.len());
|
|
|
|
|
|
|
|
(bits_per_idx as u8).encode(w)?;
|
|
|
|
|
|
|
|
if bits_per_idx == 0 {
|
|
|
|
// Single value case
|
|
|
|
debug_assert_eq!(palette.len(), 1);
|
|
|
|
VarInt(palette[0] as i32).encode(w)?;
|
|
|
|
VarInt(0).encode(w)?; // data array length
|
|
|
|
} else if bits_per_idx >= direct_threshold {
|
|
|
|
// Direct case
|
|
|
|
// Skip the palette
|
|
|
|
let idxs_per_u64 = 64 / direct_bits_per_idx;
|
2022-04-29 00:48:41 -07:00
|
|
|
let u64_count = Integer::div_ceil(&entries.len(), &idxs_per_u64);
|
2022-04-14 14:55:45 -07:00
|
|
|
|
2022-04-29 00:48:41 -07:00
|
|
|
VarInt(u64_count as i32).encode(w)?;
|
2022-04-14 14:55:45 -07:00
|
|
|
|
2022-05-16 11:53:21 -07:00
|
|
|
for _ in 0..idxs_per_u64 {
|
2022-04-14 14:55:45 -07:00
|
|
|
let mut val = 0u64;
|
|
|
|
for i in 0..idxs_per_u64 {
|
2022-05-16 11:53:21 -07:00
|
|
|
if let Some(entry) = entries.next() {
|
|
|
|
val |= (entry as u64) << (i * direct_bits_per_idx);
|
|
|
|
}
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
val.encode(w)?;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Indirect case
|
|
|
|
VarInt(palette.len() as i32).encode(w)?;
|
|
|
|
for &val in &palette {
|
|
|
|
VarInt(val as i32).encode(w)?;
|
|
|
|
}
|
|
|
|
|
|
|
|
let bits_per_idx = bits_per_idx.max(min_bits_per_idx);
|
|
|
|
let idxs_per_u64 = 64 / bits_per_idx;
|
2022-04-29 00:48:41 -07:00
|
|
|
let u64_count = Integer::div_ceil(&entries.len(), &idxs_per_u64);
|
2022-04-14 14:55:45 -07:00
|
|
|
|
2022-04-29 00:48:41 -07:00
|
|
|
VarInt(u64_count as i32).encode(w)?;
|
2022-04-14 14:55:45 -07:00
|
|
|
|
2022-05-16 11:53:21 -07:00
|
|
|
for _ in 0..u64_count {
|
2022-04-14 14:55:45 -07:00
|
|
|
let mut val = 0u64;
|
|
|
|
for i in 0..idxs_per_u64 {
|
2022-05-16 11:53:21 -07:00
|
|
|
if let Some(entry) = entries.next() {
|
|
|
|
let palette_idx = palette
|
|
|
|
.iter()
|
|
|
|
.position(|&e| e == entry)
|
2022-05-17 02:58:43 -07:00
|
|
|
.expect("entry should be in the palette")
|
|
|
|
as u64;
|
|
|
|
|
2022-05-16 11:53:21 -07:00
|
|
|
val |= palette_idx << (i * bits_per_idx);
|
2022-05-17 02:58:43 -07:00
|
|
|
}
|
2022-04-14 14:55:45 -07:00
|
|
|
}
|
|
|
|
val.encode(w)?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Calculates the log base 2 rounded up.
|
|
|
|
fn log2_ceil(n: usize) -> usize {
|
|
|
|
n.next_power_of_two().trailing_zeros() as usize
|
|
|
|
}
|
2022-06-22 08:06:54 -07:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn set_get() {
|
|
|
|
let mut chunk = Chunk::new(16, 0);
|
|
|
|
let mut chunk = ChunkMut(&mut chunk);
|
|
|
|
|
|
|
|
chunk.set_block_state(1, 2, 3, BlockState::CAKE);
|
|
|
|
assert_eq!(chunk.get_block_state(1, 2, 3), BlockState::CAKE);
|
|
|
|
|
|
|
|
chunk.set_biome(1, 2, 3, BiomeId(7));
|
|
|
|
assert_eq!(chunk.get_biome(1, 2, 3), BiomeId(7));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_panic]
|
|
|
|
fn block_state_oob() {
|
|
|
|
let mut chunk = Chunk::new(16, 0);
|
|
|
|
let mut chunk = ChunkMut(&mut chunk);
|
|
|
|
|
|
|
|
chunk.set_block_state(16, 0, 0, BlockState::CAKE);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_panic]
|
|
|
|
fn biome_oob() {
|
|
|
|
let mut chunk = Chunk::new(16, 0);
|
|
|
|
let mut chunk = ChunkMut(&mut chunk);
|
|
|
|
|
|
|
|
chunk.set_biome(4, 0, 0, BiomeId(0));
|
|
|
|
}
|
|
|
|
}
|