diff --git a/pumpkin-core/src/math/position.rs b/pumpkin-core/src/math/position.rs index cd1f7ce26..7c51dff45 100644 --- a/pumpkin-core/src/math/position.rs +++ b/pumpkin-core/src/math/position.rs @@ -1,13 +1,33 @@ +use super::vector3::Vector3; use std::fmt; +use crate::math::vector2::Vector2; +use num_traits::Euclid; use serde::{Deserialize, Serialize}; -use super::vector3::Vector3; - #[derive(Clone, Copy)] /// Aka Block Position pub struct WorldPosition(pub Vector3); +impl WorldPosition { + pub fn chunk_and_chunk_relative_position(&self) -> (Vector2, Vector3) { + let (z_chunk, z_rem) = self.0.z.div_rem_euclid(&16); + let (x_chunk, x_rem) = self.0.x.div_rem_euclid(&16); + let chunk_coordinate = Vector2 { + x: x_chunk, + z: z_chunk, + }; + + // Since we divide by 16 remnant can never exceed u8 + let relative = Vector3 { + x: x_rem, + z: z_rem, + + y: self.0.y, + }; + (chunk_coordinate, relative) + } +} impl Serialize for WorldPosition { fn serialize(&self, serializer: S) -> Result where diff --git a/pumpkin-world/src/chunk/anvil.rs b/pumpkin-world/src/chunk/anvil.rs index 983b3fea1..06705fb08 100644 --- a/pumpkin-world/src/chunk/anvil.rs +++ b/pumpkin-world/src/chunk/anvil.rs @@ -10,6 +10,7 @@ use crate::level::SaveFile; use super::{ChunkData, ChunkReader, ChunkReadingError, CompressionError}; +#[derive(Clone)] pub struct AnvilChunkReader {} impl Default for AnvilChunkReader { diff --git a/pumpkin-world/src/chunk/mod.rs b/pumpkin-world/src/chunk/mod.rs index c75de68a6..f6c71604d 100644 --- a/pumpkin-world/src/chunk/mod.rs +++ b/pumpkin-world/src/chunk/mod.rs @@ -58,7 +58,6 @@ pub struct ChunkData { pub blocks: ChunkBlocks, pub position: Vector2, } - pub struct ChunkBlocks { // TODO make this a Vec that doesn't store the upper layers that only contain air diff --git a/pumpkin-world/src/coordinates.rs b/pumpkin-world/src/coordinates.rs index fed4a49c4..f35c11453 100644 --- a/pumpkin-world/src/coordinates.rs +++ b/pumpkin-world/src/coordinates.rs @@ -1,12 +1,12 @@ use std::ops::Deref; +use crate::{WORLD_LOWEST_Y, WORLD_MAX_Y}; use derive_more::derive::{AsMut, AsRef, Display, Into}; use num_traits::{PrimInt, Signed, Unsigned}; use pumpkin_core::math::vector2::Vector2; +use pumpkin_core::math::vector3::Vector3; use serde::{Deserialize, Serialize}; -use crate::{WORLD_LOWEST_Y, WORLD_MAX_Y}; - #[derive( Serialize, Deserialize, Debug, Clone, Copy, PartialEq, Eq, AsRef, AsMut, Into, Display, )] @@ -130,3 +130,13 @@ impl ChunkRelativeXZBlockCoordinates { } } } + +impl From> for ChunkRelativeBlockCoordinates { + fn from(value: Vector3) -> Self { + Self { + x: (value.x as u8).into(), + z: (value.z as u8).into(), + y: value.y.into(), + } + } +} diff --git a/pumpkin-world/src/level.rs b/pumpkin-world/src/level.rs index 8fff3a6cf..b4207583a 100644 --- a/pumpkin-world/src/level.rs +++ b/pumpkin-world/src/level.rs @@ -1,16 +1,16 @@ use std::{collections::HashMap, path::PathBuf, sync::Arc}; -use parking_lot::{Mutex, RwLock}; -use pumpkin_core::math::vector2::Vector2; -use rayon::prelude::*; -use tokio::sync::mpsc; - use crate::{ chunk::{ anvil::AnvilChunkReader, ChunkData, ChunkParsingError, ChunkReader, ChunkReadingError, }, world_gen::{get_world_gen, Seed, WorldGenerator}, }; +use pumpkin_core::math::vector2::Vector2; +use tokio::sync::mpsc; +use tokio::sync::{Mutex, RwLock}; + +type RAMChunkStorage = Arc, Arc>>>>; /// The `Level` module provides functionality for working with chunks within or outside a Minecraft world. /// @@ -23,12 +23,12 @@ use crate::{ /// For more details on world generation, refer to the `WorldGenerator` module. pub struct Level { save_file: Option, - loaded_chunks: Arc, Arc>>>, + loaded_chunks: RAMChunkStorage, chunk_watchers: Arc, usize>>>, - chunk_reader: Box, - world_gen: Box, + chunk_reader: Arc>, + world_gen: Arc>, } - +#[derive(Clone)] pub struct SaveFile { #[expect(dead_code)] root_folder: PathBuf, @@ -37,7 +37,7 @@ pub struct SaveFile { impl Level { pub fn from_root_folder(root_folder: PathBuf) -> Self { - let world_gen = get_world_gen(Seed(0)); // TODO Read Seed from config. + let world_gen = get_world_gen(Seed(0)).into(); // TODO Read Seed from config. if root_folder.exists() { let region_folder = root_folder.join("region"); @@ -52,7 +52,7 @@ impl Level { root_folder, region_folder, }), - chunk_reader: Box::new(AnvilChunkReader::new()), + chunk_reader: Arc::new(Box::new(AnvilChunkReader::new())), loaded_chunks: Arc::new(RwLock::new(HashMap::new())), chunk_watchers: Arc::new(Mutex::new(HashMap::new())), } @@ -64,7 +64,7 @@ impl Level { Self { world_gen, save_file: None, - chunk_reader: Box::new(AnvilChunkReader::new()), + chunk_reader: Arc::new(Box::new(AnvilChunkReader::new())), loaded_chunks: Arc::new(RwLock::new(HashMap::new())), chunk_watchers: Arc::new(Mutex::new(HashMap::new())), } @@ -76,8 +76,8 @@ impl Level { /// Marks chunks as "watched" by a unique player. When no players are watching a chunk, /// it is removed from memory. Should only be called on chunks the player was not watching /// before - pub fn mark_chunk_as_newly_watched(&self, chunks: &[Vector2]) { - let mut watchers = self.chunk_watchers.lock(); + pub async fn mark_chunk_as_newly_watched(&self, chunks: &[Vector2]) { + let mut watchers = self.chunk_watchers.lock().await; for chunk in chunks { match watchers.entry(*chunk) { std::collections::hash_map::Entry::Occupied(mut occupied) => { @@ -93,9 +93,9 @@ impl Level { /// Marks chunks no longer "watched" by a unique player. When no players are watching a chunk, /// it is removed from memory. Should only be called on chunks the player was watching before - pub fn mark_chunk_as_not_watched_and_clean(&self, chunks: &[Vector2]) { + pub async fn mark_chunk_as_not_watched_and_clean(&self, chunks: &[Vector2]) { let dropped_chunks = { - let mut watchers = self.chunk_watchers.lock(); + let mut watchers = self.chunk_watchers.lock().await; chunks .iter() .filter(|chunk| match watchers.entry(**chunk) { @@ -119,7 +119,7 @@ impl Level { }) .collect::>() }; - let mut loaded_chunks = self.loaded_chunks.write(); + let mut loaded_chunks = self.loaded_chunks.write().await; let dropped_chunk_data = dropped_chunks .iter() .filter_map(|chunk| { @@ -130,7 +130,7 @@ impl Level { self.write_chunks(dropped_chunk_data); } - pub fn write_chunks(&self, _chunks_to_write: Vec<(Vector2, Arc)>) { + pub fn write_chunks(&self, _chunks_to_write: Vec<(Vector2, Arc>)>) { //TODO } @@ -138,58 +138,76 @@ impl Level { /// MUST be called from a tokio runtime thread /// /// Note: The order of the output chunks will almost never be in the same order as the order of input chunks + pub fn fetch_chunks( + &self, + chunks: &[Vector2], + channel: mpsc::Sender>>, + ) { + for chunk in chunks { + { + let chunk_location = *chunk; + let channel = channel.clone(); + let loaded_chunks = self.loaded_chunks.clone(); + let chunk_reader = self.chunk_reader.clone(); + let save_file = self.save_file.clone(); + let world_gen = self.world_gen.clone(); + tokio::spawn(async move { + let loaded_chunks_read = loaded_chunks.read().await; + let possibly_loaded_chunk = loaded_chunks_read.get(&chunk_location).cloned(); + drop(loaded_chunks_read); + match possibly_loaded_chunk { + Some(chunk) => { + let chunk = chunk.clone(); + channel.send(chunk).await.unwrap(); + } + None => { + let chunk_data = match save_file { + Some(save_file) => { + match chunk_reader.read_chunk(&save_file, &chunk_location) { + Ok(data) => Ok(Arc::new(RwLock::new(data))), + Err( + ChunkReadingError::ChunkNotExist + | ChunkReadingError::ParsingError( + ChunkParsingError::ChunkNotGenerated, + ), + ) => { + // This chunk was not generated yet. + let chunk = Arc::new(RwLock::new( + world_gen.generate_chunk(chunk_location), + )); + let mut loaded_chunks = loaded_chunks.write().await; + loaded_chunks.insert(chunk_location, chunk.clone()); + drop(loaded_chunks); + Ok(chunk) + } + Err(err) => Err(err), + } + } + None => { + // There is no savefile yet -> generate the chunks + let chunk = Arc::new(RwLock::new( + world_gen.generate_chunk(chunk_location), + )); - pub fn fetch_chunks(&self, chunks: &[Vector2], channel: mpsc::Sender>) { - chunks.into_par_iter().for_each(|at| { - let channel = channel.clone(); - - let maybe_chunk = { - let loaded_chunks = self.loaded_chunks.read(); - loaded_chunks.get(at).cloned() - } - .or_else(|| { - let chunk_data = match &self.save_file { - Some(save_file) => { - match self.chunk_reader.read_chunk(save_file, at) { - Ok(data) => Ok(Arc::new(data)), - Err( - ChunkReadingError::ChunkNotExist - | ChunkReadingError::ParsingError( - ChunkParsingError::ChunkNotGenerated, - ), - ) => { - // This chunk was not generated yet. - let chunk = Arc::new(self.world_gen.generate_chunk(*at)); - Ok(chunk) + let mut loaded_chunks = loaded_chunks.write().await; + loaded_chunks.insert(chunk_location, chunk.clone()); + Ok(chunk) + } + }; + match chunk_data { + Ok(data) => channel.send(data).await.unwrap(), + Err(err) => { + log::warn!( + "Failed to read chunk {:?}: {:?}", + chunk_location, + err + ); + } } - Err(err) => Err(err), } } - None => { - // There is no savefile yet -> generate the chunks - let chunk = Arc::new(self.world_gen.generate_chunk(*at)); - Ok(chunk) - } - }; - match chunk_data { - Ok(data) => Some(data), - Err(err) => { - // TODO: Panic here? - log::warn!("Failed to read chunk {:?}: {:?}", at, err); - None - } - } - }); - match maybe_chunk { - Some(chunk) => { - channel - .blocking_send(chunk.clone()) - .expect("Failed sending ChunkData."); - } - None => { - log::error!("Unable to send chunk {:?}!", at); - } - }; - }) + }); + } + } } } diff --git a/pumpkin/src/client/player_packet.rs b/pumpkin/src/client/player_packet.rs index feec47f28..3d893018d 100644 --- a/pumpkin/src/client/player_packet.rs +++ b/pumpkin/src/client/player_packet.rs @@ -8,8 +8,9 @@ use crate::{ }; use num_traits::FromPrimitive; use pumpkin_config::ADVANCED_CONFIG; +use pumpkin_core::math::position::WorldPosition; use pumpkin_core::{ - math::{position::WorldPosition, vector3::Vector3, wrap_degrees}, + math::{vector3::Vector3, wrap_degrees}, text::TextComponent, GameMode, }; @@ -18,9 +19,9 @@ use pumpkin_inventory::{InventoryError, WindowType}; use pumpkin_protocol::server::play::{SCloseContainer, SKeepAlive, SSetPlayerGround, SUseItem}; use pumpkin_protocol::{ client::play::{ - Animation, CAcknowledgeBlockChange, CBlockUpdate, CEntityAnimation, CEntityVelocity, - CHeadRot, CHurtAnimation, CPingResponse, CPlayerChatMessage, CUpdateEntityPos, - CUpdateEntityPosRot, CUpdateEntityRot, CWorldEvent, FilterType, + Animation, CAcknowledgeBlockChange, CEntityAnimation, CEntityVelocity, CHeadRot, + CHurtAnimation, CPingResponse, CPlayerChatMessage, CUpdateEntityPos, CUpdateEntityPosRot, + CUpdateEntityRot, FilterType, }, server::play::{ Action, ActionType, SChatCommand, SChatMessage, SClientInformationPlay, SConfirmTeleport, @@ -29,7 +30,7 @@ use pumpkin_protocol::{ SSwingArm, SUseItemOn, Status, }, }; -use pumpkin_world::block::{BlockFace, BlockState}; +use pumpkin_world::block::{BlockFace, BlockId, BlockState}; use pumpkin_world::global_registry; use super::PlayerConfig; @@ -486,13 +487,7 @@ impl Player { // TODO: currently this is always dirt replace it let entity = &self.living_entity.entity; let world = &entity.world; - world - .broadcast_packet_all(&CWorldEvent::new(2001, &location, 11, false)) - .await; - // AIR - world - .broadcast_packet_all(&CBlockUpdate::new(&location, 0.into())) - .await; + world.break_block(location).await; } } Status::CancelledDigging => { @@ -522,13 +517,7 @@ impl Player { // TODO: currently this is always dirt replace it let entity = &self.living_entity.entity; let world = &entity.world; - world - .broadcast_packet_all(&CWorldEvent::new(2001, &location, 11, false)) - .await; - // AIR - world - .broadcast_packet_all(&CBlockUpdate::new(&location, 0.into())) - .await; + world.break_block(location).await; // TODO: Send this every tick self.client .send_packet(&CAcknowledgeBlockChange::new(player_action.sequence)) @@ -592,17 +581,14 @@ impl Player { if let Ok(block_state_id) = BlockState::new(minecraft_id, None) { let entity = &self.living_entity.entity; let world = &entity.world; + world - .broadcast_packet_all(&CBlockUpdate::new( - &location, - block_state_id.get_id_mojang_repr().into(), - )) - .await; - world - .broadcast_packet_all(&CBlockUpdate::new( - &WorldPosition(location.0 + face.to_offset()), - block_state_id.get_id_mojang_repr().into(), - )) + .set_block( + WorldPosition(location.0 + face.to_offset()), + BlockId { + data: block_state_id.get_id_mojang_repr() as u16, + }, + ) .await; } } diff --git a/pumpkin/src/world/mod.rs b/pumpkin/src/world/mod.rs index 7aec512b8..f1d0e0023 100644 --- a/pumpkin/src/world/mod.rs +++ b/pumpkin/src/world/mod.rs @@ -8,8 +8,10 @@ use crate::{ }; use num_traits::ToPrimitive; use pumpkin_config::BasicConfiguration; +use pumpkin_core::math::position::WorldPosition; use pumpkin_core::math::vector2::Vector2; use pumpkin_entity::{entity_type::EntityType, EntityId}; +use pumpkin_protocol::client::play::{CBlockUpdate, CWorldEvent}; use pumpkin_protocol::{ client::play::{ CChunkData, CGameEvent, CLogin, CPlayerAbilities, CPlayerInfoUpdate, CRemoveEntities, @@ -17,10 +19,13 @@ use pumpkin_protocol::{ }, ClientPacket, VarInt, }; +use pumpkin_world::block::BlockId; +use pumpkin_world::chunk::ChunkData; +use pumpkin_world::coordinates::ChunkRelativeBlockCoordinates; use pumpkin_world::level::Level; use scoreboard::Scoreboard; -use tokio::sync::mpsc; use tokio::sync::Mutex; +use tokio::sync::{mpsc, RwLock}; pub mod scoreboard; @@ -273,15 +278,15 @@ impl World { pub async fn mark_chunks_as_not_watched(&self, chunks: &[Vector2]) { let level = self.level.lock().await; - level.mark_chunk_as_not_watched_and_clean(chunks); + level.mark_chunk_as_not_watched_and_clean(chunks).await; } pub async fn mark_chunks_as_watched(&self, chunks: &[Vector2]) { let level = self.level.lock().await; - level.mark_chunk_as_newly_watched(chunks); + level.mark_chunk_as_newly_watched(chunks).await; } - fn spawn_world_chunks(&self, client: Arc, chunks: Vec>, distance: i32) { + async fn spawn_world_chunks(&self, client: Arc, chunks: Vec>) { if client.closed.load(std::sync::atomic::Ordering::Relaxed) { log::info!( "The connection with {} has closed before world chunks were spawned", @@ -290,26 +295,17 @@ impl World { return; } let inst = std::time::Instant::now(); - let (sender, mut chunk_receiver) = mpsc::channel(distance as usize); - let client_id = client.id; + let chunks = self.get_chunks(chunks).await; - let level = self.level.clone(); - let chunks = Arc::new(chunks); tokio::spawn(async move { - log::debug!("Spawned chunk fetcher for {}", client_id); - let level = level.lock().await; - level.fetch_chunks(&chunks, sender); - }); - - tokio::spawn(async move { - log::debug!("Spawned chunk sender for {}", client_id); - while let Some(chunk_data) = chunk_receiver.recv().await { - // dbg!(chunk_pos); + for chunk_data in chunks { + let chunk_data = chunk_data.read().await; + let packet = CChunkData(&chunk_data); #[cfg(debug_assertions)] if chunk_data.position == (0, 0).into() { use pumpkin_protocol::bytebuf::ByteBuffer; let mut test = ByteBuffer::empty(); - CChunkData(&chunk_data).write(&mut test); + packet.write(&mut test); let len = test.buf().len(); log::debug!( "Chunk packet size: {}B {}KB {}MB", @@ -322,7 +318,7 @@ impl World { // TODO: Queue player packs in a queue so we don't need to check if its closed before // sending if !client.closed.load(std::sync::atomic::Ordering::Relaxed) { - client.send_packet(&CChunkData(&chunk_data)).await; + client.send_packet(&packet).await; } } @@ -374,4 +370,55 @@ impl World { self.broadcast_packet_all(&CRemoveEntities::new(&[entity.entity_id.into()])) .await; } + pub async fn set_block(&self, position: WorldPosition, block_id: BlockId) { + let (chunk_coordinate, relative_coordinates) = position.chunk_and_chunk_relative_position(); + + // Since we divide by 16 remnant can never exceed u8 + let relative = ChunkRelativeBlockCoordinates::from(relative_coordinates); + + let chunk = self.get_chunks(vec![chunk_coordinate]).await[0].clone(); + chunk.write().await.blocks.set_block(relative, block_id); + + self.broadcast_packet_all(&CBlockUpdate::new( + &position, + i32::from(block_id.data).into(), + )) + .await; + } + + pub async fn get_chunks(&self, chunks: Vec>) -> Vec>> { + let (sender, mut receive) = mpsc::channel(chunks.len()); + { + let level = self.level.clone(); + tokio::spawn(async move { level.lock().await.fetch_chunks(&chunks, sender) }); + } + tokio::spawn(async move { + let mut received = vec![]; + + while let Some(chunk) = receive.recv().await { + received.push(chunk); + } + received + }) + .await + .unwrap() + } + + pub async fn break_block(&self, position: WorldPosition) { + self.set_block(position, BlockId { data: 0 }).await; + + self.broadcast_packet_all(&CWorldEvent::new(2001, &position, 11, false)) + .await; + } + + pub async fn get_block(&self, position: WorldPosition) -> BlockId { + let (chunk, relative) = position.chunk_and_chunk_relative_position(); + let relative = ChunkRelativeBlockCoordinates::from(relative); + self.get_chunks(vec![chunk]).await[0] + .clone() + .read() + .await + .blocks + .get_block(relative) + } } diff --git a/pumpkin/src/world/player_chunker.rs b/pumpkin/src/world/player_chunker.rs index 3965ba560..7842245f9 100644 --- a/pumpkin/src/world/player_chunker.rs +++ b/pumpkin/src/world/player_chunker.rs @@ -74,7 +74,9 @@ pub async fn player_join(world: &World, player: Arc) { if !loading_chunks.is_empty() { world.mark_chunks_as_watched(&loading_chunks).await; - world.spawn_world_chunks(player.client.clone(), loading_chunks, view_distance); + world + .spawn_world_chunks(player.client.clone(), loading_chunks) + .await; } if !unloading_chunks.is_empty() { @@ -134,7 +136,8 @@ pub async fn update_position(player: &Player) { entity.world.mark_chunks_as_watched(&loading_chunks).await; entity .world - .spawn_world_chunks(player.client.clone(), loading_chunks, view_distance); + .spawn_world_chunks(player.client.clone(), loading_chunks) + .await; } if !unloading_chunks.is_empty() {