mirror of
https://github.com/italicsjenga/valence.git
synced 2025-01-11 07:11:30 +11:00
Use the block state API in chunk.rs.
This commit is contained in:
parent
27e818d6cd
commit
72fe8395cb
|
@ -2,6 +2,7 @@ use std::net::SocketAddr;
|
||||||
use std::sync::atomic::{AtomicUsize, Ordering};
|
use std::sync::atomic::{AtomicUsize, Ordering};
|
||||||
|
|
||||||
use log::LevelFilter;
|
use log::LevelFilter;
|
||||||
|
use valence::block::BlockState;
|
||||||
use valence::client::GameMode;
|
use valence::client::GameMode;
|
||||||
use valence::config::{Config, Login, ServerListPing};
|
use valence::config::{Config, Login, ServerListPing};
|
||||||
use valence::text::Color;
|
use valence::text::Color;
|
||||||
|
@ -59,7 +60,7 @@ impl Config for Game {
|
||||||
for z in 0..16 {
|
for z in 0..16 {
|
||||||
for x in 0..16 {
|
for x in 0..16 {
|
||||||
for y in 0..50 {
|
for y in 0..50 {
|
||||||
chunk.set_block_state(x, y, z, 1);
|
chunk.set_block_state(x, y, z, BlockState::STONE);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
55
src/chunk.rs
55
src/chunk.rs
|
@ -8,6 +8,7 @@ use bitvec::vec::BitVec;
|
||||||
use num::Integer;
|
use num::Integer;
|
||||||
use rayon::iter::ParallelIterator;
|
use rayon::iter::ParallelIterator;
|
||||||
|
|
||||||
|
use crate::block::BlockState;
|
||||||
use crate::glm::DVec2;
|
use crate::glm::DVec2;
|
||||||
use crate::packets::play::{
|
use crate::packets::play::{
|
||||||
BlockChange, ChunkDataAndUpdateLight, ChunkDataHeightmaps, ClientPlayPacket, MultiBlockChange,
|
BlockChange, ChunkDataAndUpdateLight, ChunkDataHeightmaps, ClientPlayPacket, MultiBlockChange,
|
||||||
|
@ -81,8 +82,8 @@ pub struct Chunk {
|
||||||
impl Chunk {
|
impl Chunk {
|
||||||
pub(crate) fn new(section_count: usize) -> Self {
|
pub(crate) fn new(section_count: usize) -> Self {
|
||||||
let sect = ChunkSection {
|
let sect = ChunkSection {
|
||||||
blocks: [0; 4096],
|
blocks: [BlockState::default(); 4096],
|
||||||
biomes: [0; 64],
|
biomes: [BiomeId::default(); 64],
|
||||||
compact_data: Vec::new(),
|
compact_data: Vec::new(),
|
||||||
modified: true,
|
modified: true,
|
||||||
};
|
};
|
||||||
|
@ -110,15 +111,15 @@ impl Chunk {
|
||||||
self.sections.len() * 16
|
self.sections.len() * 16
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_block_state(&self, x: usize, y: usize, z: usize) -> u16 {
|
pub fn get_block_state(&self, x: usize, y: usize, z: usize) -> BlockState {
|
||||||
if x < 16 && y < self.height() && z < 16 {
|
if x < 16 && y < self.height() && z < 16 {
|
||||||
self.sections[y / 16].blocks[x + z * 16 + y % 16 * 16 * 16]
|
self.sections[y / 16].blocks[x + z * 16 + y % 16 * 16 * 16]
|
||||||
} else {
|
} else {
|
||||||
0
|
BlockState::AIR
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_block_state(&mut self, x: usize, y: usize, z: usize, block: u16) {
|
pub fn set_block_state(&mut self, x: usize, y: usize, z: usize, block: BlockState) {
|
||||||
if x < 16 && y < self.height() && z < 16 {
|
if x < 16 && y < self.height() && z < 16 {
|
||||||
let sec = &mut self.sections[y / 16];
|
let sec = &mut self.sections[y / 16];
|
||||||
let idx = x + z * 16 + y % 16 * 16 * 16;
|
let idx = x + z * 16 + y % 16 * 16 * 16;
|
||||||
|
@ -134,7 +135,7 @@ impl Chunk {
|
||||||
|
|
||||||
pub fn get_biome(&self, x: usize, y: usize, z: usize) -> BiomeId {
|
pub fn get_biome(&self, x: usize, y: usize, z: usize) -> BiomeId {
|
||||||
if x < 4 && y < self.height() / 4 && z < 4 {
|
if x < 4 && y < self.height() / 4 && z < 4 {
|
||||||
BiomeId(self.sections[y / 4].biomes[x + z * 4 + y % 4 * 4 * 4])
|
self.sections[y / 4].biomes[x + z * 4 + y % 4 * 4 * 4]
|
||||||
} else {
|
} else {
|
||||||
BiomeId::default()
|
BiomeId::default()
|
||||||
}
|
}
|
||||||
|
@ -142,7 +143,7 @@ impl Chunk {
|
||||||
|
|
||||||
pub fn set_biome(&mut self, x: usize, y: usize, z: usize, b: BiomeId) {
|
pub fn set_biome(&mut self, x: usize, y: usize, z: usize, b: BiomeId) {
|
||||||
if x < 4 && y < self.height() / 4 && z < 4 {
|
if x < 4 && y < self.height() / 4 && z < 4 {
|
||||||
self.sections[y / 4].biomes[x + z * 4 + y % 4 * 4 * 4] = b.0;
|
self.sections[y / 4].biomes[x + z * 4 + y % 4 * 4 * 4] = b;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -225,19 +226,30 @@ impl Chunk {
|
||||||
|
|
||||||
sect.compact_data.clear();
|
sect.compact_data.clear();
|
||||||
|
|
||||||
// TODO: consider cave_air and void_air.
|
let non_air_block_count = sect.blocks.iter().filter(|&&b| !b.is_air()).count();
|
||||||
let non_air_block_count = sect.blocks.iter().filter(|&&b| b != 0).count();
|
|
||||||
|
|
||||||
(non_air_block_count as i16)
|
(non_air_block_count as i16)
|
||||||
.encode(&mut sect.compact_data)
|
.encode(&mut sect.compact_data)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
encode_paletted_container(§.blocks, 4, 9, 15, &mut sect.compact_data)
|
encode_paletted_container(
|
||||||
.unwrap();
|
sect.blocks.iter().map(|b| b.to_raw()),
|
||||||
|
4,
|
||||||
|
9,
|
||||||
|
15,
|
||||||
|
&mut sect.compact_data,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
// TODO: The direct bits per idx changes depending on the number of biomes in
|
// TODO: The direct bits per idx changes depending on the number of biomes in
|
||||||
// the biome registry.
|
// the biome registry.
|
||||||
encode_paletted_container(§.biomes, 0, 4, 6, &mut sect.compact_data)
|
encode_paletted_container(
|
||||||
.unwrap();
|
sect.biomes.iter().map(|b| b.0),
|
||||||
|
0,
|
||||||
|
4,
|
||||||
|
6,
|
||||||
|
&mut sect.compact_data,
|
||||||
|
)
|
||||||
|
.unwrap();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -295,8 +307,8 @@ impl From<(i32, i32)> for ChunkPos {
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
struct ChunkSection {
|
struct ChunkSection {
|
||||||
/// The blocks in this section, stored in x, z, y order.
|
/// The blocks in this section, stored in x, z, y order.
|
||||||
blocks: [u16; 4096],
|
blocks: [BlockState; 4096],
|
||||||
biomes: [u16; 64],
|
biomes: [BiomeId; 64],
|
||||||
compact_data: Vec<u8>,
|
compact_data: Vec<u8>,
|
||||||
/// If the blocks or biomes were modified.
|
/// If the blocks or biomes were modified.
|
||||||
modified: bool,
|
modified: bool,
|
||||||
|
@ -315,8 +327,9 @@ fn build_heightmap(sections: &[ChunkSection], heightmap: &mut Vec<i64>) {
|
||||||
for x in 0..16 {
|
for x in 0..16 {
|
||||||
for z in 0..16 {
|
for z in 0..16 {
|
||||||
for y in (0..height).rev() {
|
for y in (0..height).rev() {
|
||||||
|
let block = sections[y / 16].blocks[x + z * 16 + y % 16 * 16 * 16];
|
||||||
// TODO: is_solid || is_fluid heuristic for motion blocking.
|
// TODO: is_solid || is_fluid heuristic for motion blocking.
|
||||||
if sections[y / 16].blocks[x + z * 16 + y % 16 * 16 * 16] != 0 {
|
if !block.is_air() {
|
||||||
let column_height = y as u64;
|
let column_height = y as u64;
|
||||||
|
|
||||||
let i = x * 16 + z; // TODO: X or Z major?
|
let i = x * 16 + z; // TODO: X or Z major?
|
||||||
|
@ -331,15 +344,17 @@ fn build_heightmap(sections: &[ChunkSection], heightmap: &mut Vec<i64>) {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn encode_paletted_container(
|
fn encode_paletted_container(
|
||||||
entries: &[u16],
|
entries: impl ExactSizeIterator<Item = u16> + Clone,
|
||||||
min_bits_per_idx: usize,
|
min_bits_per_idx: usize,
|
||||||
direct_threshold: usize,
|
direct_threshold: usize,
|
||||||
direct_bits_per_idx: usize,
|
direct_bits_per_idx: usize,
|
||||||
w: &mut impl Write,
|
w: &mut impl Write,
|
||||||
) -> anyhow::Result<()> {
|
) -> anyhow::Result<()> {
|
||||||
|
let entries_len = entries.len();
|
||||||
|
|
||||||
let mut palette = Vec::new();
|
let mut palette = Vec::new();
|
||||||
|
|
||||||
for &entry in entries {
|
for entry in entries.clone() {
|
||||||
if !palette.contains(&entry) {
|
if !palette.contains(&entry) {
|
||||||
palette.push(entry);
|
palette.push(entry);
|
||||||
}
|
}
|
||||||
|
@ -362,7 +377,7 @@ fn encode_paletted_container(
|
||||||
|
|
||||||
VarInt(u64_count as i32).encode(w)?;
|
VarInt(u64_count as i32).encode(w)?;
|
||||||
|
|
||||||
for &entry in entries {
|
for entry in entries {
|
||||||
let mut val = 0u64;
|
let mut val = 0u64;
|
||||||
for i in 0..idxs_per_u64 {
|
for i in 0..idxs_per_u64 {
|
||||||
val |= (entry as u64) << (i * direct_bits_per_idx);
|
val |= (entry as u64) << (i * direct_bits_per_idx);
|
||||||
|
@ -382,7 +397,7 @@ fn encode_paletted_container(
|
||||||
|
|
||||||
VarInt(u64_count as i32).encode(w)?;
|
VarInt(u64_count as i32).encode(w)?;
|
||||||
|
|
||||||
for &entry in entries {
|
for entry in entries {
|
||||||
let palette_idx = palette
|
let palette_idx = palette
|
||||||
.iter()
|
.iter()
|
||||||
.position(|&e| e == entry)
|
.position(|&e| e == entry)
|
||||||
|
|
|
@ -58,10 +58,7 @@ impl<T> SlotMap<T> {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn insert(&mut self, val: T) -> Key {
|
pub fn insert(&mut self, val: T) -> Key {
|
||||||
assert!(
|
assert!(self.count < u32::MAX, "SlotMap: too many items inserted");
|
||||||
self.count < u32::MAX,
|
|
||||||
"SlotMap: too many items inserted"
|
|
||||||
);
|
|
||||||
|
|
||||||
if self.next_free_head == self.slots.len() as u32 {
|
if self.next_free_head == self.slots.len() as u32 {
|
||||||
self.slots.push(Slot {
|
self.slots.push(Slot {
|
||||||
|
|
Loading…
Reference in a new issue