diff --git a/Cargo.lock b/Cargo.lock index 5b893d2..287034b 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -115,6 +115,51 @@ dependencies = [ "cfg-if", ] +[[package]] +name = "crossbeam-deque" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9dd111b7b7f7d55b72c0a6ae361660ee5853c9af73f70c3c2ef6858b950e2e51" +dependencies = [ + "crossbeam-epoch", + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-epoch" +version = "0.9.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5b82ac4a3c2ca9c3460964f020e1402edd5753411d7737aa39c3714ad1b5420e" +dependencies = [ + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-utils" +version = "0.8.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d0a5c400df2834b80a4c3327b3aad3a4c4cd4de0629063962b03235697506a28" + +[[package]] +name = "derive_more" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4a9b99b9cbbe49445b21764dc0625032a89b145a2642e67603e1c936f5458d05" +dependencies = [ + "derive_more-impl", +] + +[[package]] +name = "derive_more-impl" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cb7330aeadfbe296029522e6c40f315320aba36fc43a5b3632f3795348f3bd22" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.87", +] + [[package]] name = "disjoint-set" version = "0.0.2" @@ -188,6 +233,17 @@ dependencies = [ "miniz_oxide", ] +[[package]] +name = "getrandom" +version = "0.2.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c4567c8db10ae91089c99af84c68c38da3ec2f087c3f82960bcdbf3656b6f4d7" +dependencies = [ + "cfg-if", + "libc", + "wasi", +] + [[package]] name = "hashbrown" version = "0.12.3" @@ -281,9 +337,9 @@ checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646" [[package]] name = "libc" -version = "0.2.146" +version = "0.2.169" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f92be4933c13fd498862a9e02a3055f8a8d9c039ce33db97306fd5a6caa7f29b" +checksum = "b5aba8db14291edd000dfcc4d620c7ebfb122c613afb886ca8803fa4e128a20a" [[package]] name = "log" @@ -504,6 +560,15 @@ version = "0.2.9" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e0a7ae3ac2f1173085d398531c705756c94a4c56843785df85a60c1a0afac116" +[[package]] +name = "ppv-lite86" +version = "0.2.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "77957b295656769bb8ad2b6a6b09d897d94f05c41b069aede1fcdaa675eaea04" +dependencies = [ + "zerocopy", +] + [[package]] name = "proc-macro-crate" version = "1.3.1" @@ -540,9 +605,9 @@ dependencies = [ [[package]] name = "proc-macro2" -version = "1.0.66" +version = "1.0.89" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "18fb31db3f9bddb2ea821cde30a9f70117e3f119938b5ee630b7403aa6e2ead9" +checksum = "f139b0662de085916d1fb67d2b4169d1addddda1919e696f3252b740b629986e" dependencies = [ "unicode-ident", ] @@ -555,17 +620,68 @@ checksum = "07589615d719a60c8dd8a4622e7946465dfef20d1a428f969e3443e7386d5f45" [[package]] name = "quote" -version = "1.0.28" +version = "1.0.37" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1b9ab9c7eadfd8df19006f1cf1a4aed13540ed5cbc047010ece5826e10825488" +checksum = "b5b9d34b8991d19d98081b46eacdd8eb58c6f2b201139f7c5f643cc155a633af" dependencies = [ "proc-macro2", ] +[[package]] +name = "rand" +version = "0.8.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404" +dependencies = [ + "libc", + "rand_chacha", + "rand_core", +] + +[[package]] +name = "rand_chacha" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88" +dependencies = [ + "ppv-lite86", + "rand_core", +] + +[[package]] +name = "rand_core" +version = "0.6.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c" +dependencies = [ + "getrandom", +] + +[[package]] +name = "rayon" +version = "1.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b418a60154510ca1a002a752ca9714984e21e4241e804d32555251faf8b78ffa" +dependencies = [ + "either", + "rayon-core", +] + +[[package]] +name = "rayon-core" +version = "1.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1465873a3dfdaa8ae7cb14b4383657caab0b3e8a0aa9ae8e04b044854c8dfce2" +dependencies = [ + "crossbeam-deque", + "crossbeam-utils", +] + [[package]] name = "redstone-compiler" version = "0.1.0" dependencies = [ + "derive_more", "disjoint-set", "eyre", "fastanvil", @@ -574,6 +690,8 @@ dependencies = [ "itertools", "petgraph", "quine-mc_cluskey", + "rand", + "rayon", "serde", "serde_json", "structopt", @@ -640,7 +758,7 @@ checksum = "aafe972d60b0b9bee71a91b92fee2d4fb3c9d7e8f6b179aa99f27203d99a4816" dependencies = [ "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.87", ] [[package]] @@ -735,7 +853,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "7b0417ef015feb394e61f9bfa3ea15125368fa13474400d14e79bc7f52c17d4b" dependencies = [ "quote", - "syn 2.0.28", + "syn 2.0.87", ] [[package]] @@ -792,9 +910,9 @@ dependencies = [ [[package]] name = "syn" -version = "2.0.28" +version = "2.0.87" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "04361975b3f5e348b2189d8dc55bc942f278b2d482a6a0365de5bdd62d351567" +checksum = "25aa4ce346d03a6dcd68dd8b4010bcb74e54e62c90c573f394c46eae99aba32d" dependencies = [ "proc-macro2", "quote", @@ -827,7 +945,7 @@ checksum = "f9456a42c5b0d803c8cd86e73dd7cc9edd429499f37a3550d286d5e86720569f" dependencies = [ "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.87", ] [[package]] @@ -877,7 +995,7 @@ checksum = "0f57e3ca2a01450b1a921183a9c9cbfda207fd822cef4ccb00a65402cbba7a74" dependencies = [ "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.87", ] [[package]] @@ -961,6 +1079,12 @@ dependencies = [ "winapi-util", ] +[[package]] +name = "wasi" +version = "0.11.0+wasi-snapshot-preview1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" + [[package]] name = "winapi" version = "0.3.9" @@ -1000,3 +1124,24 @@ checksum = "5504cc7644f4b593cbc05c4a55bf9bd4e94b867c3c0bd440934174d50482427d" dependencies = [ "memchr", ] + +[[package]] +name = "zerocopy" +version = "0.7.35" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1b9b4fd18abc82b8136838da5d50bae7bdea537c574d8dc1a34ed098d6c166f0" +dependencies = [ + "byteorder", + "zerocopy-derive", +] + +[[package]] +name = "zerocopy-derive" +version = "0.7.35" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fa4f8080344d4671fb4e831a13ad1e68092748387dfc4f55e356242fae12ce3e" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.87", +] diff --git a/Cargo.toml b/Cargo.toml index da04193..409fc7c 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -6,6 +6,7 @@ edition = "2021" # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html [dependencies] +derive_more = { version = "1.0.0", features = ["deref"] } disjoint-set = "0.0.2" eyre = "0.6.8" fastnbt = "2" @@ -13,6 +14,8 @@ fastanvil = "0.26" itertools = "0.11.0" petgraph = "0.6.3" quine-mc_cluskey = "0.2.4" +rand = "0.8.5" +rayon = "1.10.0" serde_json = "1.0.96" structopt = "0.3.26" sv-parser = "0.13.1" diff --git a/src/graph/logic/mod.rs b/src/graph/logic/mod.rs index e8d3ed5..53132f5 100644 --- a/src/graph/logic/mod.rs +++ b/src/graph/logic/mod.rs @@ -1,8 +1,20 @@ +use crate::transform::logic::LogicGraphTransformer; + use super::Graph; pub mod builder; -#[derive(Debug, Clone)] +#[derive(Debug, Clone, derive_more::Deref)] pub struct LogicGraph { + #[deref] pub graph: Graph, } + +impl LogicGraph { + pub fn prepare_place(self) -> eyre::Result { + let mut transform = LogicGraphTransformer::new(self); + transform.decompose_and()?; + transform.remove_double_neg_expression(); + Ok(transform.finish()) + } +} diff --git a/src/graph/mod.rs b/src/graph/mod.rs index 4b8138b..e1e8df5 100644 --- a/src/graph/mod.rs +++ b/src/graph/mod.rs @@ -72,6 +72,25 @@ impl GraphNodeKind { _ => unreachable!(), } } + + pub fn is_input(&self) -> bool { + matches!(self, GraphNodeKind::Input(_)) + } + + pub fn is_output(&self) -> bool { + matches!(self, GraphNodeKind::Output(_)) + } + + pub fn is_logic(&self) -> bool { + matches!(self, GraphNodeKind::Logic(_)) + } + + pub fn as_logic(&self) -> Option<&Logic> { + match self { + GraphNodeKind::Logic(inner) => Some(inner), + _ => None, + } + } } #[derive(Default, Debug, Clone)] @@ -182,7 +201,7 @@ impl Graph { .collect(); for (from, to) in replace_targets { - let mut node = other.nodes.iter_mut().find(|node| node.id == from).unwrap(); + let node = other.nodes.iter_mut().find(|node| node.id == from).unwrap(); self.find_node_by_id_mut(to) .unwrap() diff --git a/src/graph/world/builder.rs b/src/graph/world/builder.rs index 35ac219..52be6cd 100644 --- a/src/graph/world/builder.rs +++ b/src/graph/world/builder.rs @@ -44,6 +44,10 @@ impl PlaceBound { self.2 } + pub fn is_bound_on(&self, world: &World3D) -> bool { + world.size.bound_on(self.position()) + } + pub fn propagation_bound(&self, kind: &BlockKind, world: Option<&World3D>) -> Vec { let dir = self.direction(); let pos = self.position(); @@ -51,11 +55,11 @@ impl PlaceBound { match kind { BlockKind::Switch { .. } => { let result = pos - .forwards_except(&dir) + .forwards_except(dir) .into_iter() - .map(|pos_src| PlaceBound(PropagateType::Torch, pos_src, pos.diff(&pos_src))) + .map(|pos_src| PlaceBound(PropagateType::Torch, pos_src, pos.diff(pos_src))) .chain(|| -> Option { - let Some(pos) = pos.walk(&dir) else { + let Some(pos) = pos.walk(dir) else { return None; }; @@ -67,7 +71,7 @@ impl PlaceBound { } BlockKind::Redstone { state, .. } => { let world = world.unwrap(); - let has_up_block = world[&pos.up()].kind.is_cobble(); + let has_up_block = world[pos.up()].kind.is_cobble(); let mut propagate_targets = Vec::new(); propagate_targets.extend(pos.cardinal_redstone(*state)); @@ -77,12 +81,12 @@ impl PlaceBound { pos.up() .cardinal_redstone(*state) .into_iter() - .filter(|up_cardinal| world[up_cardinal].kind.is_redstone()), + .filter(|&up_cardinal| world[up_cardinal].kind.is_redstone()), ); } if let Some(down_pos) = pos.down() { - if !world[&down_pos].kind.is_cobble() { + if !world[down_pos].kind.is_cobble() { // Ensure redstone floors must have a block unreachable!(); } @@ -91,29 +95,29 @@ impl PlaceBound { propagate_targets.extend( pos.cardinal_redstone(*state) .into_iter() - .filter(|pos| !world[&pos].kind.is_cobble()) - .filter_map(|pos| pos.walk(&Direction::Bottom)) - .filter(|pos| world[&pos].kind.is_redstone()), + .filter(|&pos| !world[pos].kind.is_cobble()) + .filter_map(|pos| pos.walk(Direction::Bottom)) + .filter(|&pos| world[pos].kind.is_redstone()), ); } propagate_targets .iter() - .map(|pos_src| PlaceBound(PropagateType::Soft, *pos_src, pos_src.diff(&pos))) + .map(|pos_src| PlaceBound(PropagateType::Soft, *pos_src, pos_src.diff(pos))) .collect_vec() } BlockKind::Torch { .. } => { let result = match dir { Direction::Bottom => pos.cardinal(), Direction::East | Direction::West | Direction::South | Direction::North => { - let mut positions = pos.cardinal_except(&dir); + let mut positions = pos.cardinal_except(dir); positions.extend(pos.down()); positions } _ => unreachable!(), } .into_iter() - .map(|pos_src| PlaceBound(PropagateType::Torch, pos_src, pos_src.diff(&pos))) + .map(|pos_src| PlaceBound(PropagateType::Torch, pos_src, pos_src.diff(pos))) .chain(Some(PlaceBound( PropagateType::Hard, pos.up(), @@ -124,7 +128,7 @@ impl PlaceBound { result } BlockKind::Repeater { .. } => { - let walk = pos.walk(&&dir.inverse()); + let walk = pos.walk(dir.inverse()); let mut result: Vec = Vec::new(); if let Some(pos) = walk { @@ -140,12 +144,96 @@ impl PlaceBound { BlockKind::RedstoneBlock => pos .forwards() .into_iter() - .map(|pos_src| PlaceBound(PropagateType::Soft, pos_src, pos_src.diff(&pos))) + .map(|pos_src| PlaceBound(PropagateType::Soft, pos_src, pos_src.diff(pos))) .collect_vec(), BlockKind::Piston { .. } => unimplemented!(), BlockKind::Air | BlockKind::Cobble { .. } => unreachable!(), } } + + pub fn propagate_to(&self, world: &World3D) -> Vec<(Direction, Position)> { + let propagate_type = self.propagation_type(); + let pos = self.position(); + let dir = self.direction(); + let block = &world[pos]; + + match block.kind { + BlockKind::Air + | BlockKind::Switch { .. } + | BlockKind::RedstoneBlock + | BlockKind::Torch { .. } => Vec::new(), + // 코블에 붙어있는 레드스톤 토치, 리피터만 반응함 + BlockKind::Cobble { .. } => { + if matches!(propagate_type, PropagateType::Torch) { + return Vec::new(); + } + + let mut cardinal_propagation = pos + .cardinal_except(dir) + .iter() + .filter(|&&pos| world.size.bound_on(pos)) + .filter_map(|&pos_src| match world[pos_src].kind { + BlockKind::Torch { .. } => match propagate_type { + PropagateType::Torch => None, + PropagateType::Soft | PropagateType::Hard | PropagateType::Repeater => { + if pos_src.walk(world[pos_src].direction).unwrap() == pos { + Some((Direction::None, pos_src)) + } else { + None + } + } + }, + BlockKind::Repeater { .. } => match propagate_type { + PropagateType::Torch => None, + PropagateType::Soft | PropagateType::Hard | PropagateType::Repeater => { + if pos_src.walk(world[pos_src].direction).unwrap() == pos { + Some((Direction::None, pos_src)) + } else { + None + } + } + }, + BlockKind::Redstone { .. } => match propagate_type { + PropagateType::Soft | PropagateType::Torch => None, + PropagateType::Hard | PropagateType::Repeater => { + Some((Direction::None, pos_src)) + } + }, + _ => None, + }) + .collect_vec(); + + let up_pos = pos.up(); + if world.size.bound_on(up_pos) { + let up_block = &world[up_pos]; + if (up_block.direction == Direction::Bottom + && matches!(up_block.kind, BlockKind::Torch { .. })) + || (!matches!(propagate_type, PropagateType::Soft) + && matches!(up_block.kind, BlockKind::Redstone { .. })) + { + cardinal_propagation.push((Direction::None, up_pos)); + } + } + + cardinal_propagation + } + BlockKind::Redstone { .. } => vec![(Direction::None, pos)], + BlockKind::Repeater { .. } => { + if block.direction == dir { + vec![(Direction::None, pos)] + } else if block.direction.is_othogonal_plane(dir) { + // lock + match propagate_type { + PropagateType::Repeater => vec![(block.direction, pos)], + _ => vec![], + } + } else { + vec![] + } + } + BlockKind::Piston { .. } => todo!(), + } + } } impl WorldGraphBuilder { @@ -196,12 +284,12 @@ impl WorldGraphBuilder { } visit.insert(*pos); - let block = &self.world[&pos]; + let block = &self.world[*pos]; let bound = PlaceBound(PropagateType::Soft, *pos, block.direction); let propagation_targets = bound.propagation_bound(&block.kind, Some(&self.world)); let mut visits = propagation_targets .into_iter() - .map(|bound| self.propagate(bound)) + .map(|bound| bound.propagate_to(&self.world)) .flatten() .collect_vec(); @@ -219,87 +307,6 @@ impl WorldGraphBuilder { } } - fn propagate(&self, bound: PlaceBound) -> Vec<(Direction, Position)> { - let propagate_type = bound.propagation_type(); - let pos = bound.position(); - let dir = bound.direction(); - let block = &self.world[&pos]; - - match block.kind { - BlockKind::Air - | BlockKind::Switch { .. } - | BlockKind::RedstoneBlock - | BlockKind::Torch { .. } => Vec::new(), - // 코블에 붙어있는 레드스톤 토치, 리피터만 반응함 - BlockKind::Cobble { .. } => { - if matches!(propagate_type, PropagateType::Torch) { - return Vec::new(); - } - - let mut cardinal_propagation = pos - .cardinal_except(&dir) - .iter() - .filter_map(|pos_src| match &self.world[pos_src].kind { - BlockKind::Torch { .. } => match propagate_type { - PropagateType::Torch => None, - PropagateType::Soft | PropagateType::Hard | PropagateType::Repeater => { - if pos_src.walk(&self.world[pos_src].direction).unwrap() == pos { - Some((Direction::None, *pos_src)) - } else { - None - } - } - }, - BlockKind::Repeater { .. } => match propagate_type { - PropagateType::Torch => None, - PropagateType::Soft | PropagateType::Hard | PropagateType::Repeater => { - if pos_src.walk(&self.world[pos_src].direction).unwrap() == pos { - Some((Direction::None, *pos_src)) - } else { - None - } - } - }, - BlockKind::Redstone { .. } => match propagate_type { - PropagateType::Soft | PropagateType::Torch => None, - PropagateType::Hard | PropagateType::Repeater => { - Some((Direction::None, *pos_src)) - } - }, - _ => None, - }) - .collect_vec(); - - let up_pos = pos.up(); - let up_block = &self.world[&up_pos]; - if (up_block.direction == Direction::Bottom - && matches!(up_block.kind, BlockKind::Torch { .. })) - || (!matches!(propagate_type, PropagateType::Soft) - && matches!(up_block.kind, BlockKind::Redstone { .. })) - { - cardinal_propagation.push((Direction::None, up_pos)); - } - - cardinal_propagation - } - BlockKind::Redstone { .. } => vec![(Direction::None, pos)], - BlockKind::Repeater { .. } => { - if block.direction == dir { - vec![(Direction::None, pos)] - } else if block.direction.is_othogonal_plane(dir) { - // lock - match propagate_type { - PropagateType::Repeater => vec![(block.direction, pos)], - _ => vec![], - } - } else { - vec![] - } - } - BlockKind::Piston { .. } => todo!(), - } - } - fn build_nodes(&mut self) -> (Vec, HashMap) { let mut graph_id: HashMap = HashMap::new(); let mut nodes: HashMap = HashMap::new(); @@ -307,27 +314,27 @@ impl WorldGraphBuilder { for (index, pos) in self .queue .iter() - .filter(|pos| !self.world[*pos].kind.is_cobble()) + .filter(|&&pos| !self.world[pos].kind.is_cobble()) .enumerate() { graph_id.insert(*pos, index); nodes.insert( *pos, - GraphNode::new(index, GraphNodeKind::Block(self.world[pos])), + GraphNode::new(index, GraphNodeKind::Block(self.world[*pos])), ); } for pos in self .queue .iter() - .filter(|pos| !self.world[pos].kind.is_cobble()) + .filter(|&&pos| !self.world[pos].kind.is_cobble()) { let Some(outputs) = self.outputs.get(pos) else { continue; }; let node = nodes.get_mut(pos).unwrap(); - let mut block = self.world[pos]; + let mut block = self.world[*pos]; for (dir, pos) in outputs { let id = graph_id[&pos]; diff --git a/src/logic/mod.rs b/src/logic/mod.rs index e0e3612..010efb6 100644 --- a/src/logic/mod.rs +++ b/src/logic/mod.rs @@ -1,4 +1,4 @@ -#[derive(Debug, Clone, PartialEq)] +#[derive(Debug, Copy, Clone, PartialEq)] pub enum LogicType { Not, And, @@ -15,9 +15,18 @@ impl LogicType { LogicType::Xor => "Xor".to_owned(), } } + + pub fn is_not(&self) -> bool { + matches!(self, LogicType::Not) + } + + pub fn is_or(&self) -> bool { + matches!(self, LogicType::Or) + } } -#[derive(Debug, Clone)] +#[derive(Debug, Copy, Clone, derive_more::Deref)] pub struct Logic { + #[deref] pub logic_type: LogicType, } diff --git a/src/nbt/mod.rs b/src/nbt/mod.rs index 5b95fde..33c2806 100644 --- a/src/nbt/mod.rs +++ b/src/nbt/mod.rs @@ -82,8 +82,8 @@ impl From<&World3D> for NBTRoot { } impl NBTRoot { - pub fn load(path: &PathBuf) -> eyre::Result { - let file = File::open(path).unwrap(); + pub fn load(path: impl Into) -> eyre::Result { + let file = File::open(path.into()).unwrap(); let mut decoder = GzDecoder::new(file); let mut bytes = vec![]; @@ -92,9 +92,9 @@ impl NBTRoot { Ok(fastnbt::from_bytes(&bytes).unwrap()) } - pub fn save(&self, path: &PathBuf) { + pub fn save(&self, path: impl Into) { let new_bytes = fastnbt::to_bytes(&self).unwrap(); - let outfile = File::create(path).unwrap(); + let outfile = File::create(path.into()).unwrap(); let mut encoder = GzEncoder::new(outfile, Compression::best()); encoder.write_all(&new_bytes).unwrap(); } @@ -108,30 +108,33 @@ fn nbt_block_name(block: &Block) -> (String, String, Option) let (palette_name, specify_name, property) = match block.kind { BlockKind::Air => ("air", "air".to_owned(), None), BlockKind::Cobble { .. } => ("stone_bricks", "stone_bricks".to_owned(), None), - BlockKind::Switch { is_on } => ( - "lever", - "lever".to_owned(), - Some(NBTPaletteProperty { - face: match block.direction { - Direction::Bottom => Some("floor".to_owned()), - Direction::Top => Some("ceiling".to_owned()), - Direction::East | Direction::West | Direction::South | Direction::North => { - Some("wall".to_owned()) - } - _ => unreachable!(), - }, - facing: match block.direction { - Direction::Bottom | Direction::Top => None, - Direction::East => Some("south".to_owned()), - Direction::West => Some("north".to_owned()), - Direction::South => Some("west".to_owned()), - Direction::North => Some("east".to_owned()), - _ => unreachable!(), - }, - powered: is_on.then(|| "true".to_owned()), - ..Default::default() - }), - ), + BlockKind::Switch { is_on } => { + let facing = match block.direction { + Direction::Bottom | Direction::Top => None, + Direction::East => Some("north".to_owned()), + Direction::West => Some("south".to_owned()), + Direction::South => Some("east".to_owned()), + Direction::North => Some("west".to_owned()), + _ => unreachable!(), + }; + ( + "lever", + format!("lever_{is_on}_{}", facing.clone().unwrap_or_default()), + Some(NBTPaletteProperty { + face: match block.direction { + Direction::Bottom => Some("floor".to_owned()), + Direction::Top => Some("ceiling".to_owned()), + Direction::East | Direction::West | Direction::South | Direction::North => { + Some("wall".to_owned()) + } + _ => unreachable!(), + }, + facing, + powered: is_on.then(|| "true".to_owned()), + ..Default::default() + }), + ) + } BlockKind::Redstone { state, strength, .. } => ( @@ -151,10 +154,10 @@ fn nbt_block_name(block: &Block) -> (String, String, Option) ("redstone_torch", "redstone_torch".to_owned(), None) } else { let facing = match block.direction { - Direction::East => "south", - Direction::West => "north", - Direction::South => "west", - Direction::North => "east", + Direction::East => "north", + Direction::West => "south", + Direction::South => "east", + Direction::North => "west", _ => unreachable!(), } .to_owned(); @@ -176,17 +179,17 @@ fn nbt_block_name(block: &Block) -> (String, String, Option) .. } => { let facing = match block.direction { - Direction::East => "north", - Direction::West => "south", - Direction::South => "east", - Direction::North => "west", + Direction::East => "south", + Direction::West => "north", + Direction::South => "west", + Direction::North => "east", _ => unreachable!(), } .to_owned(); ( "repeater", - "repeater".to_owned(), + format!("repeater_{facing}_{is_on}_{is_locked}_{delay}"), Some(NBTPaletteProperty { facing: Some(facing), delay: Some(delay.to_string()), @@ -224,15 +227,15 @@ fn world3d_to_nbt(world: &World3D) -> NBTRoot { blocks.push(NBTBlock { state: palette_index[&specify_name] as i32, - pos: (pos.2 as i32, pos.0 as i32, pos.1 as i32), + pos: (pos.1 as i32, pos.2 as i32, pos.0 as i32), }); } NBTRoot { size: ( + world.size.1 as i32, world.size.2 as i32, world.size.0 as i32, - world.size.1 as i32, ), blocks, palette, @@ -594,9 +597,12 @@ mod tests { #[test] fn unittest_import_nbt_as_world() -> eyre::Result<()> { - let nbt = NBTRoot::load(&"test/alu.nbt".into())?; - nbt.save(&"test/alu-export.nbt".into()); - let g = WorldGraphBuilder::new(&nbt.to_world()).build(); + let nbt = NBTRoot::load("test/alu.nbt")?; + let world = nbt.to_world(); + let world_3d = World3D::from(&world); + world_3d.to_nbt().save("test/alu-export.nbt"); + + let g = WorldGraphBuilder::new(&world).build(); println!("{}", g.to_graphviz()); Ok(()) diff --git a/src/transform/placer/mod.rs b/src/transform/placer/mod.rs index 39c93b4..69cfdb2 100644 --- a/src/transform/placer/mod.rs +++ b/src/transform/placer/mod.rs @@ -1,16 +1,24 @@ -use std::{collections::HashSet, iter::repeat_with}; +use std::collections::HashMap; -use petgraph::stable_graph::NodeIndex; +use eyre::ensure; +use itertools::{iproduct, Itertools}; +use rayon::iter::{IntoParallelIterator, ParallelIterator}; use crate::{ graph::{ + logic::LogicGraph, world::{ builder::{PlaceBound, PropagateType}, WorldGraph, }, - GraphNodeId, GraphNodeKind, SubGraphWithGraph, + GraphNode, GraphNodeId, GraphNodeKind, + }, + logic::LogicType, + world::{ + block::{Block, BlockKind, Direction, RedstoneState}, + position::{DimSize, Position}, + world::World3D, }, - world::{block::Block, position::Position, world::World3D}, }; #[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)] @@ -24,76 +32,330 @@ impl PlacedNode { Self { position, block } } + pub fn new_cobble(position: Position) -> Self { + Self { + position, + block: Block { + kind: BlockKind::Cobble { + on_count: 0, + on_base_count: 0, + }, + direction: Direction::None, + }, + } + } + pub fn is_propagation_target(&self) -> bool { self.block.kind.is_stick_to_redstone() || self.block.kind.is_repeater() } // signal을 보낼 수 있는 부분들의 위치를 반환합니다. - pub fn propagation_bound(&self, world: Option<&World3D>) -> HashSet { + pub fn propagation_bound(&self, world: Option<&World3D>) -> Vec { PlaceBound(PropagateType::Soft, self.position, self.block.direction) .propagation_bound(&self.block.kind, world) + } + + pub fn has_conflict(&self, world: &World3D) -> bool { + if self.block.kind.is_cobble() { + return !(world[self.position].kind.is_air() || world[self.position].kind.is_cobble()); + } + + if !world[self.position].kind.is_air() { + return true; + } + + let bounds = self.propagation_bound(Some(world)); + bounds .into_iter() - .collect() + .filter(|bound| bound.is_bound_on(world)) + .any(|bound| !bound.propagate_to(world).is_empty()) } } -pub struct LocalPlacer<'a> { - graph: SubGraphWithGraph<'a>, - try_count: usize, - max_width_size: usize, - max_height_size: usize, +pub struct LocalPlacer { + graph: LogicGraph, } pub const K_MAX_LOCAL_PLACE_NODE_COUNT: usize = 25; -impl<'a> LocalPlacer<'a> { - // you should not pass side effected sub-graph - pub fn new( - graph: SubGraphWithGraph<'a>, - try_count: usize, - max_width_size: Option, - max_height_size: Option, - ) -> eyre::Result { - assert!(try_count > 0); - - Self::verify(&graph); - - Ok(Self { - graph, - try_count, - max_width_size: max_width_size.unwrap_or(usize::MAX), - max_height_size: max_height_size.unwrap_or(usize::MAX), - }) +impl LocalPlacer { + pub fn new(graph: LogicGraph) -> eyre::Result { + let result = Self { graph }; + result.verify()?; + Ok(result) } - pub fn verify(graph: &SubGraphWithGraph<'a>) { - assert!(graph.nodes.len() > 0); - assert!(graph.nodes.len() <= K_MAX_LOCAL_PLACE_NODE_COUNT); + fn verify(&self) -> eyre::Result<()> { + ensure!(self.graph.nodes.len() > 0, ""); + ensure!( + self.graph.nodes.len() <= K_MAX_LOCAL_PLACE_NODE_COUNT, + "too large graph" + ); - for node_id in &graph.nodes { - assert!(matches!( - graph.graph.find_node_by_id(*node_id).unwrap().kind, - GraphNodeKind::Logic { .. } - )); + for node_id in &self.graph.nodes { + let kind = &self.graph.find_node_by_id(node_id.id).unwrap().kind; + ensure!( + kind.is_input() || kind.is_output() || kind.is_logic(), + "cannot place" + ); + if let Some(logic) = kind.as_logic() { + ensure!(logic.is_not() || logic.is_or(), "cannot place"); + } } + + Ok(()) } - pub fn generate(&mut self) -> WorldGraph { - let try_count = self.try_count; + pub fn generate(&mut self, finish_step: Option) -> Vec { + tracing::info!("generate starts"); + let orders = self.graph.topological_order(); + let mut queue: Vec<(World3D, HashMap)> = Vec::new(); + queue.push((World3D::new(DimSize(10, 10, 5)), Default::default())); - // TODO: make parallel - repeat_with(|| self.next_place()) - .take(try_count) - .min_by_key(|(_, c)| *c) - .unwrap() - .0 + let mut step = 0; + while step < orders.len() && Some(step) != finish_step { + let node_id = orders[step]; + let node = self.graph.find_node_by_id(node_id).unwrap(); + tracing::info!("current node: {node:?}"); + let prev_step_volume = queue.len(); + let next_queue = queue + .into_par_iter() + .flat_map(|(world, pos)| { + self.place_and_route_next_node(node, &world, &pos) + .into_iter() + .map(|(world, place_position)| { + let mut nodes_position = pos.clone(); + nodes_position.insert(node_id, place_position); + (world, nodes_position) + }) + .collect_vec() + }) + .collect::>(); + step += 1; + queue = next_queue; + tracing::info!("step - {step}: {prev_step_volume} -> {}", queue.len()); + } + + tracing::info!("generate complete"); + queue.into_iter().map(|(world, _)| world).collect() } - fn next_place(&mut self) -> (WorldGraph, usize) { - todo!() + fn place_and_route_next_node( + &self, + node: &GraphNode, + world: &World3D, + positions: &HashMap, + ) -> Vec<(World3D, Position)> { + match node.kind { + GraphNodeKind::Input(_) => input_node_kind() + .into_iter() + .flat_map(|kind| generate_inputs(world, kind)) + .collect(), + GraphNodeKind::Output(_) => output_node_kind() + .into_iter() + .flat_map(|kind| generate_place_and_routes(world, positions[&node.inputs[0]], kind)) + .collect(), + GraphNodeKind::Logic(logic) => match logic.logic_type { + LogicType::Not => not_node_kind() + .into_iter() + .flat_map(|kind| { + generate_place_and_routes(world, positions[&node.inputs[0]], kind) + }) + .collect(), + LogicType::Or => { + assert_eq!(node.inputs.len(), 2); + generate_routes( + world, + positions[&node.inputs[0]], + positions[&node.inputs[1]], + ) + } + _ => unreachable!(), + }, + _ => unreachable!(), + } + } +} + +fn input_node_kind() -> Vec { + vec![ + BlockKind::Switch { is_on: true }, + // BlockKind::Cobble { + // on_count: 0, + // on_base_count: 0, + // }, + // BlockKind::Redstone { + // on_count: 0, + // state: RedstoneState::None as usize, + // strength: 0, + // }, + // BlockKind::RedstoneBlock, + ] +} + +fn output_node_kind() -> Vec { + vec![BlockKind::Redstone { + on_count: 0, + state: RedstoneState::None as usize, + strength: 0, + }] +} + +fn not_node_kind() -> Vec { + vec![BlockKind::Torch { is_on: false }] +} + +fn place_node(world: &mut World3D, node: PlacedNode) { + world[node.position] = node.block; + if node.block.kind.is_redstone() { + world.update_redstone_states(node.position); } } +fn generate_inputs(world: &World3D, kind: BlockKind) -> Vec<(World3D, Position)> { + // 일단 바닥에 두고 생성하기 + let block = Block { + kind, + direction: Direction::Bottom, + }; + + let generate_strategy = vec![ + // x == 0에서 먼저 생성 + (block, iproduct!(0..1, 0..world.size.1, 0..world.size.2)), + // x == 0에서 못 찾으면 y == 0 에서 생성 + (block, iproduct!(0..world.size.0, 0..1, 0..world.size.2)), + ]; + + for (block, positions) in generate_strategy { + let candidates = positions + .filter_map(|(x, y, z)| { + let position = Position(x, y, z); + let placed_node = PlacedNode { position, block }; + if placed_node.has_conflict(world) { + return None; + } + + let mut new_world = world.clone(); + place_node(&mut new_world, placed_node); + Some((new_world, position)) + }) + .collect_vec(); + + if !candidates.is_empty() { + return candidates; + } + } + + Vec::default() +} + +fn generate_place_and_routes( + world: &World3D, + start: Position, + kind: BlockKind, +) -> Vec<(World3D, Position)> { + match kind { + BlockKind::Torch { .. } => generate_torch_place_and_routes(world, start, kind), + _ => unimplemented!(), + } +} + +fn generate_torch_place_and_routes( + world: &World3D, + source: Position, + kind: BlockKind, +) -> Vec<(World3D, Position)> { + let torch_strategy = [ + Direction::Bottom, + Direction::East, + Direction::West, + Direction::South, + Direction::North, + ] + .into_iter() + .map(|direction| Block { kind, direction }) + .collect_vec(); + + // start에서 최소 두 칸 떨어진 곳에 위치시킨다. + let generate_strategy = torch_strategy + .into_iter() + .cartesian_product( + iproduct!(0..world.size.0, 0..world.size.1, 0..world.size.2) + .map(|(x, y, z)| Position(x, y, z)), // .filter(|pos| source.manhattan_distance(pos) > 2), + ) + .collect_vec(); + + tracing::trace!("strategy count: {}", generate_strategy.len()); + + // 1. Place Torch and Cobble + let mut place_candidates = Vec::new(); + for (block, torch_pos) in generate_strategy { + let Some(cobble_pos) = torch_pos.walk(block.direction) else { + continue; + }; + let cobble_node = PlacedNode::new_cobble(cobble_pos); + if !world.size.bound_on(cobble_pos) || cobble_node.has_conflict(&world) { + continue; + } + + let torch_node = PlacedNode::new(torch_pos, block); + if torch_node.has_conflict(world) { + continue; + } + + let mut new_world = world.clone(); + if new_world[cobble_pos].kind.is_air() { + place_node(&mut new_world, cobble_node); + } + place_node(&mut new_world, torch_node); + place_candidates.push((new_world, torch_pos, cobble_pos)); + } + + // 2. Route Source with Torch Place Target Position + let candidates = place_candidates + .into_iter() + .flat_map(|(world, torch_pos, cobble_pos)| { + generate_routes_to_cobble(&world, source, cobble_pos) + .into_iter() + .map(|(world, _)| (world, torch_pos)) + .collect_vec() + }) + .collect_vec(); + + candidates +} + +fn generate_routes_to_cobble( + world: &World3D, + source: Position, + cobble_pos: Position, +) -> Vec<(World3D, Position)> { + let source_node = PlacedNode::new(source, world[source]); + assert!(source_node.is_propagation_target() && world[cobble_pos].kind.is_cobble()); + + // (world, pos, cost) + let mut route_candidates = Vec::new(); + for start in source_node.propagation_bound(Some(world)) { + let directly_connected = start.position() == cobble_pos + && matches!(start.propagation_type(), PropagateType::Hard); + + if directly_connected { + route_candidates.push((world.clone(), start.position(), 0)); + continue; + } + } + + route_candidates.sort_by_key(|(_, _, cost)| *cost); + route_candidates + .into_iter() + .map(|(world, pos, _)| (world, pos)) + .collect() +} + +fn generate_routes(world: &World3D, first: Position, second: Position) -> Vec<(World3D, Position)> { + todo!() +} + pub struct LocalPlacerCostEstimator<'a> { graph: &'a WorldGraph, } @@ -109,3 +371,44 @@ impl<'a> LocalPlacerCostEstimator<'a> { todo!() } } + +#[cfg(test)] +mod tests { + + use rand::{seq::IteratorRandom, thread_rng}; + + use crate::{ + graph::{ + graphviz::ToGraphvizGraph, + logic::{builder::LogicGraphBuilder, LogicGraph}, + }, + nbt::{NBTRoot, ToNBT}, + transform::placer::LocalPlacer, + world::world::World3D, + }; + + fn build_graph_from_stmt(stmt: &str, output: &str) -> eyre::Result { + LogicGraphBuilder::new(stmt.to_string()).build(output.to_string()) + } + + #[test] + fn test_generate_component_and() -> eyre::Result<()> { + tracing_subscriber::fmt::init(); + + let logic_graph = build_graph_from_stmt("a&b", "c")?.prepare_place()?; + println!("{}", logic_graph.to_graphviz()); + + let mut placer = LocalPlacer::new(logic_graph)?; + let worlds = placer.generate(Some(4)); + + let mut rng = thread_rng(); + let sampled_worlds = worlds.into_iter().choose_multiple(&mut rng, 100); + + let ww = World3D::concat_tiled(sampled_worlds); + + let nbt: NBTRoot = ww.to_nbt(); + nbt.save("test/and-gate-new.nbt"); + + Ok(()) + } +} diff --git a/src/transform/world.rs b/src/transform/world.rs index 1ad60d7..783c570 100644 --- a/src/transform/world.rs +++ b/src/transform/world.rs @@ -178,7 +178,7 @@ mod tests { #[test] fn unittest_fold_redstone() -> eyre::Result<()> { - let nbt = NBTRoot::load(&"test/alu.nbt".into())?; + let nbt = NBTRoot::load("test/alu.nbt")?; let g = WorldGraphBuilder::new(&nbt.to_world()).build(); let mut transform = WorldGraphTransformer::new(g); diff --git a/src/transform/world_to_logic.rs b/src/transform/world_to_logic.rs index 2909c28..813cfb4 100644 --- a/src/transform/world_to_logic.rs +++ b/src/transform/world_to_logic.rs @@ -26,12 +26,17 @@ impl WorldToLogicTransformer { fn verify_input(graph: &WorldGraph) -> eyre::Result<()> { // verify no repeater lock let contains_lock_repeater = graph.graph.nodes.iter().any(|node| { - let GraphNodeKind::Block(block) = node.kind else { - return false; + let GraphNodeKind::Block(block) = node.kind else { + return false; }; - let BlockKind::Repeater { lock_input1, lock_input2 , ..} = block.kind else { - return false; + let BlockKind::Repeater { + lock_input1, + lock_input2, + .. + } = block.kind + else { + return false; }; lock_input1.is_some() || lock_input2.is_some() @@ -154,7 +159,7 @@ mod tests { #[test] fn unittest_world_to_logic_graph() -> eyre::Result<()> { - let nbt = NBTRoot::load(&"test/alu.nbt".into())?; + let nbt = NBTRoot::load("test/alu.nbt")?; let g = WorldGraphBuilder::new(&nbt.to_world()).build(); g.graph.verify()?; diff --git a/src/world/block.rs b/src/world/block.rs index 6ccbe90..e2359b9 100644 --- a/src/world/block.rs +++ b/src/world/block.rs @@ -124,6 +124,10 @@ impl BlockKind { ) } + pub fn is_air(&self) -> bool { + matches!(self, BlockKind::Air) + } + pub fn is_repeater(&self) -> bool { matches!(self, BlockKind::Repeater { .. }) } @@ -132,6 +136,10 @@ impl BlockKind { matches!(self, BlockKind::Redstone { .. }) } + pub fn is_torch(&self) -> bool { + matches!(self, BlockKind::Torch { .. }) + } + pub fn is_cobble(&self) -> bool { matches!(self, BlockKind::Cobble { .. }) } diff --git a/src/world/position.rs b/src/world/position.rs index 2a4eb20..a6b7453 100644 --- a/src/world/position.rs +++ b/src/world/position.rs @@ -1,6 +1,6 @@ use super::block::{Direction, RedstoneState, RedstoneStateType}; -// 위치 +// 위치 (x, y, z) #[derive(Debug, Copy, Clone, Eq, PartialEq, Hash, PartialOrd, Ord)] pub struct Position(pub usize, pub usize, pub usize); @@ -34,7 +34,7 @@ impl Position { result } - pub fn forwards_except(&self, dir: &Direction) -> Vec { + pub fn forwards_except(&self, dir: Direction) -> Vec { if let Some(pos) = self.walk(dir) { return self .forwards() @@ -86,7 +86,7 @@ impl Position { result } - pub fn cardinal_except(&self, dir: &Direction) -> Vec { + pub fn cardinal_except(&self, dir: Direction) -> Vec { let mut result = Vec::new(); if !matches!(dir, Direction::East) { @@ -120,7 +120,7 @@ impl Position { Some(Position(self.0, self.1, self.2 - 1)) } - pub fn walk(&self, dir: &Direction) -> Option { + pub fn walk(&self, dir: Direction) -> Option { match dir { Direction::None => Some(*self), Direction::Bottom => { @@ -150,7 +150,7 @@ impl Position { } } - pub fn diff(&self, tar: &Position) -> Direction { + pub fn diff(&self, tar: Position) -> Direction { if tar.0 > self.0 { Direction::East } else if tar.0 < self.0 { @@ -167,9 +167,19 @@ impl Position { unreachable!() } } + + pub fn manhattan_distance(&self, other: &Self) -> usize { + self.0.abs_diff(other.0) + self.1.abs_diff(other.1) + self.2.abs_diff(other.2) + } } // 사이즈 #[derive(Debug, Copy, Clone)] pub struct DimSize(pub usize, pub usize, pub usize); + +impl DimSize { + pub fn bound_on(&self, pos: Position) -> bool { + pos.0 < self.0 && pos.1 < self.1 && pos.2 < self.2 + } +} diff --git a/src/world/simulator.rs b/src/world/simulator.rs index 5e24e76..6db5bba 100644 --- a/src/world/simulator.rs +++ b/src/world/simulator.rs @@ -94,7 +94,7 @@ impl Simulator { self.queue.push_back(VecDeque::new()); for (pos, value) in states { - let BlockKind::Switch { is_on } = &mut self.world[&pos].kind else { + let BlockKind::Switch { is_on } = &mut self.world[pos].kind else { eyre::bail!("you can change only switch state!"); }; @@ -104,7 +104,7 @@ impl Simulator { *is_on = value; - pos.forwards_except(&self.world[&pos].direction) + pos.forwards_except(self.world[pos].direction) .into_iter() .map(|pos_src| Event { id: None, @@ -115,10 +115,10 @@ impl Simulator { EventType::TorchOff }, target_position: pos_src, - direction: pos_src.diff(&pos), + direction: pos_src.diff(pos), }) .chain(|| -> Option { - let Some(pos) = pos.walk(&self.world[&pos].direction) else { + let Some(pos) = pos.walk(self.world[pos].direction) else { return None; }; @@ -173,13 +173,13 @@ impl Simulator { } fn init(&mut self, world: &World) { - for (pos, block) in &world.blocks { + for (pos, block) in world.blocks.iter().copied() { match block.kind { BlockKind::Torch { is_on } if is_on => { - self.init_torch_event(&block.direction, pos); + self.init_torch_event(block.direction, pos); } BlockKind::Switch { is_on } if is_on => { - self.init_switch_event(&block.direction, pos); + self.init_switch_event(block.direction, pos); } BlockKind::RedstoneBlock => { self.init_redstone_block_event(pos); @@ -189,7 +189,7 @@ impl Simulator { } } - fn init_torch_event(&mut self, dir: &Direction, pos: &Position) { + fn init_torch_event(&mut self, dir: Direction, pos: Position) { tracing::debug!("produce torch event: {:?}, {:?}", dir, pos); let events = match dir { @@ -223,7 +223,7 @@ impl Simulator { self.queue[0].extend(events); } - fn init_switch_event(&mut self, dir: &Direction, pos: &Position) { + fn init_switch_event(&mut self, dir: Direction, pos: Position) { tracing::debug!("produce switch event: {:?}, {:?}", dir, pos); let events = pos @@ -253,7 +253,7 @@ impl Simulator { self.queue[0].extend(events); } - fn init_redstone_block_event(&mut self, pos: &Position) { + fn init_redstone_block_event(&mut self, pos: Position) { tracing::debug!("produce redstone block event: {:?}", pos); let events = pos.forwards().into_iter().map(|pos_src| Event { @@ -275,7 +275,7 @@ impl Simulator { while let Some(event) = self.queue.front_mut().unwrap().pop_front() { tracing::debug!("consume event: {:?}", event); - let mut block = self.world[&event.target_position]; + let mut block = self.world[event.target_position]; match block.kind { BlockKind::Air | BlockKind::Switch { .. } | BlockKind::RedstoneBlock => (), @@ -294,7 +294,7 @@ impl Simulator { BlockKind::Piston { .. } => todo!(), } - self.world[&event.target_position] = block; + self.world[event.target_position] = block; self.fill_event_id(); } @@ -327,7 +327,8 @@ impl Simulator { on_count, on_base_count, .. - } = block.kind else { + } = block.kind + else { unreachable!() }; @@ -343,7 +344,7 @@ impl Simulator { .target_position .forwards() .into_iter() - .filter(|pos| !self.world[pos].kind.is_cobble()) + .filter(|&pos| !self.world[pos].kind.is_cobble()) .map(|pos_src| Event { id: None, from_id: event.id, @@ -361,7 +362,7 @@ impl Simulator { } }, target_position: pos_src, - direction: pos_src.diff(&event.target_position), + direction: pos_src.diff(event.target_position), }) .collect::>(); @@ -375,10 +376,7 @@ impl Simulator { fn propagate_redstone_event(&mut self, block: &mut Block, event: &Event) -> eyre::Result<()> { tracing::debug!("consume redstone event: {:?}", block); - let BlockKind::Redstone { - state, - .. - } = &mut block.kind else { + let BlockKind::Redstone { state, .. } = &mut block.kind else { eyre::bail!("unreachable"); }; @@ -387,12 +385,12 @@ impl Simulator { propagate_targets.extend(event.target_position.cardinal_redstone(*state)); let up_pos = event.target_position.up(); - if !self.world[&up_pos].kind.is_cobble() { + if !self.world[up_pos].kind.is_cobble() { propagate_targets.extend(up_pos.cardinal_redstone(*state)); } if let Some(down_pos) = event.target_position.down() { - if !self.world[&down_pos].kind.is_cobble() { + if !self.world[down_pos].kind.is_cobble() { eyre::bail!("unreachable"); } @@ -403,9 +401,9 @@ impl Simulator { .target_position .cardinal_redstone(*state) .into_iter() - .filter(|pos| !self.world[&pos].kind.is_cobble()) - .filter_map(|pos| pos.walk(&Direction::Bottom)) - .filter(|pos| !self.world[&pos].kind.is_cobble()), + .filter(|&pos| !self.world[pos].kind.is_cobble()) + .filter_map(|pos| pos.walk(Direction::Bottom)) + .filter(|&pos| !self.world[pos].kind.is_cobble()), ); } @@ -416,10 +414,9 @@ impl Simulator { | EventType::RepeaterOff { .. } => {} EventType::TorchOn | EventType::HardOn => { let BlockKind::Redstone { - on_count, - strength, - .. - } = &mut block.kind else { + on_count, strength, .. + } = &mut block.kind + else { eyre::bail!("unreachable"); }; @@ -436,7 +433,7 @@ impl Simulator { from_id: event.id, event_type: EventType::SoftOn, target_position: pos, - direction: pos.diff(&event.target_position), + direction: pos.diff(event.target_position), }); self.push_event_to_current_tick(Event { @@ -451,10 +448,9 @@ impl Simulator { } EventType::TorchOff | EventType::HardOff => { let BlockKind::Redstone { - on_count, - strength, - .. - } = &mut block.kind else { + on_count, strength, .. + } = &mut block.kind + else { eyre::bail!("unreachable"); }; @@ -471,7 +467,7 @@ impl Simulator { from_id: event.id, event_type: EventType::SoftOff, target_position: pos, - direction: pos.diff(&event.target_position), + direction: pos.diff(event.target_position), }); self.push_event_to_current_tick(Event { @@ -487,10 +483,7 @@ impl Simulator { EventType::RedstoneOn { strength } => { let event_strength = strength; - let BlockKind::Redstone { - strength, - .. - } = &mut block.kind else { + let BlockKind::Redstone { strength, .. } = &mut block.kind else { eyre::bail!("unreachable"); }; @@ -498,13 +491,13 @@ impl Simulator { *strength = event_strength; propagate_targets.into_iter().for_each(|pos| { - if !self.world[&pos].kind.is_redstone() { + if !self.world[pos].kind.is_redstone() { self.push_event_to_current_tick(Event { id: None, from_id: event.id, event_type: EventType::SoftOn, target_position: pos, - direction: pos.diff(&event.target_position), + direction: pos.diff(event.target_position), }); } @@ -515,7 +508,7 @@ impl Simulator { strength: *strength - 1, }, target_position: pos, - direction: pos.diff(&event.target_position), + direction: pos.diff(event.target_position), }); }); @@ -524,10 +517,9 @@ impl Simulator { } EventType::RedstoneOff => { let BlockKind::Redstone { - on_count, - strength, - .. - } = &mut block.kind else { + on_count, strength, .. + } = &mut block.kind + else { eyre::bail!("unreachable"); }; @@ -535,13 +527,13 @@ impl Simulator { *strength = 0; propagate_targets.into_iter().for_each(|pos| { - if !self.world[&pos].kind.is_redstone() { + if !self.world[pos].kind.is_redstone() { self.push_event_to_current_tick(Event { id: None, from_id: event.id, event_type: EventType::SoftOff, target_position: pos, - direction: pos.diff(&event.target_position), + direction: pos.diff(event.target_position), }); } @@ -550,18 +542,18 @@ impl Simulator { from_id: event.id, event_type: EventType::RedstoneOff, target_position: pos, - direction: pos.diff(&event.target_position), + direction: pos.diff(event.target_position), }); }); } else { propagate_targets.into_iter().for_each(|pos| { - if !self.world[&pos].kind.is_redstone() { + if !self.world[pos].kind.is_redstone() { self.push_event_to_current_tick(Event { id: None, from_id: event.id, event_type: EventType::SoftOn, target_position: pos, - direction: pos.diff(&event.target_position), + direction: pos.diff(event.target_position), }); } @@ -570,7 +562,7 @@ impl Simulator { from_id: event.id, event_type: EventType::RedstoneOn { strength: 14 }, target_position: pos, - direction: pos.diff(&event.target_position), + direction: pos.diff(event.target_position), }); }); } @@ -586,7 +578,7 @@ impl Simulator { tracing::debug!("consume torch event: {:?}", block); // Cobble로부터 온 이벤트가 아닌 경우 - if !self.world[&event.target_position.walk(&event.direction).unwrap()] + if !self.world[event.target_position.walk(event.direction).unwrap()] .kind .is_cobble() { @@ -607,7 +599,7 @@ impl Simulator { match block.direction { Direction::Bottom => event.target_position.cardinal(), Direction::East | Direction::West | Direction::South | Direction::North => { - let mut positions = event.target_position.cardinal_except(&block.direction); + let mut positions = event.target_position.cardinal_except(block.direction); positions.extend(event.target_position.down()); positions } @@ -623,7 +615,7 @@ impl Simulator { EventType::TorchOff }, target_position: pos_src, - direction: pos_src.diff(&event.target_position), + direction: pos_src.diff(event.target_position), }) .chain(Some(Event { id: None, @@ -653,7 +645,8 @@ impl Simulator { is_locked, delay, .. - } = block.kind else { + } = block.kind + else { unreachable!() }; @@ -733,7 +726,7 @@ impl Simulator { block.kind.set_repeater_state(true)?; // propagate - let walk = event.target_position.walk(&event.direction.inverse()); + let walk = event.target_position.walk(event.direction.inverse()); if let Some(pos) = walk { self.push_event_to_next_tick(Event { @@ -741,7 +734,7 @@ impl Simulator { from_id: event.id, event_type: EventType::HardOn, target_position: pos, - direction: pos.diff(&event.target_position), + direction: pos.diff(event.target_position), }); } @@ -771,7 +764,7 @@ impl Simulator { block.kind.set_repeater_state(false)?; - let walk = event.target_position.walk(&event.direction.inverse()); + let walk = event.target_position.walk(event.direction.inverse()); if let Some(pos) = walk { self.push_event_to_next_tick(Event { @@ -779,7 +772,7 @@ impl Simulator { from_id: event.id, event_type: EventType::HardOff, target_position: pos, - direction: pos.diff(&event.target_position), + direction: pos.diff(event.target_position), }); } @@ -891,7 +884,7 @@ mod test { let sim = Simulator::from(&mock_world).unwrap(); - let BlockKind::Redstone { on_count, .. } = sim.world.map[0][1][1].kind else { + let BlockKind::Redstone { on_count, .. } = sim.world.map[0][1][1].kind else { unreachable!(); }; @@ -937,7 +930,11 @@ mod test { let sim = Simulator::from(&mock_world).unwrap(); - let BlockKind::Cobble { on_count, on_base_count } = sim.world.map[0][1][1].kind else { + let BlockKind::Cobble { + on_count, + on_base_count, + } = sim.world.map[0][1][1].kind + else { unreachable!(); }; @@ -990,7 +987,7 @@ mod test { sim.run().unwrap(); - let BlockKind::Redstone { strength , .. } = sim.world.map[0][2][2].kind else { + let BlockKind::Redstone { strength, .. } = sim.world.map[0][2][2].kind else { unreachable!(); }; diff --git a/src/world/world.rs b/src/world/world.rs index 5ebdda1..5412ced 100644 --- a/src/world/world.rs +++ b/src/world/world.rs @@ -1,3 +1,4 @@ +use std::cmp; use std::fmt::Debug; use std::{ collections::BTreeMap, @@ -15,6 +16,15 @@ pub struct World { pub blocks: Vec<(Position, Block)>, } +impl World { + pub fn new(size: DimSize) -> Self { + Self { + size, + blocks: Default::default(), + } + } +} + #[derive(Clone)] pub struct World3D { pub size: DimSize, @@ -23,6 +33,13 @@ pub struct World3D { } impl World3D { + pub fn new(size: DimSize) -> Self { + Self { + size, + map: vec![vec![vec![Block::default(); size.0]; size.1]; size.2], + } + } + pub fn iter_pos(&self) -> Vec { let mut result = Vec::new(); let (z, y, x) = (self.map.len(), self.map[0].len(), self.map[0][1].len()); @@ -41,7 +58,7 @@ impl World3D { pub fn iter_block(&self) -> Vec<(Position, &Block)> { self.iter_pos() .into_iter() - .map(|pos| (pos, &self[&pos])) + .map(|pos| (pos, &self[pos])) .collect_vec() } @@ -53,29 +70,27 @@ impl World3D { pub fn update_redstone_states(&mut self, pos: Position) { let BlockKind::Redstone { - on_count, - strength, - .. - } = self[&pos].kind else { + on_count, strength, .. + } = self[pos].kind + else { return; }; let mut state = 0; - let has_up_block = self[&pos.up()].kind.is_cobble(); + let has_up_block = self[pos.up()].kind.is_cobble(); - pos.cardinal().iter().for_each(|pos_src| { + pos.cardinal().iter().for_each(|&pos_src| { let flat_check = self[pos_src].kind.is_stick_to_redstone(); - let up_check = !has_up_block && self[&pos_src.up()].kind.is_redstone(); + let up_check = !has_up_block && self[pos_src.up()].kind.is_redstone(); let down_check = !self[pos_src].kind.is_cobble() && pos_src .down() - .map_or(false, |pos| self[&pos].kind.is_redstone()); - let Some(walk) = pos_src.walk(&self[pos_src].direction) else { - return; - }; - let flat_repeater_check = - matches!(self[pos_src].kind, BlockKind::Repeater { .. }) && pos == walk; + .map_or(false, |pos| self[pos].kind.is_redstone()); + let flat_repeater_check = self[pos_src].kind.is_repeater() + && pos_src + .walk(self[pos_src].direction) + .map_or(false, |walk| walk == pos); if !(flat_check || flat_repeater_check) && !(up_check || down_check) { return; @@ -98,12 +113,78 @@ impl World3D { } } - self[&pos].kind = BlockKind::Redstone { + self[pos].kind = BlockKind::Redstone { on_count, state, strength, }; } + + pub fn concat(&self, other: &World3D, direction: Direction) -> Self { + match direction { + Direction::None => unreachable!(), + Direction::Bottom | Direction::West | Direction::South => { + other.concat(self, direction.inverse()) + } + Direction::Top | Direction::North | Direction::East => { + let mut world = Self::new(DimSize( + if matches!(direction, Direction::East) { + self.size.0 + other.size.0 + } else { + cmp::max(self.size.0, other.size.0) + }, + if matches!(direction, Direction::North) { + self.size.1 + other.size.1 + } else { + cmp::max(self.size.1, other.size.1) + }, + if matches!(direction, Direction::Top) { + self.size.2 + other.size.2 + } else { + cmp::max(self.size.2, other.size.2) + }, + )); + + for (pos, block) in self.iter_block() { + world[pos] = block.clone(); + } + + for (mut pos, block) in other.iter_block() { + match direction { + Direction::East => pos.0 += self.size.0, + Direction::North => pos.1 += self.size.1, + Direction::Top => pos.2 += self.size.2, + _ => (), + } + world[pos] = block.clone(); + } + + world + } + } + } + + pub fn concat_tiled(worlds: Vec) -> Self { + let east_chunk_len = (worlds.len() as f32).sqrt() as usize + 1; + + let east_worlds = worlds + .chunks(east_chunk_len) + .into_iter() + .map(|worlds| { + let mut world = worlds[0].clone(); + for other in worlds.into_iter().skip(1) { + world = world.concat(other, Direction::East); + } + world + }) + .collect_vec(); + + let mut world = east_worlds[0].clone(); + for other in east_worlds.into_iter().skip(1) { + world = world.concat(&other, Direction::North); + } + world + } } impl<'a> From<&'a World> for World3D { @@ -142,16 +223,16 @@ impl<'a> From<&'a World> for World3D { } } -impl Index<&Position> for World3D { +impl Index for World3D { type Output = Block; - fn index(&self, index: &Position) -> &Self::Output { + fn index(&self, index: Position) -> &Self::Output { &self.map[index.2][index.1][index.0] } } -impl IndexMut<&Position> for World3D { - fn index_mut(&mut self, index: &Position) -> &mut Self::Output { +impl IndexMut for World3D { + fn index_mut(&mut self, index: Position) -> &mut Self::Output { &mut self.map[index.2][index.1][index.0] } } diff --git a/test/alu-export.nbt b/test/alu-export.nbt index 87de2df..e20633d 100644 Binary files a/test/alu-export.nbt and b/test/alu-export.nbt differ diff --git a/test/and-gate.nbt b/test/and-gate.nbt index bfe46df..bf43164 100644 Binary files a/test/and-gate.nbt and b/test/and-gate.nbt differ