From acdda819a55531b932b3b84c3f9ca204cee455d6 Mon Sep 17 00:00:00 2001 From: Ethan Barry Date: Mon, 25 Nov 2024 11:13:42 -0600 Subject: [PATCH 1/5] Clean up code & rm debug output. --- src/gamemanager/legal_moves.rs | 199 ++------------------- src/gamemanager/mod.rs | 1 - src/gamemanager/pseudolegal_moves/kings.rs | 5 - src/main.rs | 5 +- src/movetable.rs | 3 - 5 files changed, 16 insertions(+), 197 deletions(-) diff --git a/src/gamemanager/legal_moves.rs b/src/gamemanager/legal_moves.rs index 5ece668..c1f3e35 100644 --- a/src/gamemanager/legal_moves.rs +++ b/src/gamemanager/legal_moves.rs @@ -30,7 +30,7 @@ impl GameManager { // Based on color, union all the friendly and enemy pieces, and determine // which squares the enemy is attacking. We'll need that info for checking // whether the king is in danger. - let (friendly_pieces, enemy_pieces, enemy_attacked) = match color { + let (friendly_pieces, enemy_pieces, currently_attacked) = match color { Color::Black => { let friendly_pieces = self.bitboard.pawns_black | self.bitboard.rooks_black @@ -105,9 +105,6 @@ impl GameManager { /* ************************************************************************************* */ for mv in &pslm { - if mv.2 == Square::F2 { - dbg!(&mv.3); - } debug_assert!(mv.1 != mv.2); // Create a new GameManager here. let mut modified_gm = { @@ -140,191 +137,25 @@ impl GameManager { } // TODO: Test mask and king intersection. - } - - // A pseudolegal move may be illegal iff it puts the king in check. - // For each possible move, check legality. - 'pslms: for (piecetype, from, to, movetype) in pslm { - // Test for king safety against each enemy bitboard, - // by grabbing all the moves a super piece can make - // from the king's square. + let enemy_attacked = modified_gm.attacked_by(match color { + Color::Black => Color::White, + Color::White => Color::Black, + }); - // Create a new GameManager here. - let mut modified_gm = { - match color { - Color::Black => { - self.black_match_block(piecetype.clone(), movetype.clone(), from, to) + match color { + Color::Black => { + if modified_gm.bitboard.king_black & enemy_attacked == 0 { + // Good move; push it. + legal_moves.push((mv.0.clone(), mv.1, mv.2, mv.3.clone(), modified_gm)); } - Color::White => { - self.white_match_block(piecetype.clone(), movetype.clone(), from, to) - } - } - }; - - // Increment the fullmove clock every black move. - if color == Color::Black { - modified_gm.fullmoves += 1; - modified_gm.white_to_move = true; - } else { - modified_gm.white_to_move = false; - } - - // Increment the halfmove counter every quiet/non-pawn move. - use MoveType::*; - match movetype { - QuietMove | KingCastle | QueenCastle => { - modified_gm.halfmoves += 1; - modified_gm.en_passant_target = String::new(); // Made a quiet move instead of EPCapture. } - EPCapture => { - modified_gm.halfmoves = 0; - modified_gm.en_passant_target = String::new() + Color::White => { + if modified_gm.bitboard.king_white & enemy_attacked == 0 { + // Good move; push it. + legal_moves.push((mv.0.clone(), mv.1, mv.2, mv.3.clone(), modified_gm)); + } } - _ => modified_gm.halfmoves = 0, - } - - // Continue to check against each bitboard with moves from the corresponding piece type. - let psl_pawn_moves: Vec<(PieceType, Square, Square, MoveType)> = - pseudolegal_moves::pseudolegal_pawn_moves( - // Should be the opposite of the current color. - if color == Color::Black { - Color::White - } else { - Color::Black - }, - &MOVETABLE, - GameManager::powers_of_two(match color { - Color::Black => modified_gm.bitboard.pawns_white, - Color::White => modified_gm.bitboard.pawns_black, - }), - friendly_pieces, - enemy_pieces, - &modified_gm.en_passant_target, - ); - if psl_pawn_moves.iter().fold(0, |acc, mv| acc | mv.2.to_u64()) & friendly_king != 0 { - // Position is bad. - continue 'pslms; - } - - let psl_rook_moves: Vec<(PieceType, Square, Square, MoveType)> = - pseudolegal_moves::pseudolegal_rook_moves( - if color == Color::Black { - Color::White - } else { - Color::Black - }, - &MOVETABLE, - GameManager::powers_of_two(match color { - Color::Black => modified_gm.bitboard.rooks_white, - Color::White => modified_gm.bitboard.rooks_black, - }), - friendly_pieces, - enemy_pieces, - ); - if psl_rook_moves.iter().fold(0, |acc, mv| acc | mv.2.to_u64()) & friendly_king != 0 { - // Position is bad. - continue 'pslms; } - - let psl_knight_moves: Vec<(PieceType, Square, Square, MoveType)> = - pseudolegal_moves::pseudolegal_knight_moves( - if color == Color::Black { - Color::White - } else { - Color::Black - }, - &MOVETABLE, - GameManager::powers_of_two(match color { - Color::Black => modified_gm.bitboard.knights_white, - Color::White => modified_gm.bitboard.knights_black, - }), - friendly_pieces, - enemy_pieces, - ); - if psl_knight_moves - .iter() - .fold(0, |acc, mv| acc | mv.2.to_u64()) - & friendly_king - != 0 - { - // Position is bad. - continue 'pslms; - } - - let psl_bishop_moves: Vec<(PieceType, Square, Square, MoveType)> = - pseudolegal_moves::pseudolegal_bishop_moves( - if color == Color::Black { - Color::White - } else { - Color::Black - }, - &MOVETABLE, - GameManager::powers_of_two(match color { - Color::Black => modified_gm.bitboard.bishops_white, - Color::White => modified_gm.bitboard.bishops_black, - }), - friendly_pieces, - enemy_pieces, - ); - if psl_bishop_moves - .iter() - .fold(0, |acc, mv| acc | mv.2.to_u64()) - & friendly_king - != 0 - { - // Position is bad. - continue 'pslms; - } - - let psl_queen_moves: Vec<(PieceType, Square, Square, MoveType)> = - pseudolegal_moves::pseudolegal_queen_moves( - if color == Color::Black { - Color::White - } else { - Color::Black - }, - &MOVETABLE, - GameManager::powers_of_two(match color { - Color::Black => modified_gm.bitboard.queens_white, - Color::White => modified_gm.bitboard.queens_black, - }), - friendly_pieces, - enemy_pieces, - ); - if psl_queen_moves - .iter() - .fold(0, |acc, mv| acc | mv.2.to_u64()) - & friendly_king - != 0 - { - // Position is bad. - continue 'pslms; - } - - let psl_king_moves: Vec<(PieceType, Square, Square, MoveType)> = - pseudolegal_moves::pseudolegal_king_moves( - if color == Color::Black { - Color::White - } else { - Color::Black - }, - &MOVETABLE, - GameManager::powers_of_two(match color { - Color::Black => modified_gm.bitboard.king_white, - Color::White => modified_gm.bitboard.king_black, - }), - friendly_pieces, - friendly_rooks, - enemy_pieces, - modified_gm.castling_rights, - ); - if psl_king_moves.iter().fold(0, |acc, mv| acc | mv.2.to_u64()) & friendly_king != 0 { - // Position is bad. - continue 'pslms; - } - - // Passed all the checks against every enemy piece. King wasn't under threat after all. - legal_moves.push((piecetype, from, to, movetype, modified_gm)); } legal_moves diff --git a/src/gamemanager/mod.rs b/src/gamemanager/mod.rs index 75aa03e..f225248 100644 --- a/src/gamemanager/mod.rs +++ b/src/gamemanager/mod.rs @@ -206,7 +206,6 @@ mod test { game_manager.fullmoves, ); - dbg!(&moves); assert_eq!( moves.iter().count(), 20 /* 20 valid moves at start of game. */ diff --git a/src/gamemanager/pseudolegal_moves/kings.rs b/src/gamemanager/pseudolegal_moves/kings.rs index 61b4a0e..da96346 100644 --- a/src/gamemanager/pseudolegal_moves/kings.rs +++ b/src/gamemanager/pseudolegal_moves/kings.rs @@ -119,7 +119,6 @@ pub fn pseudolegal_king_moves( )); } } - // TODO, BUG: Why is the White match arm so much different from the Black one? Color::White => { for king in king_locations { for r in movetable.get_moves(Color::White, PieceType::King, king) { @@ -255,8 +254,6 @@ mod tests { .cloned(), ); - dbg!(&pslnm); - assert!(pslnm.iter().all(|m| moves.contains(&m.2.to_u64()))); assert_eq!(pslnm.len(), moves.len()) } @@ -289,8 +286,6 @@ mod tests { .cloned(), ); - dbg!(&pslnm); - assert!(pslnm.iter().all(|m| moves.contains(&m.2.to_u64()))); assert_eq!(pslnm.len(), moves.len()) } diff --git a/src/main.rs b/src/main.rs index 16ac65b..11a355d 100644 --- a/src/main.rs +++ b/src/main.rs @@ -1,10 +1,8 @@ #![allow(dead_code)] use std::sync::LazyLock; -use crate::types::{Color, PieceType}; use gamemanager::GameManager; use movetable::MoveTable; -use types::Square; mod bitboard; mod gamemanager; @@ -33,7 +31,7 @@ fn main() { // ) // ); } - + ucimanager::communicate(); } @@ -73,7 +71,6 @@ mod test { let game = GameManager::from_fen_string(fen); let generated_fen = game.to_fen_string(); - dbg!(&fen, &generated_fen); assert_eq!(fen, generated_fen); } } diff --git a/src/movetable.rs b/src/movetable.rs index ad4e512..bb51e50 100644 --- a/src/movetable.rs +++ b/src/movetable.rs @@ -578,8 +578,6 @@ mod test { let table = MoveTable::default(); let rays = table.get_moves(Color::Black, PieceType::Knight, Square::B8.to_u64()); - dbg!(&rays); - let mut pslm: HashSet = HashSet::new(); pslm.insert(0x800000000000); pslm.insert(0x200000000000); @@ -669,7 +667,6 @@ mod test { let all_are_members = rays.iter().all(|r| r.iter().all(|m| pslm.contains(m))); let only_seven = rays.iter().fold(0, |acc, r| acc + r.iter().count()); - dbg!(&rays); assert!(all_are_members); assert_eq!(only_seven, 7); } From 35603f49eb5937c81a087b31dc8cfc3dd4750727 Mon Sep 17 00:00:00 2001 From: Ethan Barry Date: Mon, 25 Nov 2024 15:07:36 -0600 Subject: [PATCH 2/5] Add dashmap. --- Cargo.lock | 72 ++++++++++++++++++++++++++++++++++++++++++++++++ Cargo.toml | 1 + src/movetable.rs | 4 +-- 3 files changed, 75 insertions(+), 2 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 5657680..1b5b4f4 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -32,6 +32,12 @@ version = "1.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ace50bade8e6234aa140d9a2f552bbee1db4d353f69b8217bc503490fc1a9f26" +[[package]] +name = "bitflags" +version = "2.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b048fb63fd8b5923fc5aa7b340d8e156aec7ec02f0c78fa8a6ddc2613f6f71de" + [[package]] name = "block-buffer" version = "0.10.4" @@ -126,6 +132,21 @@ dependencies = [ "typenum", ] +[[package]] +name = "dashmap" +version = "6.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5041cc499144891f3790297212f32a74fb938e5136a14943f338ef9e0ae276cf" +dependencies = [ + "cfg-if", + "crossbeam-utils", + "hashbrown", + "lock_api", + "once_cell", + "parking_lot_core", + "rayon", +] + [[package]] name = "digest" version = "0.10.7" @@ -152,6 +173,12 @@ dependencies = [ "version_check", ] +[[package]] +name = "hashbrown" +version = "0.14.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e5274423e17b7c9fc20b6e7e208532f9b19825d82dfd615708b70edd83df41f1" + [[package]] name = "iana-time-zone" version = "0.1.61" @@ -190,6 +217,16 @@ version = "0.2.164" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "433bfe06b8c75da9b2e3fbea6e5329ff87748f0b144ef75306e674c3f6f7c13f" +[[package]] +name = "lock_api" +version = "0.4.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "07af8b9cdd281b7915f413fa73f29ebd5d55d0d3f0155584dade1ff18cea1b17" +dependencies = [ + "autocfg", + "scopeguard", +] + [[package]] name = "log" version = "0.4.22" @@ -217,6 +254,19 @@ version = "1.20.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "1261fe7e33c73b354eab43b1273a57c8f967d0391e80353e51f764ac02cf6775" +[[package]] +name = "parking_lot_core" +version = "0.9.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e401f977ab385c9e4e3ab30627d6f26d00e2c73eef317493c4ec6d468726cf8" +dependencies = [ + "cfg-if", + "libc", + "redox_syscall", + "smallvec", + "windows-targets", +] + [[package]] name = "pest" version = "2.7.14" @@ -300,6 +350,15 @@ dependencies = [ "crossbeam-utils", ] +[[package]] +name = "redox_syscall" +version = "0.5.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9b6dfecf2c74bce2466cabf93f6664d6998a69eb21e39f4207930065b27b771f" +dependencies = [ + "bitflags", +] + [[package]] name = "regex" version = "1.11.1" @@ -329,6 +388,12 @@ version = "0.8.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2b15c43186be67a4fd63bee50d0303afffcef381492ebe2c5d87f324e1b8815c" +[[package]] +name = "scopeguard" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49" + [[package]] name = "sha2" version = "0.10.8" @@ -346,10 +411,17 @@ version = "1.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "0fda2ff0d084019ba4d7c6f371c95d8fd75ce3524c3cb8fb653a3023f6323e64" +[[package]] +name = "smallvec" +version = "1.13.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3c5e1a9a646d36c3599cd173a41282daf47c44583ad367b8e6837255952e5c67" + [[package]] name = "swordfish" version = "0.0.0" dependencies = [ + "dashmap", "rayon", "regex", "vampirc-uci", diff --git a/Cargo.toml b/Cargo.toml index 3d31cc0..2158906 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -4,6 +4,7 @@ version = "0.0.0" edition = "2021" [dependencies] +dashmap = { version = "6.1.0", features = ["rayon"] } rayon = "1.10.0" regex = "1.10.6" vampirc-uci = "0.11" diff --git a/src/movetable.rs b/src/movetable.rs index bb51e50..e39d047 100644 --- a/src/movetable.rs +++ b/src/movetable.rs @@ -1,5 +1,5 @@ use crate::types::{Color, PieceType}; -use std::collections::HashMap; +use dashmap::DashMap; /// A HashMap of [`(Color, PieceType, u64)`] indexing [`Vec>`] where /// the index integer is a position on the board (must be a power of two) and @@ -7,7 +7,7 @@ use std::collections::HashMap; /// can move in is a separate list. This facilitates move legality checking, /// because sliding pieces simply start at the head of the list and work out. pub struct MoveTable { - table: HashMap<(Color, PieceType, u64), Vec>>, + table: DashMap<(Color, PieceType, u64), Vec>>, } impl Default for MoveTable { From 69e9176e4906b39a5400cdf24b5163b2d3d8d3b1 Mon Sep 17 00:00:00 2001 From: Ethan Barry Date: Mon, 25 Nov 2024 15:11:39 -0600 Subject: [PATCH 3/5] Move MoveTable to movetable/mod.rs --- src/{movetable.rs => movetable/mod.rs} | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) rename src/{movetable.rs => movetable/mod.rs} (99%) diff --git a/src/movetable.rs b/src/movetable/mod.rs similarity index 99% rename from src/movetable.rs rename to src/movetable/mod.rs index e39d047..7a24ffa 100644 --- a/src/movetable.rs +++ b/src/movetable/mod.rs @@ -13,7 +13,7 @@ pub struct MoveTable { impl Default for MoveTable { /// Generates a `MoveTable` containing the possible moves for each piece type at each square fn default() -> Self { - let mut table: HashMap<(Color, PieceType, u64), Vec>> = HashMap::new(); + let table: DashMap<(Color, PieceType, u64), Vec>> = DashMap::new(); let mut shift = 0x8000000000000000; // Piece in the top left corner. for y in 0..8_usize { From 9dd6cb53c4ce95155bb6ed0165bc4d9d0cae612e Mon Sep 17 00:00:00 2001 From: Ethan Barry Date: Mon, 25 Nov 2024 15:16:51 -0600 Subject: [PATCH 4/5] Add NoArc to codebase. This adds a sound (but memory-leaky) abstraction around an unsafe dereference, which lets us access our movetable without reference counting or atomic operations or locks or anything! Yay! --- src/main.rs | 4 +++- src/movetable/mod.rs | 2 ++ src/movetable/noarc.rs | 24 ++++++++++++++++++++++++ 3 files changed, 29 insertions(+), 1 deletion(-) create mode 100644 src/movetable/noarc.rs diff --git a/src/main.rs b/src/main.rs index c315fdd..94d6512 100644 --- a/src/main.rs +++ b/src/main.rs @@ -2,7 +2,7 @@ use std::sync::LazyLock; use gamemanager::GameManager; -use movetable::MoveTable; +use movetable::{noarc, MoveTable}; mod bitboard; mod gamemanager; @@ -13,6 +13,8 @@ mod ucimanager; pub static MOVETABLE: LazyLock = std::sync::LazyLock::new(MoveTable::default); fn main() { + let movetable = noarc::NoArc::new(MoveTable::default()); + let gm = GameManager::from_fen_string("rnbq1k1r/pp1Pbppp/2p5/8/2B5/8/PPP1NnPP/RNBQK2R w KQ - 1 8"); let mvlst = gm.legal_moves(); diff --git a/src/movetable/mod.rs b/src/movetable/mod.rs index 7a24ffa..744f296 100644 --- a/src/movetable/mod.rs +++ b/src/movetable/mod.rs @@ -1,6 +1,8 @@ use crate::types::{Color, PieceType}; use dashmap::DashMap; +pub mod noarc; + /// A HashMap of [`(Color, PieceType, u64)`] indexing [`Vec>`] where /// the index integer is a position on the board (must be a power of two) and /// the list of lists is a list of rays---that is, each direction the object diff --git a/src/movetable/noarc.rs b/src/movetable/noarc.rs new file mode 100644 index 0000000..f7a46b0 --- /dev/null +++ b/src/movetable/noarc.rs @@ -0,0 +1,24 @@ +use std::ops::Deref; + +#[derive(Clone)] +pub struct NoArc { + ptr: *mut T, +} + +impl NoArc { + pub fn new(val: T) -> Self { + Self { + ptr: Box::leak(Box::new(val)), + } + } +} + +unsafe impl Sync for NoArc {} +unsafe impl Send for NoArc {} + +impl Deref for NoArc { + type Target = T; + fn deref(&self) -> &Self::Target { + unsafe { self.ptr.as_ref().unwrap() } + } +} From 2a7943c6975b45062319ff6734582cba3cc9b849 Mon Sep 17 00:00:00 2001 From: Ethan Barry Date: Mon, 25 Nov 2024 17:07:45 -0600 Subject: [PATCH 5/5] Everything changed. This overhauls the way MoveTable is used by boxing it in a new struct NoArc. NoArc is a thread-safe way to read a block of memory. Essentially every file in the engine had to change, because essentially every file used GameManager or MoveTable in some way. It shouldn't be too horrible to sort through, though! --- src/gamemanager/legal_moves.rs | 170 +++++++------------ src/gamemanager/legal_moves/perft.rs | 26 +-- src/gamemanager/mod.rs | 27 ++- src/gamemanager/pseudolegal_moves/bishops.rs | 15 +- src/gamemanager/pseudolegal_moves/kings.rs | 19 ++- src/gamemanager/pseudolegal_moves/knights.rs | 10 +- src/gamemanager/pseudolegal_moves/mod.rs | 28 +-- src/gamemanager/pseudolegal_moves/pawns.rs | 7 +- src/gamemanager/pseudolegal_moves/queens.rs | 7 +- src/gamemanager/pseudolegal_moves/rooks.rs | 10 +- src/main.rs | 29 +--- src/movetable/noarc.rs | 2 +- 12 files changed, 148 insertions(+), 202 deletions(-) diff --git a/src/gamemanager/legal_moves.rs b/src/gamemanager/legal_moves.rs index c1f3e35..a7ac53b 100644 --- a/src/gamemanager/legal_moves.rs +++ b/src/gamemanager/legal_moves.rs @@ -4,7 +4,6 @@ use crate::{ bitboard::*, gamemanager::*, types::{CastlingRights, Color, MoveType, PieceType, Square}, - MOVETABLE, }; pub mod perft; @@ -14,7 +13,10 @@ impl GameManager { /// This results in a list of possible GameManagers that could result from /// the possible moves, which are also returned. Each can be evaluated for /// strengths and weaknesses. - pub fn legal_moves(&self) -> Vec<(PieceType, Square, Square, MoveType, GameManager)> { + pub fn legal_moves( + &self, + tbl: &NoArc, + ) -> Vec<(PieceType, Square, Square, MoveType, GameManager)> { /* ************************************************************************************* */ /* WARNING: THIS FUNCTION WILL ERROR SILENTLY IF ANY COLOR-DEPENDENT LOGIC IS USED HERE. */ /* ALL LOGIC IN THIS FUNCTION MUST BE COLOR-AGNOSTIC. */ @@ -27,69 +29,15 @@ impl GameManager { let mut legal_moves: Vec<(PieceType, Square, Square, MoveType, GameManager)> = vec![]; - // Based on color, union all the friendly and enemy pieces, and determine - // which squares the enemy is attacking. We'll need that info for checking - // whether the king is in danger. - let (friendly_pieces, enemy_pieces, currently_attacked) = match color { - Color::Black => { - let friendly_pieces = self.bitboard.pawns_black - | self.bitboard.rooks_black - | self.bitboard.knights_black - | self.bitboard.bishops_black - | self.bitboard.queens_black - | self.bitboard.king_black; - let enemy_pieces = self.bitboard.pawns_white - | self.bitboard.rooks_white - | self.bitboard.knights_white - | self.bitboard.bishops_white - | self.bitboard.queens_white - | self.bitboard.king_white; - - let enemy_attacked = self.attacked_by(Color::White); - - (friendly_pieces, enemy_pieces, enemy_attacked) - } - Color::White => { - let friendly_pieces = self.bitboard.pawns_white - | self.bitboard.rooks_white - | self.bitboard.knights_white - | self.bitboard.bishops_white - | self.bitboard.queens_white - | self.bitboard.king_white; - let enemy_pieces = self.bitboard.pawns_black - | self.bitboard.rooks_black - | self.bitboard.knights_black - | self.bitboard.bishops_black - | self.bitboard.queens_black - | self.bitboard.king_black; - - let enemy_attacked = self.attacked_by(Color::Black); - - (friendly_pieces, enemy_pieces, enemy_attacked) - } - }; - - // Based on color, extract the friendly and enemy kings. - let (friendly_king, enemy_king) = match color { - Color::Black => (self.bitboard.king_black, self.bitboard.king_white), - Color::White => (self.bitboard.king_white, self.bitboard.king_black), - }; - - // Based on color, extract the friendly and enemy rooks. - let (friendly_rooks, enemy_rooks) = match color { - Color::Black => (self.bitboard.rooks_black, self.bitboard.rooks_white), - Color::White => (self.bitboard.rooks_white, self.bitboard.rooks_black), - }; - // First get all the pseudolegal moves. let pslm = pseudolegal_moves::pseudolegal_moves( color, self.bitboard, - &MOVETABLE, self.castling_rights, &self.en_passant_target, self.halfmoves, self.fullmoves, + &tbl, ); // ASSERT: We will never have Super moves in the pseudolegal moves vector. @@ -137,10 +85,13 @@ impl GameManager { } // TODO: Test mask and king intersection. - let enemy_attacked = modified_gm.attacked_by(match color { - Color::Black => Color::White, - Color::White => Color::Black, - }); + let enemy_attacked = modified_gm.attacked_by( + &tbl, + match color { + Color::Black => Color::White, + Color::White => Color::Black, + }, + ); match color { Color::Black => { @@ -191,7 +142,6 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, ..*self }, MoveType::Capture => { @@ -209,7 +159,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -239,7 +189,7 @@ impl GameManager { }, castling_rights: new_castling_rights, en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::Capture => { @@ -257,7 +207,7 @@ impl GameManager { }, castling_rights: new_castling_rights, en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -276,7 +226,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::QueenCastle => GameManager { @@ -288,7 +238,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::Capture => { @@ -305,7 +255,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -316,7 +266,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, _ => unreachable!("Kings will never make another type of move."), @@ -329,7 +279,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::Capture => { @@ -347,7 +297,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -363,7 +313,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::RPromotion => GameManager { @@ -374,7 +324,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::NPromotion => GameManager { @@ -385,7 +335,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::QPromotion => GameManager { @@ -396,7 +346,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, // On a promotion capture to X delete all enemy pieces @@ -416,7 +366,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -435,7 +385,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -454,7 +404,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -473,7 +423,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -503,7 +453,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -532,7 +482,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: String::from(target_coord), - movetable: &MOVETABLE, + ..*self } } @@ -543,7 +493,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::Capture => { @@ -560,7 +510,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -577,7 +527,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::Capture => { @@ -594,7 +544,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -636,7 +586,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::Capture => { @@ -654,7 +604,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -684,7 +634,7 @@ impl GameManager { }, castling_rights: new_castling_rights, en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::Capture => { @@ -702,7 +652,7 @@ impl GameManager { }, castling_rights: new_castling_rights, en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -721,7 +671,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::QueenCastle => GameManager { @@ -733,7 +683,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::Capture => { @@ -750,7 +700,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -761,7 +711,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, _ => unreachable!("Kings will never make another type of move."), @@ -774,7 +724,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::Capture => { @@ -792,7 +742,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -808,7 +758,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::RPromotion => GameManager { @@ -819,7 +769,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::NPromotion => GameManager { @@ -830,7 +780,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::QPromotion => GameManager { @@ -841,7 +791,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, // On a promotion capture to X delete all enemy pieces @@ -861,7 +811,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -880,7 +830,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -899,7 +849,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -918,7 +868,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -948,7 +898,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -977,7 +927,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: String::from(target_coord), - movetable: &MOVETABLE, + ..*self } } @@ -988,7 +938,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::Capture => { @@ -1005,7 +955,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -1022,7 +972,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self }, MoveType::Capture => { @@ -1039,7 +989,7 @@ impl GameManager { }, castling_rights: self.castling_rights.clone(), en_passant_target: self.en_passant_target.clone(), - movetable: &MOVETABLE, + ..*self } } @@ -1058,7 +1008,7 @@ mod tests { #[test] fn test_en_passant() { let gm = GameManager::black_match_block( - &GameManager::from_fen_string("6k1/5p2/4p3/2p1P3/1pP2P2/1P6/8/6K1 b - c3 0 1"), + &GameManager::from_fen_str("6k1/5p2/4p3/2p1P3/1pP2P2/1P6/8/6K1 b - c3 0 1"), crate::types::PieceType::Pawn, crate::types::MoveType::EPCapture, crate::types::Square::B4, diff --git a/src/gamemanager/legal_moves/perft.rs b/src/gamemanager/legal_moves/perft.rs index d2e08b3..fe3f871 100644 --- a/src/gamemanager/legal_moves/perft.rs +++ b/src/gamemanager/legal_moves/perft.rs @@ -1,23 +1,13 @@ -use crate::types::Move; - -use super::GameManager; +use super::{GameManager, MoveTable, NoArc}; use rayon::prelude::*; -pub fn perft(depth: u16, maxdepth: u16, mv: Move, gm: GameManager) { - if depth > maxdepth { - return; - } - let mvlst = gm.legal_moves(); - let count = mvlst.iter().count(); +pub fn perft(depth: u16, maxdepth: u16, gm: GameManager, tbl: &NoArc) -> u64 { if depth == maxdepth { - //let s = format!("{}{}: ", mv.1.to_str(), mv.2.to_str()).to_ascii_lowercase(); - //println!("{}", s); - //eprintln!("MOVE AT DEPTH {depth}"); - //println!("{s}{}", mvlst.iter().count()); + 1 + } else { + gm.legal_moves(tbl) + .into_par_iter() + .map(|mv| perft(depth + 1, maxdepth, mv.4, tbl)) + .sum::() } - mvlst - .into_par_iter() - .for_each(|(pc, from, to, mvtp, modgm)| { - perft(depth + 1, maxdepth, (pc, from, to, mvtp.clone()), modgm) - }); } diff --git a/src/gamemanager/mod.rs b/src/gamemanager/mod.rs index f225248..61e7521 100644 --- a/src/gamemanager/mod.rs +++ b/src/gamemanager/mod.rs @@ -1,11 +1,7 @@ -#![allow(dead_code, unused_variables, unused_mut)] -use std::sync::LazyLock; - use crate::{ bitboard, - movetable::MoveTable, + movetable::{noarc::NoArc, MoveTable}, types::{CastlingRecord, Color}, - MOVETABLE, }; use bitboard::BitBoard; use pseudolegal_moves::pseudolegal_moves; @@ -33,7 +29,6 @@ pub struct GameManager { * fullmove number - number of completed turns (increment when black moves) */ pub bitboard: BitBoard, - pub movetable: &'static LazyLock, pub white_to_move: bool, pub castling_rights: CastlingRecord, pub en_passant_target: String, @@ -42,11 +37,10 @@ pub struct GameManager { } impl Default for GameManager { - /// Constructs a new `GameManager`, set to Chess's starting position + /// Constructs a new `GameManager` set to startpos. fn default() -> Self { GameManager { bitboard: BitBoard::default(), - movetable: &MOVETABLE, white_to_move: true, castling_rights: CastlingRecord::default(), en_passant_target: String::new(), @@ -60,13 +54,12 @@ impl GameManager { /// A utility method for generating a new `GameManager` from a FEN string\ /// * `fen` - a `&str` representing a game's state in FEN /// * `returns` - a `GameManager` as generated from the FEN - pub fn from_fen_string(fen: &str) -> Self { + pub fn from_fen_str(fen: &str) -> Self { if Self::is_valid_fen(fen) { let tokens: Vec = fen.split_whitespace().map(str::to_string).collect(); GameManager { //board space validation implemented at higher level (is_valid_fen()) bitboard: BitBoard::from_fen_string(&tokens[0]), - movetable: &MOVETABLE, white_to_move: tokens[1] == "w", castling_rights: CastlingRecord::try_from(tokens[2].as_str()) .expect("We expect FEN strings to be well-formed."), @@ -151,15 +144,15 @@ impl GameManager { } /// Returns a bitmask of all the pieces attacked by the given color on this GameManager's state. - pub fn attacked_by(&self, color: Color) -> u64 { + pub fn attacked_by(&self, tbl: &NoArc, color: Color) -> u64 { let moves = pseudolegal_moves( color, self.bitboard, - &MOVETABLE, self.castling_rights, &self.en_passant_target, self.halfmoves, self.fullmoves, + tbl, ); moves @@ -172,7 +165,11 @@ impl GameManager { #[cfg(test)] mod test { use super::GameManager; - use crate::{gamemanager::pseudolegal_moves::*, types::Color}; + use crate::{ + gamemanager::pseudolegal_moves::*, + movetable::{noarc::NoArc, MoveTable}, + types::Color, + }; #[test] fn check_psl_moves_1() { @@ -180,11 +177,11 @@ mod test { let moves = pseudolegal_moves( Color::Black, game_manager.bitboard, - &game_manager.movetable, game_manager.castling_rights, &game_manager.en_passant_target, game_manager.halfmoves, game_manager.fullmoves, + &NoArc::new(MoveTable::default()), ); assert_eq!( @@ -199,11 +196,11 @@ mod test { let moves = pseudolegal_moves( Color::White, game_manager.bitboard, - &game_manager.movetable, game_manager.castling_rights, &game_manager.en_passant_target, game_manager.halfmoves, game_manager.fullmoves, + &NoArc::new(MoveTable::default()), ); assert_eq!( diff --git a/src/gamemanager/pseudolegal_moves/bishops.rs b/src/gamemanager/pseudolegal_moves/bishops.rs index d614306..ac38478 100644 --- a/src/gamemanager/pseudolegal_moves/bishops.rs +++ b/src/gamemanager/pseudolegal_moves/bishops.rs @@ -1,4 +1,7 @@ -use crate::{movetable::MoveTable, types::*}; +use crate::{ + movetable::{noarc::NoArc, MoveTable}, + types::*, +}; /// Returns all pseudolegal moves the knights can make from their positions. /// ## Inputs @@ -41,10 +44,10 @@ use crate::{movetable::MoveTable, types::*}; /// ``` pub fn pseudolegal_bishop_moves( color: Color, - movetable: &MoveTable, bishop_locations: Vec, friendly_pieces: u64, enemy_pieces: u64, + movetable: &NoArc, ) -> Vec<(PieceType, Square, Square, MoveType)> { let mut bishop_pseudo_legal_moves = Vec::new(); @@ -88,17 +91,21 @@ pub fn pseudolegal_bishop_moves( #[cfg(test)] mod test { - use crate::{gamemanager::pseudolegal_moves::bishops, movetable::MoveTable, types::*}; + use crate::{ + gamemanager::pseudolegal_moves::bishops, + movetable::{noarc::NoArc, MoveTable}, + types::*, + }; use std::collections::HashSet; #[test] fn check_bishop_pslm() { let pslnm = bishops::pseudolegal_bishop_moves( Color::Black, - &MoveTable::default(), vec![0x20000000_00000000], 0xFFAF5000_00000000, 0xFFFF, + &NoArc::new(MoveTable::default()), ); let moves: HashSet = HashSet::from_iter( vec![ diff --git a/src/gamemanager/pseudolegal_moves/kings.rs b/src/gamemanager/pseudolegal_moves/kings.rs index da96346..31fb191 100644 --- a/src/gamemanager/pseudolegal_moves/kings.rs +++ b/src/gamemanager/pseudolegal_moves/kings.rs @@ -1,4 +1,7 @@ -use crate::{movetable::MoveTable, types::*}; +use crate::{ + movetable::{noarc::NoArc, MoveTable}, + types::*, +}; /// Returns all pseudolegal moves the kings can make from their positions. /// ## Inputs @@ -43,12 +46,12 @@ use crate::{movetable::MoveTable, types::*}; /// ``` pub fn pseudolegal_king_moves( color: Color, - movetable: &MoveTable, king_locations: Vec, friendly_pieces: u64, friendly_rooks: u64, enemy_pieces: u64, castling_rights: CastlingRecord, + movetable: &NoArc, ) -> Vec { let mut king_pseudo_legal_moves = Vec::new(); @@ -190,7 +193,11 @@ pub fn pseudolegal_king_moves( #[cfg(test)] mod tests { - use crate::{gamemanager::pseudolegal_moves::kings, movetable::MoveTable, types::*}; + use crate::{ + gamemanager::pseudolegal_moves::kings, + movetable::{noarc::NoArc, MoveTable}, + types::*, + }; use std::collections::HashSet; use Square::*; @@ -198,7 +205,6 @@ mod tests { fn check_king_pslm() { let pslnm = kings::pseudolegal_king_moves( Color::Black, - &MoveTable::default(), vec![B5.to_u64()], 0, 0, @@ -207,6 +213,7 @@ mod tests { black: CastlingRights::Neither, white: CastlingRights::Neither, }, + &NoArc::new(MoveTable::default()), ); let moves: HashSet = HashSet::from_iter( vec![ @@ -230,7 +237,6 @@ mod tests { fn check_black_king_pslm_castling() { let pslnm = kings::pseudolegal_king_moves( Color::Black, - &MoveTable::default(), vec![E8.to_u64()], 0, Square::A8.to_u64() | Square::H8.to_u64(), @@ -239,6 +245,7 @@ mod tests { black: CastlingRights::Both, white: CastlingRights::Neither, }, + &NoArc::new(MoveTable::default()), ); let moves: HashSet = HashSet::from_iter( vec![ @@ -262,7 +269,6 @@ mod tests { fn check_white_king_pslm_castling() { let pslnm = kings::pseudolegal_king_moves( Color::White, - &MoveTable::default(), vec![E1.to_u64()], 0, Square::A1.to_u64() | Square::H1.to_u64(), @@ -271,6 +277,7 @@ mod tests { black: CastlingRights::Neither, white: CastlingRights::Both, }, + &NoArc::new(MoveTable::default()), ); let moves: HashSet = HashSet::from_iter( vec![ diff --git a/src/gamemanager/pseudolegal_moves/knights.rs b/src/gamemanager/pseudolegal_moves/knights.rs index 02e8f76..f96dd30 100644 --- a/src/gamemanager/pseudolegal_moves/knights.rs +++ b/src/gamemanager/pseudolegal_moves/knights.rs @@ -1,4 +1,7 @@ -use crate::{movetable::MoveTable, types::*}; +use crate::{ + movetable::{noarc::NoArc, MoveTable}, + types::*, +}; /// Returns all pseudolegal moves the knights can make from their positions. /// ## Inputs @@ -29,10 +32,10 @@ use crate::{movetable::MoveTable, types::*}; /// ``` pub fn pseudolegal_knight_moves( color: Color, - movetable: &MoveTable, knight_locations: Vec, friendly_pieces: u64, enemy_pieces: u64, + movetable: &NoArc, ) -> Vec { let mut knight_pseudo_legal_moves = Vec::new(); @@ -73,6 +76,7 @@ pub fn pseudolegal_knight_moves( #[cfg(test)] mod tests { use crate::gamemanager::pseudolegal_moves::knights; + use crate::movetable::noarc::NoArc; use crate::{movetable::MoveTable, types::*}; use std::collections::HashSet; @@ -80,10 +84,10 @@ mod tests { fn check_knight_pslm() { let pslnm = knights::pseudolegal_knight_moves( Color::Black, - &MoveTable::default(), vec![0x40000000_00000000], 0xFFFF0000_00000000, 0xFFFF, + &NoArc::new(MoveTable::default()), ); let moves: HashSet = HashSet::from_iter( vec![0x00008000_00000000, 0x00002000_00000000] diff --git a/src/gamemanager/pseudolegal_moves/mod.rs b/src/gamemanager/pseudolegal_moves/mod.rs index 955725d..1119ed4 100644 --- a/src/gamemanager/pseudolegal_moves/mod.rs +++ b/src/gamemanager/pseudolegal_moves/mod.rs @@ -13,7 +13,7 @@ pub mod rooks; use crate::{ bitboard::BitBoard, gamemanager::GameManager, - movetable::MoveTable, + movetable::{noarc::NoArc, MoveTable}, types::{CastlingRecord, Color, Move}, }; @@ -23,11 +23,11 @@ use crate::{ pub fn pseudolegal_moves( color: Color, bitboard: BitBoard, - movetable: &MoveTable, castling_rights: CastlingRecord, en_passant_target: &str, halfmoves: u32, fullmoves: u32, + movetable: &NoArc, ) -> Vec { let mut pseudolegal_moves: Vec = Vec::new(); @@ -65,58 +65,58 @@ pub fn pseudolegal_moves( let mut pawn_pseudo_legal_moves = pawns::pseudolegal_pawn_moves( color, - movetable, pawns, friendly_pieces, enemy_pieces, en_passant_target, + &movetable, ); pseudolegal_moves.append(&mut pawn_pseudo_legal_moves); let mut knight_pseudo_legal_moves = knights::pseudolegal_knight_moves( color, - movetable, knights, friendly_pieces, enemy_pieces, + &movetable, ); pseudolegal_moves.append(&mut knight_pseudo_legal_moves); let mut bishop_pseudo_legal_moves = bishops::pseudolegal_bishop_moves( color, - movetable, bishops, friendly_pieces, enemy_pieces, + &movetable, ); pseudolegal_moves.append(&mut bishop_pseudo_legal_moves); let mut rook_pseudo_legal_moves = rooks::pseudolegal_rook_moves( color, - movetable, rooks, friendly_pieces, enemy_pieces, + &movetable, ); pseudolegal_moves.append(&mut rook_pseudo_legal_moves); let mut queen_pseudo_legal_moves = queens::pseudolegal_queen_moves( color, - movetable, queens, friendly_pieces, enemy_pieces, + &movetable, ); pseudolegal_moves.append(&mut queen_pseudo_legal_moves); let mut king_pseudo_legal_moves = kings::pseudolegal_king_moves( color, - movetable, kings, friendly_pieces, bitboard.rooks_black, enemy_pieces, castling_rights, + &movetable, ); pseudolegal_moves.append(&mut king_pseudo_legal_moves); } @@ -145,58 +145,58 @@ pub fn pseudolegal_moves( let mut pawn_pseudo_legal_moves = pawns::pseudolegal_pawn_moves( color, - movetable, pawns, friendly_pieces, enemy_pieces, en_passant_target, + &movetable, ); pseudolegal_moves.append(&mut pawn_pseudo_legal_moves); let mut knight_pseudo_legal_moves = knights::pseudolegal_knight_moves( color, - movetable, knights, friendly_pieces, enemy_pieces, + &movetable, ); pseudolegal_moves.append(&mut knight_pseudo_legal_moves); let mut bishop_pseudo_legal_moves = bishops::pseudolegal_bishop_moves( color, - movetable, bishops, friendly_pieces, enemy_pieces, + &movetable, ); pseudolegal_moves.append(&mut bishop_pseudo_legal_moves); let mut rook_pseudo_legal_moves = rooks::pseudolegal_rook_moves( color, - movetable, rooks, friendly_pieces, enemy_pieces, + &movetable, ); pseudolegal_moves.append(&mut rook_pseudo_legal_moves); let mut queen_pseudo_legal_moves = queens::pseudolegal_queen_moves( color, - movetable, queens, friendly_pieces, enemy_pieces, + &movetable, ); pseudolegal_moves.append(&mut queen_pseudo_legal_moves); let mut king_pseudo_legal_moves = kings::pseudolegal_king_moves( color, - movetable, kings, friendly_pieces, bitboard.rooks_white, enemy_pieces, castling_rights, + &movetable, ); pseudolegal_moves.append(&mut king_pseudo_legal_moves); } diff --git a/src/gamemanager/pseudolegal_moves/pawns.rs b/src/gamemanager/pseudolegal_moves/pawns.rs index f2bcdf7..814c286 100644 --- a/src/gamemanager/pseudolegal_moves/pawns.rs +++ b/src/gamemanager/pseudolegal_moves/pawns.rs @@ -1,4 +1,7 @@ -use crate::{movetable::MoveTable, types::*}; +use crate::{ + movetable::{noarc::NoArc, MoveTable}, + types::*, +}; /// A method returning a list of pseudo-legal pawn moves playable according to /// the information encoded in this instance of GameManager @@ -11,11 +14,11 @@ use crate::{movetable::MoveTable, types::*}; /// the target `Square`, and the `MoveType`) pub fn pseudolegal_pawn_moves( color: Color, - movetable: &MoveTable, pawn_locations: Vec, friendly_pieces: u64, enemy_pieces: u64, en_passant_target: &str, + movetable: &NoArc, ) -> Vec { let mut pawn_pseudo_legal_moves = Vec::new(); diff --git a/src/gamemanager/pseudolegal_moves/queens.rs b/src/gamemanager/pseudolegal_moves/queens.rs index 3a44f67..9026501 100644 --- a/src/gamemanager/pseudolegal_moves/queens.rs +++ b/src/gamemanager/pseudolegal_moves/queens.rs @@ -1,11 +1,14 @@ -use crate::{movetable::MoveTable, types::*}; +use crate::{ + movetable::{noarc::NoArc, MoveTable}, + types::*, +}; pub fn pseudolegal_queen_moves( color: Color, - movetable: &MoveTable, queen_locations: Vec, friendly_pieces: u64, enemy_pieces: u64, + movetable: &NoArc, ) -> Vec { let mut queen_pseudo_legal_moves = Vec::new(); diff --git a/src/gamemanager/pseudolegal_moves/rooks.rs b/src/gamemanager/pseudolegal_moves/rooks.rs index e02f377..15ba3c2 100644 --- a/src/gamemanager/pseudolegal_moves/rooks.rs +++ b/src/gamemanager/pseudolegal_moves/rooks.rs @@ -1,4 +1,7 @@ -use crate::{movetable::MoveTable, types::*}; +use crate::{ + movetable::{noarc::NoArc, MoveTable}, + types::*, +}; /// Returns all pseudolegal moves the rooks can make from their positions. /// ## Inputs @@ -13,10 +16,10 @@ use crate::{movetable::MoveTable, types::*}; /// which expands to `(PieceType, Square, Square, MoveType)`. pub fn pseudolegal_rook_moves( color: Color, - movetable: &MoveTable, rook_locations: Vec, friendly_pieces: u64, enemy_pieces: u64, + movetable: &NoArc, ) -> Vec { let mut rook_pseudo_legal_moves = Vec::new(); @@ -87,6 +90,7 @@ pub fn pseudolegal_rook_moves( #[cfg(test)] mod tests { use crate::gamemanager::pseudolegal_moves::rooks; + use crate::movetable::noarc::NoArc; use crate::{movetable::MoveTable, types::*}; use std::collections::HashSet; @@ -96,10 +100,10 @@ mod tests { let pslm = rooks::pseudolegal_rook_moves( Color::Black, - &MoveTable::default(), vec![B5.to_u64()], 0, 0, + &NoArc::new(MoveTable::default()), ); let moves: HashSet = HashSet::from_iter( vec![ diff --git a/src/main.rs b/src/main.rs index 94d6512..a263703 100644 --- a/src/main.rs +++ b/src/main.rs @@ -1,7 +1,6 @@ #![allow(dead_code)] -use std::sync::LazyLock; -use gamemanager::GameManager; +use gamemanager::{legal_moves::perft::perft, GameManager}; use movetable::{noarc, MoveTable}; mod bitboard; @@ -10,29 +9,11 @@ mod movetable; mod types; mod ucimanager; -pub static MOVETABLE: LazyLock = std::sync::LazyLock::new(MoveTable::default); - fn main() { - let movetable = noarc::NoArc::new(MoveTable::default()); + let tbl = noarc::NoArc::new(MoveTable::default()); - let gm = - GameManager::from_fen_string("rnbq1k1r/pp1Pbppp/2p5/8/2B5/8/PPP1NnPP/RNBQK2R w KQ - 1 8"); - let mvlst = gm.legal_moves(); - for mv in &mvlst { - let s = format!("{}{}: ", mv.1.to_str(), mv.2.to_str()).to_ascii_lowercase(); - println!("{}{}", s, mv.4.legal_moves().iter().count()); - // println!( - // "{}: 1", - // format!( - // "{s}{}", - // mv.4.legal_moves() - // .iter() - // .map(|newmv| format!("{}{}", newmv.1.to_str(), newmv.2.to_str())) - // .fold(String::new(), |acc, s| acc.to_owned() + " " + &s) - // .to_ascii_lowercase() - // ) - // ); - } + let gm = GameManager::default(); + println!("Searched {} nodes.", perft(0, 6, gm, &tbl)); } #[cfg(test)] @@ -68,7 +49,7 @@ mod test { ]; for fen in tests { - let game = GameManager::from_fen_string(fen); + let game = GameManager::from_fen_str(fen); let generated_fen = game.to_fen_string(); assert_eq!(fen, generated_fen); diff --git a/src/movetable/noarc.rs b/src/movetable/noarc.rs index f7a46b0..ee7d88d 100644 --- a/src/movetable/noarc.rs +++ b/src/movetable/noarc.rs @@ -1,6 +1,6 @@ use std::ops::Deref; -#[derive(Clone)] +#[derive(Copy, Clone)] pub struct NoArc { ptr: *mut T, }