valence/src/chunk.rs

437 lines
12 KiB
Rust
Raw Normal View History

2022-04-14 14:55:45 -07:00
// TODO: https://github.com/rust-lang/rust/issues/88581 for div_ceil
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;
use std::ops::Deref;
2022-04-14 14:55:45 -07:00
use bitvec::vec::BitVec;
use num::Integer;
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;
use crate::{BiomeId, Server, Ticks};
2022-04-14 14:55:45 -07:00
pub struct Chunks {
chunks: HashMap<ChunkPos, Chunk>,
server: Server,
section_count: u32,
2022-04-29 00:48:41 -07:00
}
impl Chunks {
pub(crate) fn new(server: Server, section_count: u32) -> Self {
Self {
chunks: HashMap::new(),
server,
section_count,
}
2022-04-29 00:48:41 -07:00
}
pub fn count(&self) -> usize {
self.chunks.len()
2022-04-29 00:48:41 -07:00
}
pub fn get(&self, pos: ChunkPos) -> Option<&Chunk> {
self.chunks.get(&pos)
2022-04-29 00:48:41 -07:00
}
pub fn clear(&mut self) {
self.chunks.clear();
2022-04-29 00:48:41 -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
}
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-04-29 00:48:41 -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-05-16 11:53:21 -07:00
pub fn create(&mut self, pos: impl Into<ChunkPos>) -> bool {
let chunk = Chunk::new(self.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
}
pub fn delete(&mut self, pos: ChunkPos) -> bool {
self.0.chunks.remove(&pos).is_some()
2022-04-29 00:48:41 -07:00
}
pub fn get_mut(&mut self, pos: ChunkPos) -> Option<ChunkMut> {
self.0.chunks.get_mut(&pos).map(ChunkMut)
2022-04-29 00:48:41 -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
}
}
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>,
/// The MOTION_BLOCKING heightmap
2022-04-14 14:55:45 -07:00
heightmap: Vec<i64>,
modified: bool,
created_tick: Ticks,
2022-04-14 14:55:45 -07:00
}
impl Chunk {
pub(crate) fn new(section_count: u32, current_tick: Ticks) -> Self {
2022-04-14 14:55:45 -07:00
let sect = ChunkSection {
2022-04-30 19:05:38 -07:00
blocks: [BlockState::default(); 4096],
biomes: [BiomeId::default(); 64],
2022-04-14 14:55:45 -07:00
compact_data: Vec::new(),
modified: true,
};
let mut chunk = Self {
sections: vec![sect; section_count as usize].into(),
2022-04-14 14:55:45 -07:00
heightmap: Vec::new(),
modified: true,
created_tick: current_tick,
2022-04-14 14:55:45 -07:00
};
ChunkMut(&mut chunk).apply_modifications();
2022-04-14 14:55:45 -07:00
chunk
}
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 {
self.sections[y / 16].blocks[x + z * 16 + y % 16 * 16 * 16]
} 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()
}
}
/// 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();
for sect in self.sections.iter() {
blocks_and_biomes.extend_from_slice(&sect.compact_data);
2022-04-14 14:55:45 -07:00
}
ChunkDataAndUpdateLight {
chunk_x: pos.x,
chunk_z: pos.z,
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,
// 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(),
// 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(),
}
}
/// Gets the unapplied changes to this chunk as a block change packet.
pub(crate) fn block_change_packet(&self, pos: ChunkPos) -> Option<BlockChangePacket> {
if !self.modified {
return None;
}
// TODO
None
}
}
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) {
if x < 16 && y < self.height() && z < 16 {
let sec = &mut self.0.sections[y / 16];
let idx = x + z * 16 + y % 16 * 16 * 16;
if block != sec.blocks[idx] {
sec.blocks[idx] = block;
// TODO: set the modified bit.
sec.modified = true;
self.0.modified = true;
// TODO: update block entity if b could have block entity data.
}
}
}
pub fn set_biome(&mut self, x: usize, y: usize, z: usize, b: BiomeId) {
if x < 4 && y < self.height() / 4 && z < 4 {
self.0.sections[y / 4].biomes[x + z * 4 + y % 4 * 4 * 4] = b;
}
}
2022-04-14 14:55:45 -07:00
pub(crate) fn apply_modifications(&mut self) {
if self.modified {
self.0.modified = false;
2022-04-14 14:55:45 -07:00
for sect in self.0.sections.iter_mut() {
2022-04-14 14:55:45 -07:00
if sect.modified {
sect.modified = false;
sect.compact_data.clear();
2022-04-30 19:05:38 -07:00
let non_air_block_count = sect.blocks.iter().filter(|&&b| !b.is_air()).count();
2022-04-14 14:55:45 -07:00
(non_air_block_count as i16)
.encode(&mut sect.compact_data)
.unwrap();
2022-04-30 19:05:38 -07:00
encode_paletted_container(
sect.blocks.iter().map(|b| b.to_raw()),
4,
9,
15,
&mut sect.compact_data,
)
.unwrap();
2022-04-14 14:55:45 -07:00
// TODO: The direct bits per idx changes depending on the number of biomes in
// the biome registry.
2022-04-30 19:05:38 -07:00
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
}
}
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 }
}
}
impl From<(i32, i32)> for ChunkPos {
fn from((x, z): (i32, i32)) -> Self {
ChunkPos { x, z }
}
}
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-04-14 14:55:45 -07:00
/// A 16x16x16 section of blocks, biomes, and light in a chunk.
#[derive(Clone)]
struct ChunkSection {
/// The blocks in this section, stored in x, z, y order.
2022-04-30 19:05:38 -07:00
blocks: [BlockState; 4096],
biomes: [BiomeId; 64],
2022-04-14 14:55:45 -07:00
compact_data: Vec<u8>,
/// If the blocks or biomes were modified.
modified: bool,
}
/// 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-04-30 19:05:38 -07:00
let block = sections[y / 16].blocks[x + z * 16 + y % 16 * 16 * 16];
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
}